528 lines
309 KiB
JavaScript
528 lines
309 KiB
JavaScript
/*
|
|
* ATTENTION: An "eval-source-map" devtool has been used.
|
|
* This devtool is neither made for production nor for readable output files.
|
|
* It uses "eval()" calls to create a separate source file with attached SourceMaps in the browser devtools.
|
|
* If you are trying to read the output file, select a different devtool (https://webpack.js.org/configuration/devtool/)
|
|
* or disable the default devtool with "devtool: false".
|
|
* If you are looking for production-ready output files, see mode: "production" (https://webpack.js.org/configuration/mode/).
|
|
*/
|
|
(() => {
|
|
var exports = {};
|
|
exports.id = "app/api/courrier/emails/route";
|
|
exports.ids = ["app/api/courrier/emails/route"];
|
|
exports.modules = {
|
|
|
|
/***/ "(rsc)/./app/api/auth/[...nextauth]/route.ts":
|
|
/*!*********************************************!*\
|
|
!*** ./app/api/auth/[...nextauth]/route.ts ***!
|
|
\*********************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ GET: () => (/* binding */ handler),\n/* harmony export */ POST: () => (/* binding */ handler),\n/* harmony export */ authOptions: () => (/* binding */ authOptions)\n/* harmony export */ });\n/* harmony import */ var next_auth__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! next-auth */ \"(rsc)/./node_modules/next-auth/index.js\");\n/* harmony import */ var next_auth__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(next_auth__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var next_auth_providers_keycloak__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! next-auth/providers/keycloak */ \"(rsc)/./node_modules/next-auth/providers/keycloak.js\");\n/* harmony import */ var jwt_decode__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! jwt-decode */ \"(rsc)/./node_modules/jwt-decode/build/esm/index.js\");\n\n\n\nfunction getRequiredEnvVar(name) {\n const value = process.env[name];\n if (!value) {\n throw new Error(`Missing required environment variable: ${name}`);\n }\n return value;\n}\nasync function refreshAccessToken(token) {\n try {\n const response = await fetch(`${process.env.KEYCLOAK_ISSUER}/protocol/openid-connect/token`, {\n headers: {\n \"Content-Type\": \"application/x-www-form-urlencoded\"\n },\n body: new URLSearchParams({\n client_id: process.env.KEYCLOAK_CLIENT_ID,\n client_secret: process.env.KEYCLOAK_CLIENT_SECRET,\n grant_type: \"refresh_token\",\n refresh_token: token.refreshToken\n }),\n method: \"POST\"\n });\n const refreshedTokens = await response.json();\n if (!response.ok) {\n throw refreshedTokens;\n }\n return {\n ...token,\n accessToken: refreshedTokens.access_token,\n refreshToken: refreshedTokens.refresh_token ?? token.refreshToken,\n accessTokenExpires: Date.now() + refreshedTokens.expires_in * 1000\n };\n } catch (error) {\n console.error(\"Error refreshing access token:\", error);\n return {\n ...token,\n error: \"RefreshAccessTokenError\"\n };\n }\n}\nconst authOptions = {\n providers: [\n (0,next_auth_providers_keycloak__WEBPACK_IMPORTED_MODULE_1__[\"default\"])({\n clientId: getRequiredEnvVar(\"KEYCLOAK_CLIENT_ID\"),\n clientSecret: getRequiredEnvVar(\"KEYCLOAK_CLIENT_SECRET\"),\n issuer: getRequiredEnvVar(\"KEYCLOAK_ISSUER\"),\n authorization: {\n params: {\n scope: \"openid profile email roles\"\n }\n },\n profile (profile) {\n console.log('Keycloak profile callback:', {\n rawProfile: profile,\n rawRoles: profile.roles,\n realmAccess: profile.realm_access,\n groups: profile.groups\n });\n // Get roles from realm_access\n const roles = profile.realm_access?.roles || [];\n console.log('Profile callback raw roles:', roles);\n // Clean up roles by removing ROLE_ prefix and converting to lowercase\n const cleanRoles = roles.map((role)=>role.replace(/^ROLE_/, '').toLowerCase());\n console.log('Profile callback cleaned roles:', cleanRoles);\n return {\n id: profile.sub,\n name: profile.name ?? profile.preferred_username,\n email: profile.email,\n first_name: profile.given_name ?? '',\n last_name: profile.family_name ?? '',\n username: profile.preferred_username ?? profile.email?.split('@')[0] ?? '',\n role: cleanRoles\n };\n }\n })\n ],\n session: {\n strategy: \"jwt\",\n maxAge: 30 * 24 * 60 * 60\n },\n callbacks: {\n async jwt ({ token, account, profile }) {\n if (account && profile) {\n const keycloakProfile = profile;\n const roles = keycloakProfile.realm_access?.roles || [];\n const cleanRoles = roles.map((role)=>role.replace(/^ROLE_/, '').toLowerCase());\n token.accessToken = account.access_token ?? '';\n token.refreshToken = account.refresh_token ?? '';\n token.accessTokenExpires = account.expires_at ?? 0;\n token.sub = keycloakProfile.sub;\n token.role = cleanRoles;\n token.username = keycloakProfile.preferred_username ?? '';\n token.first_name = keycloakProfile.given_name ?? '';\n token.last_name = keycloakProfile.family_name ?? '';\n } else if (token.accessToken) {\n try {\n const decoded = (0,jwt_decode__WEBPACK_IMPORTED_MODULE_2__.jwtDecode)(token.accessToken);\n if (decoded.realm_access?.roles) {\n const roles = decoded.realm_access.roles;\n const cleanRoles = roles.map((role)=>role.replace(/^ROLE_/, '').toLowerCase());\n token.role = cleanRoles;\n }\n } catch (error) {\n console.error('Error decoding token:', error);\n }\n }\n if (Date.now() < token.accessTokenExpires * 1000) {\n return token;\n }\n return refreshAccessToken(token);\n },\n async session ({ session, token }) {\n if (token.error) {\n throw new Error(token.error);\n }\n const userRoles = Array.isArray(token.role) ? token.role : [];\n session.user = {\n id: token.sub ?? '',\n email: token.email ?? null,\n name: token.name ?? null,\n image: null,\n username: token.username ?? '',\n first_name: token.first_name ?? '',\n last_name: token.last_name ?? '',\n role: userRoles,\n nextcloudInitialized: false\n };\n session.accessToken = token.accessToken;\n return session;\n }\n },\n pages: {\n signIn: '/signin',\n error: '/signin'\n },\n debug: \"development\" === 'development'\n};\nconst handler = next_auth__WEBPACK_IMPORTED_MODULE_0___default()(authOptions);\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./app/api/auth/[...nextauth]/route.ts","mappings":"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","sources":["/home/alma/nextgen/Neah-mail/app/api/auth/[...nextauth]/route.ts"],"sourcesContent":["import NextAuth, { NextAuthOptions } from \"next-auth\";\nimport KeycloakProvider from \"next-auth/providers/keycloak\";\nimport { jwtDecode } from \"jwt-decode\";\n\ninterface KeycloakProfile {\n  sub: string;\n  email?: string;\n  name?: string;\n  roles?: string[];\n  preferred_username?: string;\n  given_name?: string;\n  family_name?: string;\n  realm_access?: {\n    roles: string[];\n  };\n}\n\ninterface DecodedToken {\n  realm_access?: {\n    roles: string[];\n  };\n  [key: string]: any;\n}\n\ndeclare module \"next-auth\" {\n  interface Session {\n    user: {\n      id: string;\n      name?: string | null;\n      email?: string | null;\n      image?: string | null;\n      username: string;\n      first_name: string;\n      last_name: string;\n      role: string[];\n      nextcloudInitialized?: boolean;\n    };\n    accessToken?: string;\n  }\n\n  interface JWT {\n    sub?: string;\n    accessToken?: string;\n    refreshToken?: string;\n    accessTokenExpires?: number;\n    role?: string[];\n    username?: string;\n    first_name?: string;\n    last_name?: string;\n    error?: string;\n    email?: string | null;\n    name?: string | null;\n  }\n}\n\nfunction getRequiredEnvVar(name: string): string {\n  const value = process.env[name];\n  if (!value) {\n    throw new Error(`Missing required environment variable: ${name}`);\n  }\n  return value;\n}\n\nasync function refreshAccessToken(token: JWT) {\n  try {\n    const response = await fetch(`${process.env.KEYCLOAK_ISSUER}/protocol/openid-connect/token`, {\n      headers: { \"Content-Type\": \"application/x-www-form-urlencoded\" },\n      body: new URLSearchParams({\n        client_id: process.env.KEYCLOAK_CLIENT_ID!,\n        client_secret: process.env.KEYCLOAK_CLIENT_SECRET!,\n        grant_type: \"refresh_token\",\n        refresh_token: token.refreshToken,\n      }),\n      method: \"POST\",\n    });\n\n    const refreshedTokens = await response.json();\n\n    if (!response.ok) {\n      throw refreshedTokens;\n    }\n\n    return {\n      ...token,\n      accessToken: refreshedTokens.access_token,\n      refreshToken: refreshedTokens.refresh_token ?? token.refreshToken,\n      accessTokenExpires: Date.now() + refreshedTokens.expires_in * 1000,\n    };\n  } catch (error) {\n    console.error(\"Error refreshing access token:\", error);\n    return {\n      ...token,\n      error: \"RefreshAccessTokenError\",\n    };\n  }\n}\n\nexport const authOptions: NextAuthOptions = {\n  providers: [\n    KeycloakProvider({\n      clientId: getRequiredEnvVar(\"KEYCLOAK_CLIENT_ID\"),\n      clientSecret: getRequiredEnvVar(\"KEYCLOAK_CLIENT_SECRET\"),\n      issuer: getRequiredEnvVar(\"KEYCLOAK_ISSUER\"),\n      authorization: {\n        params: {\n          scope: \"openid profile email roles\"\n        }\n      },\n      profile(profile) {\n        console.log('Keycloak profile callback:', {\n          rawProfile: profile,\n          rawRoles: profile.roles,\n          realmAccess: profile.realm_access,\n          groups: profile.groups\n        });\n\n        // Get roles from realm_access\n        const roles = profile.realm_access?.roles || [];\n        console.log('Profile callback raw roles:', roles);\n\n        // Clean up roles by removing ROLE_ prefix and converting to lowercase\n        const cleanRoles = roles.map((role: string) => \n          role.replace(/^ROLE_/, '').toLowerCase()\n        );\n\n        console.log('Profile callback cleaned roles:', cleanRoles);\n\n        return {\n          id: profile.sub,\n          name: profile.name ?? profile.preferred_username,\n          email: profile.email,\n          first_name: profile.given_name ?? '',\n          last_name: profile.family_name ?? '',\n          username: profile.preferred_username ?? profile.email?.split('@')[0] ?? '',\n          role: cleanRoles,\n        }\n      },\n    }),\n  ],\n  session: {\n    strategy: \"jwt\",\n    maxAge: 30 * 24 * 60 * 60, // 30 days\n  },\n  callbacks: {\n    async jwt({ token, account, profile }) {\n      if (account && profile) {\n        const keycloakProfile = profile as KeycloakProfile;\n        const roles = keycloakProfile.realm_access?.roles || [];\n        const cleanRoles = roles.map((role: string) => \n          role.replace(/^ROLE_/, '').toLowerCase()\n        );\n\n        token.accessToken = account.access_token ?? '';\n        token.refreshToken = account.refresh_token ?? '';\n        token.accessTokenExpires = account.expires_at ?? 0;\n        token.sub = keycloakProfile.sub;\n        token.role = cleanRoles;\n        token.username = keycloakProfile.preferred_username ?? '';\n        token.first_name = keycloakProfile.given_name ?? '';\n        token.last_name = keycloakProfile.family_name ?? '';\n      } else if (token.accessToken) {\n        try {\n          const decoded = jwtDecode<DecodedToken>(token.accessToken);\n          if (decoded.realm_access?.roles) {\n            const roles = decoded.realm_access.roles;\n            const cleanRoles = roles.map((role: string) => \n              role.replace(/^ROLE_/, '').toLowerCase()\n            );\n            token.role = cleanRoles;\n          }\n        } catch (error) {\n          console.error('Error decoding token:', error);\n        }\n      }\n\n      if (Date.now() < (token.accessTokenExpires as number) * 1000) {\n        return token;\n      }\n\n      return refreshAccessToken(token);\n    },\n    async session({ session, token }) {\n      if (token.error) {\n        throw new Error(token.error);\n      }\n\n      const userRoles = Array.isArray(token.role) ? token.role : [];\n      session.user = {\n        id: token.sub ?? '',\n        email: token.email ?? null,\n        name: token.name ?? null,\n        image: null,\n        username: token.username ?? '',\n        first_name: token.first_name ?? '',\n        last_name: token.last_name ?? '',\n        role: userRoles,\n        nextcloudInitialized: false,\n      };\n      session.accessToken = token.accessToken;\n\n      return session;\n    }\n  },\n  pages: {\n    signIn: '/signin',\n    error: '/signin',\n  },\n  debug: process.env.NODE_ENV === 'development',\n};\n\nconst handler = NextAuth(authOptions);\nexport { handler as GET, handler as POST };\n\ninterface JWT {\n  accessToken: string;\n  refreshToken: string;\n  accessTokenExpires: number;\n}\n\ninterface Profile {\n  sub?: string;\n  email?: string;\n  name?: string;\n  roles?: string[];\n}\n\n"],"names":["NextAuth","KeycloakProvider","jwtDecode","getRequiredEnvVar","name","value","process","env","Error","refreshAccessToken","token","response","fetch","KEYCLOAK_ISSUER","headers","body","URLSearchParams","client_id","KEYCLOAK_CLIENT_ID","client_secret","KEYCLOAK_CLIENT_SECRET","grant_type","refresh_token","refreshToken","method","refreshedTokens","json","ok","accessToken","access_token","accessTokenExpires","Date","now","expires_in","error","console","authOptions","providers","clientId","clientSecret","issuer","authorization","params","scope","profile","log","rawProfile","rawRoles","roles","realmAccess","realm_access","groups","cleanRoles","map","role","replace","toLowerCase","id","sub","preferred_username","email","first_name","given_name","last_name","family_name","username","split","session","strategy","maxAge","callbacks","jwt","account","keycloakProfile","expires_at","decoded","userRoles","Array","isArray","user","image","nextcloudInitialized","pages","signIn","debug","handler","GET","POST"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./app/api/auth/[...nextauth]/route.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./app/api/courrier/emails/route.ts":
|
|
/*!******************************************!*\
|
|
!*** ./app/api/courrier/emails/route.ts ***!
|
|
\******************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ GET: () => (/* binding */ GET)\n/* harmony export */ });\n/* harmony import */ var next_server__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! next/server */ \"(rsc)/./node_modules/next/dist/api/server.js\");\n/* harmony import */ var next_auth__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! next-auth */ \"(rsc)/./node_modules/next-auth/index.js\");\n/* harmony import */ var next_auth__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(next_auth__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var _app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! @/app/api/auth/[...nextauth]/route */ \"(rsc)/./app/api/auth/[...nextauth]/route.ts\");\n/* harmony import */ var _lib_services_email_service__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! @/lib/services/email-service */ \"(rsc)/./lib/services/email-service.ts\");\n/* harmony import */ var _lib_redis__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! @/lib/redis */ \"(rsc)/./lib/redis.ts\");\n\n\n\n\n\nasync function GET(request) {\n try {\n // Authenticate user\n const session = await (0,next_auth__WEBPACK_IMPORTED_MODULE_1__.getServerSession)(_app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_2__.authOptions);\n if (!session || !session.user?.id) {\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json({\n error: \"Not authenticated\"\n }, {\n status: 401\n });\n }\n // Extract query parameters\n const { searchParams } = new URL(request.url);\n const page = parseInt(searchParams.get(\"page\") || \"1\");\n const perPage = parseInt(searchParams.get(\"perPage\") || \"20\");\n const folder = searchParams.get(\"folder\") || \"INBOX\";\n const searchQuery = searchParams.get(\"search\") || \"\";\n const accountId = searchParams.get(\"accountId\") || \"\";\n const checkOnly = searchParams.get(\"checkOnly\") === \"true\";\n // Log exact parameters received by the API\n console.log(`[API/emails] Received request with: folder=${folder}, accountId=${accountId}, page=${page}, checkOnly=${checkOnly}`);\n // Parameter normalization\n // If folder contains an account prefix, extract it but DO NOT use it\n // Always prioritize the explicit accountId parameter\n let normalizedFolder = folder;\n let effectiveAccountId = accountId || 'default';\n if (folder.includes(':')) {\n const parts = folder.split(':');\n normalizedFolder = parts[1];\n console.log(`[API/emails] Folder has prefix, normalized to ${normalizedFolder}`);\n }\n console.log(`[API/emails] Using normalized parameters: folder=${normalizedFolder}, accountId=${effectiveAccountId}`);\n // Try to get from Redis cache first, but only if it's not a search query and not checkOnly\n if (!searchQuery && !checkOnly) {\n console.log(`[API/emails] Checking Redis cache for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}`);\n const cachedEmails = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_4__.getCachedEmailList)(session.user.id, effectiveAccountId, normalizedFolder, page, perPage);\n if (cachedEmails) {\n console.log(`[API/emails] Using Redis cached emails for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}`);\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json(cachedEmails);\n }\n }\n console.log(`[API/emails] Redis cache miss for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}, fetching emails from IMAP`);\n // Use the email service to fetch emails\n const emailsResult = await (0,_lib_services_email_service__WEBPACK_IMPORTED_MODULE_3__.getEmails)(session.user.id, normalizedFolder, page, perPage, effectiveAccountId, checkOnly);\n console.log(`[API/emails] Successfully fetched ${emailsResult.emails.length} emails from IMAP for account ${effectiveAccountId}`);\n // Return result\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json(emailsResult);\n } catch (error) {\n console.error(\"[API/emails] Error fetching emails:\", error);\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json({\n error: \"Failed to fetch emails\",\n message: error.message\n }, {\n status: 500\n });\n }\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./app/api/courrier/emails/route.ts","mappings":";;;;;;;;;;AAA2C;AACE;AACoB;AACR;AAKpC;AAEd,eAAeK,IAAIC,OAAgB;IACxC,IAAI;QACF,oBAAoB;QACpB,MAAMC,UAAU,MAAMN,2DAAgBA,CAACC,qEAAWA;QAClD,IAAI,CAACK,WAAW,CAACA,QAAQC,IAAI,EAAEC,IAAI;YACjC,OAAOT,qDAAYA,CAACU,IAAI,CACtB;gBAAEC,OAAO;YAAoB,GAC7B;gBAAEC,QAAQ;YAAI;QAElB;QAEA,2BAA2B;QAC3B,MAAM,EAAEC,YAAY,EAAE,GAAG,IAAIC,IAAIR,QAAQS,GAAG;QAC5C,MAAMC,OAAOC,SAASJ,aAAaK,GAAG,CAAC,WAAW;QAClD,MAAMC,UAAUF,SAASJ,aAAaK,GAAG,CAAC,cAAc;QACxD,MAAME,SAASP,aAAaK,GAAG,CAAC,aAAa;QAC7C,MAAMG,cAAcR,aAAaK,GAAG,CAAC,aAAa;QAClD,MAAMI,YAAYT,aAAaK,GAAG,CAAC,gBAAgB;QACnD,MAAMK,YAAYV,aAAaK,GAAG,CAAC,iBAAiB;QAEpD,2CAA2C;QAC3CM,QAAQC,GAAG,CAAC,CAAC,2CAA2C,EAAEL,OAAO,YAAY,EAAEE,UAAU,OAAO,EAAEN,KAAK,YAAY,EAAEO,WAAW;QAEhI,0BAA0B;QAC1B,qEAAqE;QACrE,qDAAqD;QACrD,IAAIG,mBAAmBN;QACvB,IAAIO,qBAAqBL,aAAa;QAEtC,IAAIF,OAAOQ,QAAQ,CAAC,MAAM;YACxB,MAAMC,QAAQT,OAAOU,KAAK,CAAC;YAC3BJ,mBAAmBG,KAAK,CAAC,EAAE;YAE3BL,QAAQC,GAAG,CAAC,CAAC,8CAA8C,EAAEC,kBAAkB;QACjF;QAEAF,QAAQC,GAAG,CAAC,CAAC,iDAAiD,EAAEC,iBAAiB,YAAY,EAAEC,oBAAoB;QAEnH,2FAA2F;QAC3F,IAAI,CAACN,eAAe,CAACE,WAAW;YAC9BC,QAAQC,GAAG,CAAC,CAAC,sCAAsC,EAAElB,QAAQC,IAAI,CAACC,EAAE,CAAC,CAAC,EAAEkB,mBAAmB,CAAC,EAAED,iBAAiB,CAAC,EAAEV,KAAK,CAAC,EAAEG,SAAS;YACnI,MAAMY,eAAe,MAAM3B,8DAAkBA,CAC3CG,QAAQC,IAAI,CAACC,EAAE,EACfkB,oBACAD,kBACAV,MACAG;YAEF,IAAIY,cAAc;gBAChBP,QAAQC,GAAG,CAAC,CAAC,2CAA2C,EAAElB,QAAQC,IAAI,CAACC,EAAE,CAAC,CAAC,EAAEkB,mBAAmB,CAAC,EAAED,iBAAiB,CAAC,EAAEV,KAAK,CAAC,EAAEG,SAAS;gBACxI,OAAOnB,qDAAYA,CAACU,IAAI,CAACqB;YAC3B;QACF;QAEAP,QAAQC,GAAG,CAAC,CAAC,kCAAkC,EAAElB,QAAQC,IAAI,CAACC,EAAE,CAAC,CAAC,EAAEkB,mBAAmB,CAAC,EAAED,iBAAiB,CAAC,EAAEV,KAAK,CAAC,EAAEG,QAAQ,2BAA2B,CAAC;QAE1J,wCAAwC;QACxC,MAAMa,eAAe,MAAM7B,sEAASA,CAClCI,QAAQC,IAAI,CAACC,EAAE,EACfiB,kBACAV,MACAG,SACAQ,oBACAJ;QAGFC,QAAQC,GAAG,CAAC,CAAC,kCAAkC,EAAEO,aAAaC,MAAM,CAACC,MAAM,CAAC,8BAA8B,EAAEP,oBAAoB;QAEhI,gBAAgB;QAChB,OAAO3B,qDAAYA,CAACU,IAAI,CAACsB;IAC3B,EAAE,OAAOrB,OAAY;QACnBa,QAAQb,KAAK,CAAC,uCAAuCA;QACrD,OAAOX,qDAAYA,CAACU,IAAI,CACtB;YAAEC,OAAO;YAA0BwB,SAASxB,MAAMwB,OAAO;QAAC,GAC1D;YAAEvB,QAAQ;QAAI;IAElB;AACF","sources":["/home/alma/nextgen/Neah-mail/app/api/courrier/emails/route.ts"],"sourcesContent":["import { NextResponse } from 'next/server';\nimport { getServerSession } from 'next-auth';\nimport { authOptions } from '@/app/api/auth/[...nextauth]/route';\nimport { getEmails } from '@/lib/services/email-service';\nimport { \n  getCachedEmailList, \n  cacheEmailList, \n  invalidateFolderCache \n} from '@/lib/redis';\n\nexport async function GET(request: Request) {\n  try {\n    // Authenticate user\n    const session = await getServerSession(authOptions);\n    if (!session || !session.user?.id) {\n      return NextResponse.json(\n        { error: \"Not authenticated\" },\n        { status: 401 }\n      );\n    }\n\n    // Extract query parameters\n    const { searchParams } = new URL(request.url);\n    const page = parseInt(searchParams.get(\"page\") || \"1\");\n    const perPage = parseInt(searchParams.get(\"perPage\") || \"20\");\n    const folder = searchParams.get(\"folder\") || \"INBOX\";\n    const searchQuery = searchParams.get(\"search\") || \"\";\n    const accountId = searchParams.get(\"accountId\") || \"\";\n    const checkOnly = searchParams.get(\"checkOnly\") === \"true\";\n    \n    // Log exact parameters received by the API\n    console.log(`[API/emails] Received request with: folder=${folder}, accountId=${accountId}, page=${page}, checkOnly=${checkOnly}`);\n    \n    // Parameter normalization\n    // If folder contains an account prefix, extract it but DO NOT use it\n    // Always prioritize the explicit accountId parameter\n    let normalizedFolder = folder;\n    let effectiveAccountId = accountId || 'default';\n    \n    if (folder.includes(':')) {\n      const parts = folder.split(':');\n      normalizedFolder = parts[1];\n      \n      console.log(`[API/emails] Folder has prefix, normalized to ${normalizedFolder}`);\n    }\n    \n    console.log(`[API/emails] Using normalized parameters: folder=${normalizedFolder}, accountId=${effectiveAccountId}`);\n    \n    // Try to get from Redis cache first, but only if it's not a search query and not checkOnly\n    if (!searchQuery && !checkOnly) {\n      console.log(`[API/emails] Checking Redis cache for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}`);\n      const cachedEmails = await getCachedEmailList(\n        session.user.id,\n        effectiveAccountId,\n        normalizedFolder,\n        page,\n        perPage\n      );\n      if (cachedEmails) {\n        console.log(`[API/emails] Using Redis cached emails for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}`);\n        return NextResponse.json(cachedEmails);\n      }\n    }\n\n    console.log(`[API/emails] Redis cache miss for ${session.user.id}:${effectiveAccountId}:${normalizedFolder}:${page}:${perPage}, fetching emails from IMAP`);\n\n    // Use the email service to fetch emails\n    const emailsResult = await getEmails(\n      session.user.id,\n      normalizedFolder,\n      page,\n      perPage,\n      effectiveAccountId,\n      checkOnly\n    );\n\n    console.log(`[API/emails] Successfully fetched ${emailsResult.emails.length} emails from IMAP for account ${effectiveAccountId}`);\n\n    // Return result\n    return NextResponse.json(emailsResult);\n  } catch (error: any) {\n    console.error(\"[API/emails] Error fetching emails:\", error);\n    return NextResponse.json(\n      { error: \"Failed to fetch emails\", message: error.message },\n      { status: 500 }\n    );\n  }\n} "],"names":["NextResponse","getServerSession","authOptions","getEmails","getCachedEmailList","GET","request","session","user","id","json","error","status","searchParams","URL","url","page","parseInt","get","perPage","folder","searchQuery","accountId","checkOnly","console","log","normalizedFolder","effectiveAccountId","includes","parts","split","cachedEmails","emailsResult","emails","length","message"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./app/api/courrier/emails/route.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/env.ts":
|
|
/*!********************!*\
|
|
!*** ./lib/env.ts ***!
|
|
\********************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ env: () => (/* binding */ env)\n/* harmony export */ });\n/* harmony import */ var zod__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! zod */ \"(rsc)/./node_modules/zod/lib/index.mjs\");\n\nconst envSchema = zod__WEBPACK_IMPORTED_MODULE_0__.z.object({\n NODE_ENV: zod__WEBPACK_IMPORTED_MODULE_0__.z.enum([\n \"development\",\n \"test\",\n \"production\"\n ]).default(\"development\"),\n DATABASE_URL: zod__WEBPACK_IMPORTED_MODULE_0__.z.string().url(),\n NEWSDB_URL: zod__WEBPACK_IMPORTED_MODULE_0__.z.string().regex(/^postgresql:\\/\\//, \"Must be a valid PostgreSQL URL\"),\n NEWS_API_URL: zod__WEBPACK_IMPORTED_MODULE_0__.z.string().url(),\n KEYCLOAK_CLIENT_ID: zod__WEBPACK_IMPORTED_MODULE_0__.z.string(),\n KEYCLOAK_CLIENT_SECRET: zod__WEBPACK_IMPORTED_MODULE_0__.z.string(),\n KEYCLOAK_REALM: zod__WEBPACK_IMPORTED_MODULE_0__.z.string(),\n KEYCLOAK_ISSUER: zod__WEBPACK_IMPORTED_MODULE_0__.z.string().url(),\n LEANTIME_TOKEN: zod__WEBPACK_IMPORTED_MODULE_0__.z.string(),\n LEANTIME_API_URL: zod__WEBPACK_IMPORTED_MODULE_0__.z.string().url(),\n ROCKET_CHAT_TOKEN: zod__WEBPACK_IMPORTED_MODULE_0__.z.string(),\n ROCKET_CHAT_USER_ID: zod__WEBPACK_IMPORTED_MODULE_0__.z.string()\n});\nconst env = envSchema.parse(process.env);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9saWIvZW52LnRzIiwibWFwcGluZ3MiOiI7Ozs7O0FBQXdCO0FBRXhCLE1BQU1DLFlBQVlELGtDQUFDQSxDQUFDRSxNQUFNLENBQUM7SUFDekJDLFVBQVVILGtDQUFDQSxDQUFDSSxJQUFJLENBQUM7UUFBQztRQUFlO1FBQVE7S0FBYSxFQUFFQyxPQUFPLENBQUM7SUFDaEVDLGNBQWNOLGtDQUFDQSxDQUFDTyxNQUFNLEdBQUdDLEdBQUc7SUFDNUJDLFlBQVlULGtDQUFDQSxDQUFDTyxNQUFNLEdBQUdHLEtBQUssQ0FBQyxvQkFBb0I7SUFDakRDLGNBQWNYLGtDQUFDQSxDQUFDTyxNQUFNLEdBQUdDLEdBQUc7SUFDNUJJLG9CQUFvQlosa0NBQUNBLENBQUNPLE1BQU07SUFDNUJNLHdCQUF3QmIsa0NBQUNBLENBQUNPLE1BQU07SUFDaENPLGdCQUFnQmQsa0NBQUNBLENBQUNPLE1BQU07SUFDeEJRLGlCQUFpQmYsa0NBQUNBLENBQUNPLE1BQU0sR0FBR0MsR0FBRztJQUMvQlEsZ0JBQWdCaEIsa0NBQUNBLENBQUNPLE1BQU07SUFDeEJVLGtCQUFrQmpCLGtDQUFDQSxDQUFDTyxNQUFNLEdBQUdDLEdBQUc7SUFDaENVLG1CQUFtQmxCLGtDQUFDQSxDQUFDTyxNQUFNO0lBQzNCWSxxQkFBcUJuQixrQ0FBQ0EsQ0FBQ08sTUFBTTtBQUMvQjtBQUVPLE1BQU1hLE1BQU1uQixVQUFVb0IsS0FBSyxDQUFDQyxRQUFRRixHQUFHLEVBQUUiLCJzb3VyY2VzIjpbIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbGliL2Vudi50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB6IH0gZnJvbSBcInpvZFwiO1xuXG5jb25zdCBlbnZTY2hlbWEgPSB6Lm9iamVjdCh7XG4gIE5PREVfRU5WOiB6LmVudW0oW1wiZGV2ZWxvcG1lbnRcIiwgXCJ0ZXN0XCIsIFwicHJvZHVjdGlvblwiXSkuZGVmYXVsdChcImRldmVsb3BtZW50XCIpLFxuICBEQVRBQkFTRV9VUkw6IHouc3RyaW5nKCkudXJsKCksXG4gIE5FV1NEQl9VUkw6IHouc3RyaW5nKCkucmVnZXgoL15wb3N0Z3Jlc3FsOlxcL1xcLy8sIFwiTXVzdCBiZSBhIHZhbGlkIFBvc3RncmVTUUwgVVJMXCIpLFxuICBORVdTX0FQSV9VUkw6IHouc3RyaW5nKCkudXJsKCksXG4gIEtFWUNMT0FLX0NMSUVOVF9JRDogei5zdHJpbmcoKSxcbiAgS0VZQ0xPQUtfQ0xJRU5UX1NFQ1JFVDogei5zdHJpbmcoKSxcbiAgS0VZQ0xPQUtfUkVBTE06IHouc3RyaW5nKCksXG4gIEtFWUNMT0FLX0lTU1VFUjogei5zdHJpbmcoKS51cmwoKSxcbiAgTEVBTlRJTUVfVE9LRU46IHouc3RyaW5nKCksXG4gIExFQU5USU1FX0FQSV9VUkw6IHouc3RyaW5nKCkudXJsKCksXG4gIFJPQ0tFVF9DSEFUX1RPS0VOOiB6LnN0cmluZygpLFxuICBST0NLRVRfQ0hBVF9VU0VSX0lEOiB6LnN0cmluZygpLFxufSk7XG5cbmV4cG9ydCBjb25zdCBlbnYgPSBlbnZTY2hlbWEucGFyc2UocHJvY2Vzcy5lbnYpOyAiXSwibmFtZXMiOlsieiIsImVudlNjaGVtYSIsIm9iamVjdCIsIk5PREVfRU5WIiwiZW51bSIsImRlZmF1bHQiLCJEQVRBQkFTRV9VUkwiLCJzdHJpbmciLCJ1cmwiLCJORVdTREJfVVJMIiwicmVnZXgiLCJORVdTX0FQSV9VUkwiLCJLRVlDTE9BS19DTElFTlRfSUQiLCJLRVlDTE9BS19DTElFTlRfU0VDUkVUIiwiS0VZQ0xPQUtfUkVBTE0iLCJLRVlDTE9BS19JU1NVRVIiLCJMRUFOVElNRV9UT0tFTiIsIkxFQU5USU1FX0FQSV9VUkwiLCJST0NLRVRfQ0hBVF9UT0tFTiIsIlJPQ0tFVF9DSEFUX1VTRVJfSUQiLCJlbnYiLCJwYXJzZSIsInByb2Nlc3MiXSwiaWdub3JlTGlzdCI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///(rsc)/./lib/env.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/prisma.ts":
|
|
/*!***********************!*\
|
|
!*** ./lib/prisma.ts ***!
|
|
\***********************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ prisma: () => (/* binding */ prisma)\n/* harmony export */ });\n/* harmony import */ var _prisma_client__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @prisma/client */ \"@prisma/client\");\n/* harmony import */ var _prisma_client__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_prisma_client__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _lib_env__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @/lib/env */ \"(rsc)/./lib/env.ts\");\n// front/lib/prisma.ts\n\n\nconst globalForPrisma = globalThis;\n// Main database client\nconst prisma = globalForPrisma.prisma || new _prisma_client__WEBPACK_IMPORTED_MODULE_0__.PrismaClient({\n datasources: {\n db: {\n url: _lib_env__WEBPACK_IMPORTED_MODULE_1__.env.DATABASE_URL\n }\n },\n log: [\n 'query'\n ]\n});\nif (true) {\n globalForPrisma.prisma = prisma;\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9saWIvcHJpc21hLnRzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQSxzQkFBc0I7QUFDdUI7QUFDZDtBQUUvQixNQUFNRSxrQkFBa0JDO0FBSXhCLHVCQUF1QjtBQUNoQixNQUFNQyxTQUNYRixnQkFBZ0JFLE1BQU0sSUFDdEIsSUFBSUosd0RBQVlBLENBQUM7SUFDZkssYUFBYTtRQUNYQyxJQUFJO1lBQ0ZDLEtBQUtOLHlDQUFHQSxDQUFDTyxZQUFZO1FBQ3ZCO0lBQ0Y7SUFDQUMsS0FBSztRQUFDO0tBQVE7QUFDaEIsR0FBRTtBQUVKLElBQUlDLElBQXFDLEVBQUU7SUFDekNSLGdCQUFnQkUsTUFBTSxHQUFHQTtBQUMzQiIsInNvdXJjZXMiOlsiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9saWIvcHJpc21hLnRzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIGZyb250L2xpYi9wcmlzbWEudHNcbmltcG9ydCB7IFByaXNtYUNsaWVudCB9IGZyb20gJ0BwcmlzbWEvY2xpZW50J1xuaW1wb3J0IHsgZW52IH0gZnJvbSAnQC9saWIvZW52J1xuXG5jb25zdCBnbG9iYWxGb3JQcmlzbWEgPSBnbG9iYWxUaGlzIGFzIHVua25vd24gYXMge1xuICBwcmlzbWE6IFByaXNtYUNsaWVudCB8IHVuZGVmaW5lZDtcbn1cblxuLy8gTWFpbiBkYXRhYmFzZSBjbGllbnRcbmV4cG9ydCBjb25zdCBwcmlzbWEgPVxuICBnbG9iYWxGb3JQcmlzbWEucHJpc21hIHx8XG4gIG5ldyBQcmlzbWFDbGllbnQoe1xuICAgIGRhdGFzb3VyY2VzOiB7XG4gICAgICBkYjoge1xuICAgICAgICB1cmw6IGVudi5EQVRBQkFTRV9VUkxcbiAgICAgIH1cbiAgICB9LFxuICAgIGxvZzogWydxdWVyeSddLFxuICB9KVxuXG5pZiAocHJvY2Vzcy5lbnYuTk9ERV9FTlYgIT09ICdwcm9kdWN0aW9uJykge1xuICBnbG9iYWxGb3JQcmlzbWEucHJpc21hID0gcHJpc21hO1xufVxuIl0sIm5hbWVzIjpbIlByaXNtYUNsaWVudCIsImVudiIsImdsb2JhbEZvclByaXNtYSIsImdsb2JhbFRoaXMiLCJwcmlzbWEiLCJkYXRhc291cmNlcyIsImRiIiwidXJsIiwiREFUQUJBU0VfVVJMIiwibG9nIiwicHJvY2VzcyJdLCJpZ25vcmVMaXN0IjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./lib/prisma.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/redis.ts":
|
|
/*!**********************!*\
|
|
!*** ./lib/redis.ts ***!
|
|
\**********************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ KEYS: () => (/* binding */ KEYS),\n/* harmony export */ TTL: () => (/* binding */ TTL),\n/* harmony export */ cacheEmailContent: () => (/* binding */ cacheEmailContent),\n/* harmony export */ cacheEmailCredentials: () => (/* binding */ cacheEmailCredentials),\n/* harmony export */ cacheEmailList: () => (/* binding */ cacheEmailList),\n/* harmony export */ cacheImapSession: () => (/* binding */ cacheImapSession),\n/* harmony export */ closeRedisConnection: () => (/* binding */ closeRedisConnection),\n/* harmony export */ decryptData: () => (/* binding */ decryptData),\n/* harmony export */ encryptData: () => (/* binding */ encryptData),\n/* harmony export */ getCachedEmailContent: () => (/* binding */ getCachedEmailContent),\n/* harmony export */ getCachedEmailCredentials: () => (/* binding */ getCachedEmailCredentials),\n/* harmony export */ getCachedEmailList: () => (/* binding */ getCachedEmailList),\n/* harmony export */ getCachedImapSession: () => (/* binding */ getCachedImapSession),\n/* harmony export */ getEmailCredentials: () => (/* binding */ getEmailCredentials),\n/* harmony export */ getRedisClient: () => (/* binding */ getRedisClient),\n/* harmony export */ getRedisStatus: () => (/* binding */ getRedisStatus),\n/* harmony export */ invalidateEmailContentCache: () => (/* binding */ invalidateEmailContentCache),\n/* harmony export */ invalidateFolderCache: () => (/* binding */ invalidateFolderCache),\n/* harmony export */ invalidateUserEmailCache: () => (/* binding */ invalidateUserEmailCache),\n/* harmony export */ warmupRedisCache: () => (/* binding */ warmupRedisCache)\n/* harmony export */ });\n/* harmony import */ var ioredis__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ioredis */ \"(rsc)/./node_modules/ioredis/built/index.js\");\n/* harmony import */ var ioredis__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(ioredis__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var crypto_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! crypto-js */ \"(rsc)/./node_modules/crypto-js/index.js\");\n/* harmony import */ var crypto_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(crypto_js__WEBPACK_IMPORTED_MODULE_1__);\n\n\n// Initialize Redis client\nlet redisClient = null;\nlet isConnecting = false;\nlet connectionAttempts = 0;\nconst MAX_RECONNECT_ATTEMPTS = 5;\n/**\n * Get a Redis client instance (singleton pattern) with improved connection management\n */ function getRedisClient() {\n if (redisClient && redisClient.status === 'ready') {\n return redisClient;\n }\n if (isConnecting) {\n // If we're already trying to connect, return the existing client\n // This prevents multiple simultaneous connection attempts\n if (redisClient) return redisClient;\n // This is a fallback in case we're connecting but don't have a client yet\n console.warn('Redis connection in progress, creating temporary client');\n }\n if (!redisClient) {\n isConnecting = true;\n connectionAttempts = 0;\n // Set Redis connection parameters from environment variables only\n const redisOptions = {\n host: process.env.REDIS_HOST,\n port: process.env.REDIS_PORT ? parseInt(process.env.REDIS_PORT) : undefined,\n password: process.env.REDIS_PASSWORD,\n retryStrategy: (times)=>{\n connectionAttempts = times;\n if (times > MAX_RECONNECT_ATTEMPTS) {\n console.error(`Redis connection failed after ${times} attempts, giving up`);\n return null; // Stop trying to reconnect\n }\n const delay = Math.min(times * 100, 5000);\n console.log(`Redis reconnect attempt ${times}, retrying in ${delay}ms`);\n return delay;\n },\n maxRetriesPerRequest: 5,\n enableOfflineQueue: true,\n connectTimeout: 10000,\n disconnectTimeout: 2000,\n keepAlive: 10000,\n keyPrefix: '' // No prefix to keep keys clean\n };\n console.log('Connecting to Redis using environment variables');\n redisClient = new (ioredis__WEBPACK_IMPORTED_MODULE_0___default())(redisOptions);\n redisClient.on('error', (err)=>{\n console.error('Redis connection error:', err);\n // Only set to null if we've exceeded max attempts\n if (connectionAttempts > MAX_RECONNECT_ATTEMPTS) {\n console.error('Redis connection failed permanently, will create new client on next request');\n redisClient = null;\n isConnecting = false;\n }\n });\n redisClient.on('connect', ()=>{\n console.log('Successfully connected to Redis');\n isConnecting = false;\n connectionAttempts = 0;\n });\n redisClient.on('reconnecting', ()=>{\n console.log('Reconnecting to Redis...');\n isConnecting = true;\n });\n redisClient.on('ready', ()=>{\n console.log('Redis connection warmed up');\n isConnecting = false;\n });\n redisClient.on('end', ()=>{\n console.log('Redis connection ended');\n // Don't set to null here - let the error handler decide\n });\n }\n return redisClient;\n}\n/**\n * Close Redis connection (useful for serverless environments)\n */ async function closeRedisConnection() {\n if (redisClient) {\n await redisClient.quit();\n redisClient = null;\n }\n}\n// Encryption key from environment variable or fallback\nconst getEncryptionKey = ()=>{\n return process.env.REDIS_ENCRYPTION_KEY || 'default-encryption-key-change-in-production';\n};\n/**\n * Encrypt sensitive data before storing in Redis\n */ function encryptData(data) {\n return crypto_js__WEBPACK_IMPORTED_MODULE_1___default().AES.encrypt(data, getEncryptionKey()).toString();\n}\n/**\n * Decrypt sensitive data retrieved from Redis\n */ function decryptData(encryptedData) {\n const bytes = crypto_js__WEBPACK_IMPORTED_MODULE_1___default().AES.decrypt(encryptedData, getEncryptionKey());\n return bytes.toString((crypto_js__WEBPACK_IMPORTED_MODULE_1___default().enc).Utf8);\n}\n// Cache key definitions\nconst KEYS = {\n CREDENTIALS: (userId, accountId)=>`email:credentials:${userId}:${accountId}`,\n SESSION: (userId)=>`email:session:${userId}`,\n EMAIL_LIST: (userId, accountId, folder, page, perPage)=>`email:list:${userId}:${accountId}:${folder}:${page}:${perPage}`,\n EMAIL_CONTENT: (userId, accountId, emailId)=>`email:content:${userId}:${accountId}:${emailId}`\n};\n// TTL constants in seconds\nconst TTL = {\n CREDENTIALS: 60 * 60 * 24,\n SESSION: 60 * 60 * 4,\n EMAIL_LIST: 60 * 5,\n EMAIL_CONTENT: 60 * 15 // 15 minutes\n};\n/**\n * Cache email credentials in Redis\n */ async function cacheEmailCredentials(userId, accountId, credentials) {\n const redis = getRedisClient();\n const key = KEYS.CREDENTIALS(userId, accountId);\n // Validate credentials before caching\n if (!credentials.email || !credentials.host || !credentials.password && !credentials.useOAuth) {\n console.error(`Cannot cache incomplete credentials for user ${userId}`);\n return;\n }\n try {\n console.log(`Caching credentials for user ${userId}`);\n // Create a copy without the password to store\n const secureCredentials = {\n email: credentials.email,\n host: credentials.host,\n port: credentials.port,\n secure: credentials.secure ?? true,\n // Include the extended fields\n ...credentials.smtp_host && {\n smtp_host: credentials.smtp_host\n },\n ...credentials.smtp_port && {\n smtp_port: credentials.smtp_port\n },\n ...credentials.smtp_secure !== undefined && {\n smtp_secure: credentials.smtp_secure\n },\n ...credentials.display_name && {\n display_name: credentials.display_name\n },\n ...credentials.color && {\n color: credentials.color\n },\n // Include OAuth fields\n ...credentials.useOAuth !== undefined && {\n useOAuth: credentials.useOAuth\n },\n ...credentials.accessToken && {\n accessToken: credentials.accessToken\n },\n ...credentials.refreshToken && {\n refreshToken: credentials.refreshToken\n },\n ...credentials.tokenExpiry && {\n tokenExpiry: credentials.tokenExpiry\n }\n };\n // Encrypt password if provided\n if (credentials.password) {\n try {\n const encrypted = encryptData(credentials.password);\n console.log(`Successfully encrypted password for user ${userId}`);\n secureCredentials.encryptedPassword = encrypted;\n } catch (encryptError) {\n console.error(`Failed to encrypt password for user ${userId}:`, encryptError);\n // Continue anyway since we might have OAuth tokens\n }\n }\n await redis.set(key, JSON.stringify(secureCredentials), 'EX', TTL.CREDENTIALS);\n console.log(`Credentials cached for user ${userId}`);\n } catch (error) {\n console.error(`Error caching credentials for user ${userId}:`, error);\n }\n}\n/**\n * Get email credentials from Redis\n */ async function getEmailCredentials(userId, accountId) {\n const redis = getRedisClient();\n const key = KEYS.CREDENTIALS(userId, accountId);\n try {\n const credStr = await redis.get(key);\n if (!credStr) {\n return null;\n }\n const creds = JSON.parse(credStr);\n let password;\n // Handle OAuth accounts (they might not have a password)\n if (creds.encryptedPassword) {\n try {\n // Decrypt the password\n password = decryptData(creds.encryptedPassword);\n } catch (decryptError) {\n console.error(`Failed to decrypt password for user ${userId}:`, decryptError);\n // For OAuth accounts, we can continue without a password\n if (!creds.useOAuth) {\n return null;\n }\n }\n }\n // Return the full credentials with decrypted password if available\n const result = {\n email: creds.email,\n host: creds.host,\n port: creds.port,\n secure: creds.secure ?? true,\n ...password && {\n password\n },\n ...creds.smtp_host && {\n smtp_host: creds.smtp_host\n },\n ...creds.smtp_port && {\n smtp_port: creds.smtp_port\n },\n ...creds.smtp_secure !== undefined && {\n smtp_secure: creds.smtp_secure\n },\n ...creds.display_name && {\n display_name: creds.display_name\n },\n ...creds.color && {\n color: creds.color\n },\n // Include OAuth fields\n ...creds.useOAuth !== undefined && {\n useOAuth: creds.useOAuth\n },\n ...creds.accessToken && {\n accessToken: creds.accessToken\n },\n ...creds.refreshToken && {\n refreshToken: creds.refreshToken\n },\n ...creds.tokenExpiry && {\n tokenExpiry: creds.tokenExpiry\n }\n };\n return result;\n } catch (error) {\n console.error(`Error getting credentials for user ${userId}:`, error);\n return null;\n }\n}\n/**\n * Cache IMAP session data for quick reconnection\n */ async function cacheImapSession(userId, sessionData) {\n const redis = getRedisClient();\n const key = KEYS.SESSION(userId);\n // Always update the lastActive timestamp\n sessionData.lastActive = Date.now();\n await redis.set(key, JSON.stringify(sessionData), 'EX', TTL.SESSION);\n}\n/**\n * Get cached IMAP session data\n */ async function getCachedImapSession(userId) {\n const redis = getRedisClient();\n const key = KEYS.SESSION(userId);\n const cachedData = await redis.get(key);\n if (!cachedData) return null;\n return JSON.parse(cachedData);\n}\n/**\n * Cache email list in Redis\n */ async function cacheEmailList(userId, accountId, folder, page, perPage, data) {\n const redis = getRedisClient();\n const key = KEYS.EMAIL_LIST(userId, accountId, folder, page, perPage);\n await redis.set(key, JSON.stringify(data), 'EX', TTL.EMAIL_LIST);\n}\n/**\n * Get cached email list from Redis\n */ async function getCachedEmailList(userId, accountId, folder, page, perPage) {\n const redis = getRedisClient();\n const key = KEYS.EMAIL_LIST(userId, accountId, folder, page, perPage);\n const cachedData = await redis.get(key);\n if (!cachedData) return null;\n return JSON.parse(cachedData);\n}\n/**\n * Cache email content in Redis\n */ async function cacheEmailContent(userId, accountId, emailId, data) {\n const redis = getRedisClient();\n const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n await redis.set(key, JSON.stringify(data), 'EX', TTL.EMAIL_CONTENT);\n}\n/**\n * Get cached email content from Redis\n */ async function getCachedEmailContent(userId, accountId, emailId) {\n const redis = getRedisClient();\n const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n const cachedData = await redis.get(key);\n if (!cachedData) return null;\n return JSON.parse(cachedData);\n}\n/**\n * Invalidate all email caches for a folder\n */ async function invalidateFolderCache(userId, accountId, folder) {\n const redis = getRedisClient();\n const pattern = `email:list:${userId}:${accountId}:${folder}:*`;\n // Use SCAN to find and delete keys matching the pattern\n let cursor = '0';\n do {\n const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', pattern, 'COUNT', 100);\n cursor = nextCursor;\n if (keys.length > 0) {\n await redis.del(...keys);\n }\n }while (cursor !== '0');\n}\n/**\n * Invalidate email content cache\n */ async function invalidateEmailContentCache(userId, accountId, emailId) {\n const redis = getRedisClient();\n const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n await redis.del(key);\n}\n/**\n * Warm up Redis connection to avoid cold starts\n */ async function warmupRedisCache() {\n try {\n // Ping Redis to establish connection early\n const redis = getRedisClient();\n await redis.ping();\n console.log('Redis connection warmed up');\n return true;\n } catch (error) {\n console.error('Error warming up Redis:', error);\n return false;\n }\n}\n/**\n * Get Redis connection status\n */ async function getRedisStatus() {\n try {\n const redis = getRedisClient();\n const pong = await redis.ping();\n return {\n status: 'connected',\n ping: pong\n };\n } catch (error) {\n return {\n status: 'error',\n error: error instanceof Error ? error.message : String(error)\n };\n }\n}\n/**\n * Invalidate all user email caches (email lists and content)\n */ async function invalidateUserEmailCache(userId) {\n const redis = getRedisClient();\n // Patterns to delete\n const patterns = [\n `email:list:${userId}:*`,\n `email:content:${userId}:*`\n ];\n for (const pattern of patterns){\n let cursor = '0';\n do {\n const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', pattern, 'COUNT', 100);\n cursor = nextCursor;\n if (keys.length > 0) {\n await redis.del(...keys);\n }\n }while (cursor !== '0');\n }\n}\n/**\n * Get cached email credentials from Redis\n * @deprecated Use getEmailCredentials instead\n */ async function getCachedEmailCredentials(userId, accountId) {\n return getEmailCredentials(userId, accountId);\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./lib/redis.ts","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;AAA4B;AACK;AAEjC,0BAA0B;AAC1B,IAAIE,cAA4B;AAChC,IAAIC,eAAe;AACnB,IAAIC,qBAAqB;AACzB,MAAMC,yBAAyB;AAE/B;;CAEC,GACM,SAASC;IACd,IAAIJ,eAAeA,YAAYK,MAAM,KAAK,SAAS;QACjD,OAAOL;IACT;IAEA,IAAIC,cAAc;QAChB,iEAAiE;QACjE,0DAA0D;QAC1D,IAAID,aAAa,OAAOA;QAExB,0EAA0E;QAC1EM,QAAQC,IAAI,CAAC;IACf;IAEA,IAAI,CAACP,aAAa;QAChBC,eAAe;QACfC,qBAAqB;QAErB,kEAAkE;QAClE,MAAMM,eAAe;YACnBC,MAAMC,QAAQC,GAAG,CAACC,UAAU;YAC5BC,MAAMH,QAAQC,GAAG,CAACG,UAAU,GAAGC,SAASL,QAAQC,GAAG,CAACG,UAAU,IAAIE;YAClEC,UAAUP,QAAQC,GAAG,CAACO,cAAc;YACpCC,eAAe,CAACC;gBACdlB,qBAAqBkB;gBACrB,IAAIA,QAAQjB,wBAAwB;oBAClCG,QAAQe,KAAK,CAAC,CAAC,8BAA8B,EAAED,MAAM,oBAAoB,CAAC;oBAC1E,OAAO,MAAM,2BAA2B;gBAC1C;gBACA,MAAME,QAAQC,KAAKC,GAAG,CAACJ,QAAQ,KAAK;gBACpCd,QAAQmB,GAAG,CAAC,CAAC,wBAAwB,EAAEL,MAAM,cAAc,EAAEE,MAAM,EAAE,CAAC;gBACtE,OAAOA;YACT;YACAI,sBAAsB;YACtBC,oBAAoB;YACpBC,gBAAgB;YAChBC,mBAAmB;YACnBC,WAAW;YACXC,WAAW,GAAG,+BAA+B;QAC/C;QAEAzB,QAAQmB,GAAG,CAAC;QACZzB,cAAc,IAAIF,gDAAKA,CAACU;QAExBR,YAAYgC,EAAE,CAAC,SAAS,CAACC;YACvB3B,QAAQe,KAAK,CAAC,2BAA2BY;YAEzC,kDAAkD;YAClD,IAAI/B,qBAAqBC,wBAAwB;gBAC/CG,QAAQe,KAAK,CAAC;gBACdrB,cAAc;gBACdC,eAAe;YACjB;QACF;QAEAD,YAAYgC,EAAE,CAAC,WAAW;YACxB1B,QAAQmB,GAAG,CAAC;YACZxB,eAAe;YACfC,qBAAqB;QACvB;QAEAF,YAAYgC,EAAE,CAAC,gBAAgB;YAC7B1B,QAAQmB,GAAG,CAAC;YACZxB,eAAe;QACjB;QAEAD,YAAYgC,EAAE,CAAC,SAAS;YACtB1B,QAAQmB,GAAG,CAAC;YACZxB,eAAe;QACjB;QAEAD,YAAYgC,EAAE,CAAC,OAAO;YACpB1B,QAAQmB,GAAG,CAAC;QACZ,wDAAwD;QAC1D;IACF;IAEA,OAAOzB;AACT;AAEA;;CAEC,GACM,eAAekC;IACpB,IAAIlC,aAAa;QACf,MAAMA,YAAYmC,IAAI;QACtBnC,cAAc;IAChB;AACF;AAEA,uDAAuD;AACvD,MAAMoC,mBAAmB;IACvB,OAAO1B,QAAQC,GAAG,CAAC0B,oBAAoB,IAAI;AAC7C;AAEA;;CAEC,GACM,SAASC,YAAYC,IAAY;IACtC,OAAOxC,oDAAY,CAAC0C,OAAO,CAACF,MAAMH,oBAAoBM,QAAQ;AAChE;AAEA;;CAEC,GACM,SAASC,YAAYC,aAAqB;IAC/C,MAAMC,QAAQ9C,oDAAY,CAAC+C,OAAO,CAACF,eAAeR;IAClD,OAAOS,MAAMH,QAAQ,CAAC3C,sDAAY,CAACiD,IAAI;AACzC;AAEA,wBAAwB;AACjB,MAAMC,OAAO;IAClBC,aAAa,CAACC,QAAgBC,YAAsB,CAAC,kBAAkB,EAAED,OAAO,CAAC,EAAEC,WAAW;IAC9FC,SAAS,CAACF,SAAmB,CAAC,cAAc,EAAEA,QAAQ;IACtDG,YAAY,CAACH,QAAgBC,WAAmBG,QAAgBC,MAAcC,UAC5E,CAAC,WAAW,EAAEN,OAAO,CAAC,EAAEC,UAAU,CAAC,EAAEG,OAAO,CAAC,EAAEC,KAAK,CAAC,EAAEC,SAAS;IAClEC,eAAe,CAACP,QAAgBC,WAAmBO,UACjD,CAAC,cAAc,EAAER,OAAO,CAAC,EAAEC,UAAU,CAAC,EAAEO,SAAS;AACrD,EAAE;AAEF,2BAA2B;AACpB,MAAMC,MAAM;IACjBV,aAAa,KAAK,KAAK;IACvBG,SAAS,KAAK,KAAK;IACnBC,YAAY,KAAK;IACjBI,eAAe,KAAK,GAAO,aAAa;AAC1C,EAAE;AA4BF;;CAEC,GACM,eAAeG,sBACpBV,MAAc,EACdC,SAAiB,EACjBU,WAA6B;IAE7B,MAAMC,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKC,WAAW,CAACC,QAAQC;IAErC,sCAAsC;IACtC,IAAI,CAACU,YAAYG,KAAK,IAAI,CAACH,YAAYrD,IAAI,IAAK,CAACqD,YAAY7C,QAAQ,IAAI,CAAC6C,YAAYI,QAAQ,EAAG;QAC/F5D,QAAQe,KAAK,CAAC,CAAC,6CAA6C,EAAE8B,QAAQ;QACtE;IACF;IAEA,IAAI;QACF7C,QAAQmB,GAAG,CAAC,CAAC,6BAA6B,EAAE0B,QAAQ;QAEpD,8CAA8C;QAC9C,MAAMgB,oBAAsC;YAC1CF,OAAOH,YAAYG,KAAK;YACxBxD,MAAMqD,YAAYrD,IAAI;YACtBI,MAAMiD,YAAYjD,IAAI;YACtBuD,QAAQN,YAAYM,MAAM,IAAI;YAC9B,8BAA8B;YAC9B,GAAIN,YAAYO,SAAS,IAAI;gBAAEA,WAAWP,YAAYO,SAAS;YAAC,CAAC;YACjE,GAAIP,YAAYQ,SAAS,IAAI;gBAAEA,WAAWR,YAAYQ,SAAS;YAAC,CAAC;YACjE,GAAIR,YAAYS,WAAW,KAAKvD,aAAa;gBAAEuD,aAAaT,YAAYS,WAAW;YAAC,CAAC;YACrF,GAAIT,YAAYU,YAAY,IAAI;gBAAEA,cAAcV,YAAYU,YAAY;YAAC,CAAC;YAC1E,GAAIV,YAAYW,KAAK,IAAI;gBAAEA,OAAOX,YAAYW,KAAK;YAAC,CAAC;YACrD,uBAAuB;YACvB,GAAIX,YAAYI,QAAQ,KAAKlD,aAAa;gBAAEkD,UAAUJ,YAAYI,QAAQ;YAAC,CAAC;YAC5E,GAAIJ,YAAYY,WAAW,IAAI;gBAAEA,aAAaZ,YAAYY,WAAW;YAAC,CAAC;YACvE,GAAIZ,YAAYa,YAAY,IAAI;gBAAEA,cAAcb,YAAYa,YAAY;YAAC,CAAC;YAC1E,GAAIb,YAAYc,WAAW,IAAI;gBAAEA,aAAad,YAAYc,WAAW;YAAC,CAAC;QACzE;QAEA,+BAA+B;QAC/B,IAAId,YAAY7C,QAAQ,EAAE;YACxB,IAAI;gBACF,MAAM4D,YAAYvC,YAAYwB,YAAY7C,QAAQ;gBAClDX,QAAQmB,GAAG,CAAC,CAAC,yCAAyC,EAAE0B,QAAQ;gBAChEgB,kBAAkBW,iBAAiB,GAAGD;YACxC,EAAE,OAAOE,cAAc;gBACrBzE,QAAQe,KAAK,CAAC,CAAC,oCAAoC,EAAE8B,OAAO,CAAC,CAAC,EAAE4B;YAChE,mDAAmD;YACrD;QACF;QAEA,MAAMhB,MAAMiB,GAAG,CAAChB,KAAKiB,KAAKC,SAAS,CAACf,oBAAoB,MAAMP,IAAIV,WAAW;QAC7E5C,QAAQmB,GAAG,CAAC,CAAC,4BAA4B,EAAE0B,QAAQ;IACrD,EAAE,OAAO9B,OAAO;QACdf,QAAQe,KAAK,CAAC,CAAC,mCAAmC,EAAE8B,OAAO,CAAC,CAAC,EAAE9B;IACjE;AACF;AAEA;;CAEC,GACM,eAAe8D,oBACpBhC,MAAc,EACdC,SAAiB;IAEjB,MAAMW,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKC,WAAW,CAACC,QAAQC;IAErC,IAAI;QACF,MAAMgC,UAAU,MAAMrB,MAAMsB,GAAG,CAACrB;QAEhC,IAAI,CAACoB,SAAS;YACZ,OAAO;QACT;QAEA,MAAME,QAAQL,KAAKM,KAAK,CAACH;QAEzB,IAAInE;QAEJ,yDAAyD;QACzD,IAAIqE,MAAMR,iBAAiB,EAAE;YAC3B,IAAI;gBACF,uBAAuB;gBACvB7D,WAAW0B,YAAY2C,MAAMR,iBAAiB;YAChD,EAAE,OAAOU,cAAc;gBACrBlF,QAAQe,KAAK,CAAC,CAAC,oCAAoC,EAAE8B,OAAO,CAAC,CAAC,EAAEqC;gBAChE,yDAAyD;gBACzD,IAAI,CAACF,MAAMpB,QAAQ,EAAE;oBACnB,OAAO;gBACT;YACF;QACF;QAEA,mEAAmE;QACnE,MAAMuB,SAA2B;YAC/BxB,OAAOqB,MAAMrB,KAAK;YAClBxD,MAAM6E,MAAM7E,IAAI;YAChBI,MAAMyE,MAAMzE,IAAI;YAChBuD,QAAQkB,MAAMlB,MAAM,IAAI;YACxB,GAAInD,YAAY;gBAAEA;YAAS,CAAC;YAC5B,GAAIqE,MAAMjB,SAAS,IAAI;gBAAEA,WAAWiB,MAAMjB,SAAS;YAAC,CAAC;YACrD,GAAIiB,MAAMhB,SAAS,IAAI;gBAAEA,WAAWgB,MAAMhB,SAAS;YAAC,CAAC;YACrD,GAAIgB,MAAMf,WAAW,KAAKvD,aAAa;gBAAEuD,aAAae,MAAMf,WAAW;YAAC,CAAC;YACzE,GAAIe,MAAMd,YAAY,IAAI;gBAAEA,cAAcc,MAAMd,YAAY;YAAC,CAAC;YAC9D,GAAIc,MAAMb,KAAK,IAAI;gBAAEA,OAAOa,MAAMb,KAAK;YAAC,CAAC;YACzC,uBAAuB;YACvB,GAAIa,MAAMpB,QAAQ,KAAKlD,aAAa;gBAAEkD,UAAUoB,MAAMpB,QAAQ;YAAC,CAAC;YAChE,GAAIoB,MAAMZ,WAAW,IAAI;gBAAEA,aAAaY,MAAMZ,WAAW;YAAC,CAAC;YAC3D,GAAIY,MAAMX,YAAY,IAAI;gBAAEA,cAAcW,MAAMX,YAAY;YAAC,CAAC;YAC9D,GAAIW,MAAMV,WAAW,IAAI;gBAAEA,aAAaU,MAAMV,WAAW;YAAC,CAAC;QAC7D;QAEA,OAAOa;IACT,EAAE,OAAOpE,OAAO;QACdf,QAAQe,KAAK,CAAC,CAAC,mCAAmC,EAAE8B,OAAO,CAAC,CAAC,EAAE9B;QAC/D,OAAO;IACT;AACF;AAEA;;CAEC,GACM,eAAeqE,iBACpBvC,MAAc,EACdwC,WAA4B;IAE5B,MAAM5B,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKI,OAAO,CAACF;IAEzB,yCAAyC;IACzCwC,YAAYC,UAAU,GAAGC,KAAKC,GAAG;IAEjC,MAAM/B,MAAMiB,GAAG,CAAChB,KAAKiB,KAAKC,SAAS,CAACS,cAAc,MAAM/B,IAAIP,OAAO;AACrE;AAEA;;CAEC,GACM,eAAe0C,qBACpB5C,MAAc;IAEd,MAAMY,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKI,OAAO,CAACF;IAEzB,MAAM6C,aAAa,MAAMjC,MAAMsB,GAAG,CAACrB;IACnC,IAAI,CAACgC,YAAY,OAAO;IAExB,OAAOf,KAAKM,KAAK,CAACS;AACpB;AAEA;;CAEC,GACM,eAAeC,eACpB9C,MAAc,EACdC,SAAiB,EACjBG,MAAc,EACdC,IAAY,EACZC,OAAe,EACflB,IAAS;IAET,MAAMwB,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKK,UAAU,CAACH,QAAQC,WAAWG,QAAQC,MAAMC;IAE7D,MAAMM,MAAMiB,GAAG,CAAChB,KAAKiB,KAAKC,SAAS,CAAC3C,OAAO,MAAMqB,IAAIN,UAAU;AACjE;AAEA;;CAEC,GACM,eAAe4C,mBACpB/C,MAAc,EACdC,SAAiB,EACjBG,MAAc,EACdC,IAAY,EACZC,OAAe;IAEf,MAAMM,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKK,UAAU,CAACH,QAAQC,WAAWG,QAAQC,MAAMC;IAE7D,MAAMuC,aAAa,MAAMjC,MAAMsB,GAAG,CAACrB;IACnC,IAAI,CAACgC,YAAY,OAAO;IAExB,OAAOf,KAAKM,KAAK,CAACS;AACpB;AAEA;;CAEC,GACM,eAAeG,kBACpBhD,MAAc,EACdC,SAAiB,EACjBO,OAAe,EACfpB,IAAS;IAET,MAAMwB,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKS,aAAa,CAACP,QAAQC,WAAWO;IAElD,MAAMI,MAAMiB,GAAG,CAAChB,KAAKiB,KAAKC,SAAS,CAAC3C,OAAO,MAAMqB,IAAIF,aAAa;AACpE;AAEA;;CAEC,GACM,eAAe0C,sBACpBjD,MAAc,EACdC,SAAiB,EACjBO,OAAe;IAEf,MAAMI,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKS,aAAa,CAACP,QAAQC,WAAWO;IAElD,MAAMqC,aAAa,MAAMjC,MAAMsB,GAAG,CAACrB;IACnC,IAAI,CAACgC,YAAY,OAAO;IAExB,OAAOf,KAAKM,KAAK,CAACS;AACpB;AAEA;;CAEC,GACM,eAAeK,sBACpBlD,MAAc,EACdC,SAAiB,EACjBG,MAAc;IAEd,MAAMQ,QAAQ3D;IACd,MAAMkG,UAAU,CAAC,WAAW,EAAEnD,OAAO,CAAC,EAAEC,UAAU,CAAC,EAAEG,OAAO,EAAE,CAAC;IAE/D,wDAAwD;IACxD,IAAIgD,SAAS;IACb,GAAG;QACD,MAAM,CAACC,YAAYC,KAAK,GAAG,MAAM1C,MAAM2C,IAAI,CAACH,QAAQ,SAASD,SAAS,SAAS;QAC/EC,SAASC;QAET,IAAIC,KAAKE,MAAM,GAAG,GAAG;YACnB,MAAM5C,MAAM6C,GAAG,IAAIH;QACrB;IACF,QAASF,WAAW,KAAK;AAC3B;AAEA;;CAEC,GACM,eAAeM,4BACpB1D,MAAc,EACdC,SAAiB,EACjBO,OAAe;IAEf,MAAMI,QAAQ3D;IACd,MAAM4D,MAAMf,KAAKS,aAAa,CAACP,QAAQC,WAAWO;IAElD,MAAMI,MAAM6C,GAAG,CAAC5C;AAClB;AAEA;;CAEC,GACM,eAAe8C;IACpB,IAAI;QACF,2CAA2C;QAC3C,MAAM/C,QAAQ3D;QACd,MAAM2D,MAAMgD,IAAI;QAChBzG,QAAQmB,GAAG,CAAC;QACZ,OAAO;IACT,EAAE,OAAOJ,OAAO;QACdf,QAAQe,KAAK,CAAC,2BAA2BA;QACzC,OAAO;IACT;AACF;AAEA;;CAEC,GACM,eAAe2F;IAKpB,IAAI;QACF,MAAMjD,QAAQ3D;QACd,MAAM6G,OAAO,MAAMlD,MAAMgD,IAAI;QAC7B,OAAO;YACL1G,QAAQ;YACR0G,MAAME;QACR;IACF,EAAE,OAAO5F,OAAO;QACd,OAAO;YACLhB,QAAQ;YACRgB,OAAOA,iBAAiB6F,QAAQ7F,MAAM8F,OAAO,GAAGC,OAAO/F;QACzD;IACF;AACF;AAEA;;CAEC,GACM,eAAegG,yBACpBlE,MAAc;IAEd,MAAMY,QAAQ3D;IAEd,qBAAqB;IACrB,MAAMkH,WAAW;QACf,CAAC,WAAW,EAAEnE,OAAO,EAAE,CAAC;QACxB,CAAC,cAAc,EAAEA,OAAO,EAAE,CAAC;KAC5B;IAED,KAAK,MAAMmD,WAAWgB,SAAU;QAC9B,IAAIf,SAAS;QACb,GAAG;YACD,MAAM,CAACC,YAAYC,KAAK,GAAG,MAAM1C,MAAM2C,IAAI,CAACH,QAAQ,SAASD,SAAS,SAAS;YAC/EC,SAASC;YAET,IAAIC,KAAKE,MAAM,GAAG,GAAG;gBACnB,MAAM5C,MAAM6C,GAAG,IAAIH;YACrB;QACF,QAASF,WAAW,KAAK;IAC3B;AACF;AAEA;;;CAGC,GACM,eAAegB,0BACpBpE,MAAc,EACdC,SAAiB;IAEjB,OAAO+B,oBAAoBhC,QAAQC;AACrC","sources":["/home/alma/nextgen/Neah-mail/lib/redis.ts"],"sourcesContent":["import Redis from 'ioredis';\nimport CryptoJS from 'crypto-js';\n\n// Initialize Redis client\nlet redisClient: Redis | null = null;\nlet isConnecting = false;\nlet connectionAttempts = 0;\nconst MAX_RECONNECT_ATTEMPTS = 5;\n\n/**\n * Get a Redis client instance (singleton pattern) with improved connection management\n */\nexport function getRedisClient(): Redis {\n  if (redisClient && redisClient.status === 'ready') {\n    return redisClient;\n  }\n  \n  if (isConnecting) {\n    // If we're already trying to connect, return the existing client\n    // This prevents multiple simultaneous connection attempts\n    if (redisClient) return redisClient;\n    \n    // This is a fallback in case we're connecting but don't have a client yet\n    console.warn('Redis connection in progress, creating temporary client');\n  }\n  \n  if (!redisClient) {\n    isConnecting = true;\n    connectionAttempts = 0;\n    \n    // Set Redis connection parameters from environment variables only\n    const redisOptions = {\n      host: process.env.REDIS_HOST,\n      port: process.env.REDIS_PORT ? parseInt(process.env.REDIS_PORT) : undefined,\n      password: process.env.REDIS_PASSWORD,\n      retryStrategy: (times: number) => {\n        connectionAttempts = times;\n        if (times > MAX_RECONNECT_ATTEMPTS) {\n          console.error(`Redis connection failed after ${times} attempts, giving up`);\n          return null; // Stop trying to reconnect\n        }\n        const delay = Math.min(times * 100, 5000);\n        console.log(`Redis reconnect attempt ${times}, retrying in ${delay}ms`);\n        return delay;\n      },\n      maxRetriesPerRequest: 5,\n      enableOfflineQueue: true,\n      connectTimeout: 10000, // 10 seconds\n      disconnectTimeout: 2000, // 2 seconds\n      keepAlive: 10000, // 10 seconds\n      keyPrefix: '' // No prefix to keep keys clean\n    };\n    \n    console.log('Connecting to Redis using environment variables');\n    redisClient = new Redis(redisOptions);\n    \n    redisClient.on('error', (err) => {\n      console.error('Redis connection error:', err);\n      \n      // Only set to null if we've exceeded max attempts\n      if (connectionAttempts > MAX_RECONNECT_ATTEMPTS) {\n        console.error('Redis connection failed permanently, will create new client on next request');\n        redisClient = null;\n        isConnecting = false;\n      }\n    });\n    \n    redisClient.on('connect', () => {\n      console.log('Successfully connected to Redis');\n      isConnecting = false;\n      connectionAttempts = 0;\n    });\n    \n    redisClient.on('reconnecting', () => {\n      console.log('Reconnecting to Redis...');\n      isConnecting = true;\n    });\n    \n    redisClient.on('ready', () => {\n      console.log('Redis connection warmed up');\n      isConnecting = false;\n    });\n    \n    redisClient.on('end', () => {\n      console.log('Redis connection ended');\n      // Don't set to null here - let the error handler decide\n    });\n  }\n  \n  return redisClient;\n}\n\n/**\n * Close Redis connection (useful for serverless environments)\n */\nexport async function closeRedisConnection(): Promise<void> {\n  if (redisClient) {\n    await redisClient.quit();\n    redisClient = null;\n  }\n}\n\n// Encryption key from environment variable or fallback\nconst getEncryptionKey = () => {\n  return process.env.REDIS_ENCRYPTION_KEY || 'default-encryption-key-change-in-production';\n};\n\n/**\n * Encrypt sensitive data before storing in Redis\n */\nexport function encryptData(data: string): string {\n  return CryptoJS.AES.encrypt(data, getEncryptionKey()).toString();\n}\n\n/**\n * Decrypt sensitive data retrieved from Redis\n */\nexport function decryptData(encryptedData: string): string {\n  const bytes = CryptoJS.AES.decrypt(encryptedData, getEncryptionKey());\n  return bytes.toString(CryptoJS.enc.Utf8);\n}\n\n// Cache key definitions\nexport const KEYS = {\n  CREDENTIALS: (userId: string, accountId: string) => `email:credentials:${userId}:${accountId}`,\n  SESSION: (userId: string) => `email:session:${userId}`,\n  EMAIL_LIST: (userId: string, accountId: string, folder: string, page: number, perPage: number) => \n    `email:list:${userId}:${accountId}:${folder}:${page}:${perPage}`,\n  EMAIL_CONTENT: (userId: string, accountId: string, emailId: string) => \n    `email:content:${userId}:${accountId}:${emailId}`\n};\n\n// TTL constants in seconds\nexport const TTL = {\n  CREDENTIALS: 60 * 60 * 24, // 24 hours\n  SESSION: 60 * 60 * 4,      // 4 hours (increased from 30 minutes)\n  EMAIL_LIST: 60 * 5,        // 5 minutes\n  EMAIL_CONTENT: 60 * 15     // 15 minutes\n};\n\ninterface EmailCredentials {\n  email: string;\n  password?: string;\n  host: string;\n  port: number;\n  secure?: boolean;\n  encryptedPassword?: string;\n  smtp_host?: string;\n  smtp_port?: number;\n  smtp_secure?: boolean;\n  display_name?: string;\n  color?: string;\n  useOAuth?: boolean;\n  accessToken?: string;\n  refreshToken?: string;\n  tokenExpiry?: number;\n}\n\ninterface ImapSessionData {\n  connectionId?: string;\n  lastActive: number;\n  mailboxes?: string[];\n  lastVisit?: number;\n  defaultAccountId?: string;\n}\n\n/**\n * Cache email credentials in Redis\n */\nexport async function cacheEmailCredentials(\n  userId: string,\n  accountId: string,\n  credentials: EmailCredentials\n): Promise<void> {\n  const redis = getRedisClient();\n  const key = KEYS.CREDENTIALS(userId, accountId);\n  \n  // Validate credentials before caching\n  if (!credentials.email || !credentials.host || (!credentials.password && !credentials.useOAuth)) {\n    console.error(`Cannot cache incomplete credentials for user ${userId}`);\n    return;\n  }\n  \n  try {\n    console.log(`Caching credentials for user ${userId}`);\n    \n    // Create a copy without the password to store\n    const secureCredentials: EmailCredentials = {\n      email: credentials.email,\n      host: credentials.host,\n      port: credentials.port,\n      secure: credentials.secure ?? true,\n      // Include the extended fields\n      ...(credentials.smtp_host && { smtp_host: credentials.smtp_host }),\n      ...(credentials.smtp_port && { smtp_port: credentials.smtp_port }),\n      ...(credentials.smtp_secure !== undefined && { smtp_secure: credentials.smtp_secure }),\n      ...(credentials.display_name && { display_name: credentials.display_name }),\n      ...(credentials.color && { color: credentials.color }),\n      // Include OAuth fields\n      ...(credentials.useOAuth !== undefined && { useOAuth: credentials.useOAuth }),\n      ...(credentials.accessToken && { accessToken: credentials.accessToken }),\n      ...(credentials.refreshToken && { refreshToken: credentials.refreshToken }),\n      ...(credentials.tokenExpiry && { tokenExpiry: credentials.tokenExpiry })\n    };\n    \n    // Encrypt password if provided\n    if (credentials.password) {\n      try {\n        const encrypted = encryptData(credentials.password);\n        console.log(`Successfully encrypted password for user ${userId}`);\n        secureCredentials.encryptedPassword = encrypted;\n      } catch (encryptError) {\n        console.error(`Failed to encrypt password for user ${userId}:`, encryptError);\n        // Continue anyway since we might have OAuth tokens\n      }\n    }\n    \n    await redis.set(key, JSON.stringify(secureCredentials), 'EX', TTL.CREDENTIALS);\n    console.log(`Credentials cached for user ${userId}`);\n  } catch (error) {\n    console.error(`Error caching credentials for user ${userId}:`, error);\n  }\n}\n\n/**\n * Get email credentials from Redis\n */\nexport async function getEmailCredentials(\n  userId: string,\n  accountId: string\n): Promise<EmailCredentials | null> {\n  const redis = getRedisClient();\n  const key = KEYS.CREDENTIALS(userId, accountId);\n  \n  try {\n    const credStr = await redis.get(key);\n    \n    if (!credStr) {\n      return null;\n    }\n    \n    const creds = JSON.parse(credStr) as EmailCredentials;\n    \n    let password: string | undefined;\n    \n    // Handle OAuth accounts (they might not have a password)\n    if (creds.encryptedPassword) {\n      try {\n        // Decrypt the password\n        password = decryptData(creds.encryptedPassword);\n      } catch (decryptError) {\n        console.error(`Failed to decrypt password for user ${userId}:`, decryptError);\n        // For OAuth accounts, we can continue without a password\n        if (!creds.useOAuth) {\n          return null;\n        }\n      }\n    }\n    \n    // Return the full credentials with decrypted password if available\n    const result: EmailCredentials = {\n      email: creds.email,\n      host: creds.host,\n      port: creds.port,\n      secure: creds.secure ?? true,\n      ...(password && { password }),\n      ...(creds.smtp_host && { smtp_host: creds.smtp_host }),\n      ...(creds.smtp_port && { smtp_port: creds.smtp_port }),\n      ...(creds.smtp_secure !== undefined && { smtp_secure: creds.smtp_secure }),\n      ...(creds.display_name && { display_name: creds.display_name }),\n      ...(creds.color && { color: creds.color }),\n      // Include OAuth fields\n      ...(creds.useOAuth !== undefined && { useOAuth: creds.useOAuth }),\n      ...(creds.accessToken && { accessToken: creds.accessToken }),\n      ...(creds.refreshToken && { refreshToken: creds.refreshToken }),\n      ...(creds.tokenExpiry && { tokenExpiry: creds.tokenExpiry })\n    };\n    \n    return result;\n  } catch (error) {\n    console.error(`Error getting credentials for user ${userId}:`, error);\n    return null;\n  }\n}\n\n/**\n * Cache IMAP session data for quick reconnection\n */\nexport async function cacheImapSession(\n  userId: string,\n  sessionData: ImapSessionData\n): Promise<void> {\n  const redis = getRedisClient();\n  const key = KEYS.SESSION(userId);\n  \n  // Always update the lastActive timestamp\n  sessionData.lastActive = Date.now();\n  \n  await redis.set(key, JSON.stringify(sessionData), 'EX', TTL.SESSION);\n}\n\n/**\n * Get cached IMAP session data\n */\nexport async function getCachedImapSession(\n  userId: string\n): Promise<ImapSessionData | null> {\n  const redis = getRedisClient();\n  const key = KEYS.SESSION(userId);\n  \n  const cachedData = await redis.get(key);\n  if (!cachedData) return null;\n  \n  return JSON.parse(cachedData) as ImapSessionData;\n}\n\n/**\n * Cache email list in Redis\n */\nexport async function cacheEmailList(\n  userId: string,\n  accountId: string,\n  folder: string,\n  page: number,\n  perPage: number,\n  data: any\n): Promise<void> {\n  const redis = getRedisClient();\n  const key = KEYS.EMAIL_LIST(userId, accountId, folder, page, perPage);\n  \n  await redis.set(key, JSON.stringify(data), 'EX', TTL.EMAIL_LIST);\n}\n\n/**\n * Get cached email list from Redis\n */\nexport async function getCachedEmailList(\n  userId: string,\n  accountId: string,\n  folder: string,\n  page: number,\n  perPage: number\n): Promise<any | null> {\n  const redis = getRedisClient();\n  const key = KEYS.EMAIL_LIST(userId, accountId, folder, page, perPage);\n  \n  const cachedData = await redis.get(key);\n  if (!cachedData) return null;\n  \n  return JSON.parse(cachedData);\n}\n\n/**\n * Cache email content in Redis\n */\nexport async function cacheEmailContent(\n  userId: string,\n  accountId: string,\n  emailId: string,\n  data: any\n): Promise<void> {\n  const redis = getRedisClient();\n  const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n  \n  await redis.set(key, JSON.stringify(data), 'EX', TTL.EMAIL_CONTENT);\n}\n\n/**\n * Get cached email content from Redis\n */\nexport async function getCachedEmailContent(\n  userId: string,\n  accountId: string,\n  emailId: string\n): Promise<any | null> {\n  const redis = getRedisClient();\n  const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n  \n  const cachedData = await redis.get(key);\n  if (!cachedData) return null;\n  \n  return JSON.parse(cachedData);\n}\n\n/**\n * Invalidate all email caches for a folder\n */\nexport async function invalidateFolderCache(\n  userId: string,\n  accountId: string,\n  folder: string\n): Promise<void> {\n  const redis = getRedisClient();\n  const pattern = `email:list:${userId}:${accountId}:${folder}:*`;\n  \n  // Use SCAN to find and delete keys matching the pattern\n  let cursor = '0';\n  do {\n    const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', pattern, 'COUNT', 100);\n    cursor = nextCursor;\n    \n    if (keys.length > 0) {\n      await redis.del(...keys);\n    }\n  } while (cursor !== '0');\n}\n\n/**\n * Invalidate email content cache\n */\nexport async function invalidateEmailContentCache(\n  userId: string,\n  accountId: string,\n  emailId: string\n): Promise<void> {\n  const redis = getRedisClient();\n  const key = KEYS.EMAIL_CONTENT(userId, accountId, emailId);\n  \n  await redis.del(key);\n}\n\n/**\n * Warm up Redis connection to avoid cold starts\n */\nexport async function warmupRedisCache(): Promise<boolean> {\n  try {\n    // Ping Redis to establish connection early\n    const redis = getRedisClient();\n    await redis.ping();\n    console.log('Redis connection warmed up');\n    return true;\n  } catch (error) {\n    console.error('Error warming up Redis:', error);\n    return false;\n  }\n}\n\n/**\n * Get Redis connection status\n */\nexport async function getRedisStatus(): Promise<{\n  status: 'connected' | 'error';\n  ping?: string;\n  error?: string;\n}> {\n  try {\n    const redis = getRedisClient();\n    const pong = await redis.ping();\n    return {\n      status: 'connected',\n      ping: pong\n    };\n  } catch (error) {\n    return {\n      status: 'error',\n      error: error instanceof Error ? error.message : String(error)\n    };\n  }\n}\n\n/**\n * Invalidate all user email caches (email lists and content)\n */\nexport async function invalidateUserEmailCache(\n  userId: string\n): Promise<void> {\n  const redis = getRedisClient();\n  \n  // Patterns to delete\n  const patterns = [\n    `email:list:${userId}:*`,\n    `email:content:${userId}:*`\n  ];\n  \n  for (const pattern of patterns) {\n    let cursor = '0';\n    do {\n      const [nextCursor, keys] = await redis.scan(cursor, 'MATCH', pattern, 'COUNT', 100);\n      cursor = nextCursor;\n      \n      if (keys.length > 0) {\n        await redis.del(...keys);\n      }\n    } while (cursor !== '0');\n  }\n}\n\n/**\n * Get cached email credentials from Redis\n * @deprecated Use getEmailCredentials instead\n */\nexport async function getCachedEmailCredentials(\n  userId: string,\n  accountId: string\n): Promise<EmailCredentials | null> {\n  return getEmailCredentials(userId, accountId);\n} "],"names":["Redis","CryptoJS","redisClient","isConnecting","connectionAttempts","MAX_RECONNECT_ATTEMPTS","getRedisClient","status","console","warn","redisOptions","host","process","env","REDIS_HOST","port","REDIS_PORT","parseInt","undefined","password","REDIS_PASSWORD","retryStrategy","times","error","delay","Math","min","log","maxRetriesPerRequest","enableOfflineQueue","connectTimeout","disconnectTimeout","keepAlive","keyPrefix","on","err","closeRedisConnection","quit","getEncryptionKey","REDIS_ENCRYPTION_KEY","encryptData","data","AES","encrypt","toString","decryptData","encryptedData","bytes","decrypt","enc","Utf8","KEYS","CREDENTIALS","userId","accountId","SESSION","EMAIL_LIST","folder","page","perPage","EMAIL_CONTENT","emailId","TTL","cacheEmailCredentials","credentials","redis","key","email","useOAuth","secureCredentials","secure","smtp_host","smtp_port","smtp_secure","display_name","color","accessToken","refreshToken","tokenExpiry","encrypted","encryptedPassword","encryptError","set","JSON","stringify","getEmailCredentials","credStr","get","creds","parse","decryptError","result","cacheImapSession","sessionData","lastActive","Date","now","getCachedImapSession","cachedData","cacheEmailList","getCachedEmailList","cacheEmailContent","getCachedEmailContent","invalidateFolderCache","pattern","cursor","nextCursor","keys","scan","length","del","invalidateEmailContentCache","warmupRedisCache","ping","getRedisStatus","pong","Error","message","String","invalidateUserEmailCache","patterns","getCachedEmailCredentials"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./lib/redis.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/services/email-service.ts":
|
|
/*!***************************************!*\
|
|
!*** ./lib/services/email-service.ts ***!
|
|
\***************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ getEmailContent: () => (/* binding */ getEmailContent),\n/* harmony export */ getEmails: () => (/* binding */ getEmails),\n/* harmony export */ getImapConnection: () => (/* binding */ getImapConnection),\n/* harmony export */ getMailboxes: () => (/* binding */ getMailboxes),\n/* harmony export */ getUserEmailCredentials: () => (/* binding */ getUserEmailCredentials),\n/* harmony export */ markEmailReadStatus: () => (/* binding */ markEmailReadStatus),\n/* harmony export */ saveUserEmailCredentials: () => (/* binding */ saveUserEmailCredentials),\n/* harmony export */ sendEmail: () => (/* binding */ sendEmail),\n/* harmony export */ testEmailConnection: () => (/* binding */ testEmailConnection),\n/* harmony export */ toggleEmailFlag: () => (/* binding */ toggleEmailFlag)\n/* harmony export */ });\n/* harmony import */ var private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! private-next-rsc-server-reference */ \"(rsc)/./node_modules/next/dist/build/webpack/loaders/next-flight-loader/server-reference.js\");\n/* harmony import */ var private_next_rsc_action_encryption__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! private-next-rsc-action-encryption */ \"(rsc)/./node_modules/next/dist/server/app-render/encryption.js\");\n/* harmony import */ var private_next_rsc_action_encryption__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(private_next_rsc_action_encryption__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var server_only__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! server-only */ \"(rsc)/./node_modules/next/dist/compiled/server-only/empty.js\");\n/* harmony import */ var server_only__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(server_only__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var imapflow__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! imapflow */ \"(rsc)/./node_modules/imapflow/lib/imap-flow.js\");\n/* harmony import */ var nodemailer__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! nodemailer */ \"(rsc)/./node_modules/nodemailer/lib/nodemailer.js\");\n/* harmony import */ var _lib_prisma__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! @/lib/prisma */ \"(rsc)/./lib/prisma.ts\");\n/* harmony import */ var mailparser__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! mailparser */ \"(rsc)/./node_modules/mailparser/index.js\");\n/* harmony import */ var mailparser__WEBPACK_IMPORTED_MODULE_6___default = /*#__PURE__*/__webpack_require__.n(mailparser__WEBPACK_IMPORTED_MODULE_6__);\n/* harmony import */ var _lib_redis__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! @/lib/redis */ \"(rsc)/./lib/redis.ts\");\n/* harmony import */ var _token_refresh__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./token-refresh */ \"(rsc)/./lib/services/token-refresh.ts\");\n/* harmony import */ var private_next_rsc_action_validate__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! private-next-rsc-action-validate */ \"(rsc)/./node_modules/next/dist/build/webpack/loaders/next-flight-loader/action-validate.js\");\n/* __next_internal_action_entry_do_not_use__ {\"4010844c55b83dfb1f105bebefe0eefe1e6b431900\":\"testEmailConnection\",\"60249f33dc41bab8693201a3f19f5e5fb46e641c28\":\"getImapConnection\",\"60852a4e84650a79792bf7dba8eab6e2994fad2674\":\"sendEmail\",\"608647185521cdbec25a6e83fd03959d7becd6a6cd\":\"getMailboxes\",\"609e97c061f87c9d92e6b4c180de319248e8263787\":\"getUserEmailCredentials\",\"70576421f3f1a8e0b47693f06bf0b900d321800592\":\"saveUserEmailCredentials\",\"78725b1539278f6847adbb3f678020efc4d204d6de\":\"getEmailContent\",\"7c9e86d7555bbe449b808bd2ae7479f82af99409fe\":\"markEmailReadStatus\",\"7ccc49d190a944aede514b28ad88d4a52971336211\":\"toggleEmailFlag\",\"7e6fee2e8d6b5661c87219a81dea11090773f206cd\":\"getEmails\"} */ \n\n\n\n\n\n\n\n\n// Connection pool to reuse IMAP clients\nconst connectionPool = {};\n// Track overall connection metrics\nlet totalConnectionRequests = 0;\nlet totalNewConnections = 0;\nlet totalReuseConnections = 0;\nlet totalConnectionErrors = 0;\nlet lastMetricsReset = Date.now();\n// CRITICAL PERFORMANCE FIX: Increase idle timeout from 15 minutes to 30 minutes\n// This will keep connections alive longer and reduce reconnection delays\nconst CONNECTION_TIMEOUT = 30 * 60 * 1000; // Increased to 30 minutes (was 15 minutes)\nconst MAX_POOL_SIZE = 20; // Maximum number of connections to keep in the pool\nconst CONNECTION_CHECK_INTERVAL = 60 * 1000; // Check every minute\nconst MIN_POOL_SIZE = 2; // Keep at least this many active connections per user\n// Clean up idle connections periodically\nsetInterval(()=>{\n const now = Date.now();\n const connectionKeys = Object.keys(connectionPool);\n // If we've been collecting metrics for more than an hour, log and reset\n if (now - lastMetricsReset > 60 * 60 * 1000) {\n console.log(`[IMAP METRICS] Total requests: ${totalConnectionRequests}, New connections: ${totalNewConnections}, Reused: ${totalReuseConnections}, Errors: ${totalConnectionErrors}, Success rate: ${((totalReuseConnections + totalNewConnections) / totalConnectionRequests * 100).toFixed(2)}%`);\n totalConnectionRequests = 0;\n totalNewConnections = 0;\n totalReuseConnections = 0;\n totalConnectionErrors = 0;\n lastMetricsReset = now;\n }\n // PERFORMANCE FIX: Group connections by user for better management\n const connectionsByUser = {};\n connectionKeys.forEach((key)=>{\n const userId = key.split(':')[0];\n if (!connectionsByUser[userId]) {\n connectionsByUser[userId] = [];\n }\n connectionsByUser[userId].push(key);\n });\n // PERFORMANCE FIX: Manage pool size per user\n Object.entries(connectionsByUser).forEach(([userId, userConnections])=>{\n // Sort connections by last used (oldest first)\n const sortedConnections = userConnections.map((key)=>({\n key,\n lastUsed: connectionPool[key].lastUsed\n })).sort((a, b)=>a.lastUsed - b.lastUsed);\n // Keep the most recently used connections up to the min pool size\n const connectionsToKeep = sortedConnections.slice(-MIN_POOL_SIZE);\n const keepKeys = new Set(connectionsToKeep.map((conn)=>conn.key));\n // Check the rest for idle timeout\n sortedConnections.forEach(({ key, lastUsed })=>{\n // Skip connections to keep and those that are in the process of connecting\n if (keepKeys.has(key) || connectionPool[key].isConnecting) {\n return;\n }\n // Only close connections idle for too long\n if (now - lastUsed > CONNECTION_TIMEOUT) {\n console.log(`Closing idle IMAP connection for ${key} (idle for ${Math.round((now - lastUsed) / 1000)}s)`);\n try {\n if (connectionPool[key].client.usable) {\n connectionPool[key].client.logout().catch((err)=>{\n console.error(`Error closing idle connection for ${key}:`, err);\n });\n }\n } catch (error) {\n console.error(`Error checking connection status for ${key}:`, error);\n } finally{\n delete connectionPool[key];\n console.log(`Removed idle connection for ${key} from pool (pool size: ${Object.keys(connectionPool).length})`);\n }\n }\n });\n });\n // Log connection pool status with more details\n const activeCount = connectionKeys.filter((key)=>{\n const conn = connectionPool[key];\n return !conn.isConnecting && (conn.client?.usable || false);\n }).length;\n const connectingCount = connectionKeys.filter((key)=>connectionPool[key].isConnecting).length;\n console.log(`[IMAP POOL] Size: ${connectionKeys.length}, Active: ${activeCount}, Connecting: ${connectingCount}, Max: ${MAX_POOL_SIZE}`);\n}, CONNECTION_CHECK_INTERVAL);\n/**\n * Get IMAP connection for a user, reusing existing connections when possible\n * with improved connection handling and error recovery\n */ async function getImapConnection(userId, accountId) {\n const startTime = Date.now();\n totalConnectionRequests++;\n console.log(`Getting IMAP connection for user ${userId}${accountId ? ` account ${accountId}` : ''}`);\n // Special handling for 'default' accountId - find the first available account\n if (!accountId || accountId === 'default') {\n console.log(`No specific account provided or 'default' requested, trying to find first account for user ${userId}`);\n // Try getting the account ID from cache to avoid database query\n const sessionData = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.getCachedImapSession)(userId);\n if (sessionData && sessionData.defaultAccountId) {\n accountId = sessionData.defaultAccountId;\n console.log(`Using cached default account ID: ${accountId}`);\n } else {\n // Query to find all accounts for this user\n const accounts = await _lib_prisma__WEBPACK_IMPORTED_MODULE_5__.prisma.mailCredentials.findMany({\n where: {\n userId\n },\n orderBy: {\n createdAt: 'asc'\n },\n take: 1\n });\n if (accounts && accounts.length > 0) {\n const firstAccount = accounts[0];\n console.log(`Using first available account: ${firstAccount.id} (${firstAccount.email})`);\n accountId = firstAccount.id;\n // Cache default account ID for future use\n if (sessionData) {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheImapSession)(userId, {\n ...sessionData,\n defaultAccountId: accountId,\n lastActive: Date.now()\n });\n } else {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheImapSession)(userId, {\n lastActive: Date.now(),\n defaultAccountId: accountId\n });\n }\n } else {\n totalConnectionErrors++;\n throw new Error('No email accounts configured for this user');\n }\n }\n }\n // Use accountId in connection key to ensure different accounts get different connections\n const connectionKey = `${userId}:${accountId}`;\n // If we already have a connection for this key\n if (connectionPool[connectionKey]) {\n const connection = connectionPool[connectionKey];\n // If a connection is being established, wait for it\n if (connection.isConnecting && connection.connectionPromise) {\n console.log(`Connection in progress for ${connectionKey}, waiting for existing connection`);\n try {\n const client = await connection.connectionPromise;\n connection.lastUsed = Date.now();\n totalReuseConnections++;\n console.log(`[IMAP] Reused pending connection for ${connectionKey} in ${Date.now() - startTime}ms`);\n return client;\n } catch (error) {\n console.error(`Error waiting for connection for ${connectionKey}:`, error);\n // Fall through to create new connection\n }\n }\n // Try to use existing connection if it's usable\n try {\n // PERFORMANCE FIX: More robust connection status checking\n if (connection.client && connection.client.usable) {\n // Touch the connection to mark it as recently used\n connection.lastUsed = Date.now();\n console.log(`Reusing existing IMAP connection for ${connectionKey}`);\n // Update session data in Redis\n await updateSessionData(userId, accountId);\n totalReuseConnections++;\n console.log(`[IMAP] Successfully reused connection for ${connectionKey} in ${Date.now() - startTime}ms`);\n return connection.client;\n } else {\n console.log(`Existing connection for ${connectionKey} not usable, recreating`);\n // Will create a new connection below\n }\n } catch (error) {\n console.warn(`Error checking existing connection for ${connectionKey}:`, error);\n // Will create a new connection below\n }\n }\n // If we get here, we need a new connection\n console.log(`Creating new IMAP connection for ${connectionKey}`);\n // First try to get credentials from Redis cache\n let credentials = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.getCachedEmailCredentials)(userId, accountId);\n console.log(`Retrieved credentials from Redis cache for ${userId}:${accountId}:`, credentials ? {\n email: credentials.email,\n hasPassword: !!credentials.password,\n useOAuth: !!credentials.useOAuth,\n hasAccessToken: !!credentials.accessToken,\n hasRefreshToken: !!credentials.refreshToken\n } : 'No credentials found in cache');\n // If not in cache, get from database and cache them\n if (!credentials) {\n console.log(`Credentials not found in cache for ${userId}${accountId ? ` account ${accountId}` : ''}, attempting database lookup`);\n // Fetch directly from database\n const dbCredentials = await _lib_prisma__WEBPACK_IMPORTED_MODULE_5__.prisma.mailCredentials.findFirst({\n where: {\n AND: [\n {\n userId\n },\n accountId ? {\n id: accountId\n } : {}\n ]\n }\n });\n if (!dbCredentials) {\n console.error(`No credentials found for user ${userId}${accountId ? ` account ${accountId}` : ''}`);\n totalConnectionErrors++;\n throw new Error('Email account credentials not found');\n }\n console.log(`Database lookup returned credentials for ${dbCredentials.email}:`, {\n email: dbCredentials.email,\n hasPassword: !!dbCredentials.password,\n fields: Object.keys(dbCredentials)\n });\n // Create our credentials object from database data\n credentials = {\n email: dbCredentials.email,\n password: dbCredentials.password || '',\n host: dbCredentials.host,\n port: dbCredentials.port,\n secure: dbCredentials.secure,\n smtp_host: dbCredentials.smtp_host || undefined,\n smtp_port: dbCredentials.smtp_port || undefined,\n smtp_secure: dbCredentials.smtp_secure ?? false,\n display_name: dbCredentials.display_name || undefined,\n color: dbCredentials.color || undefined\n };\n }\n // Cast to extended type\n const extendedCreds = credentials;\n // MICROSOFT FIX: Detect Microsoft accounts by hostname and set OAuth flag\n if (extendedCreds.host === 'outlook.office365.com') {\n console.log(`Microsoft account detected (${extendedCreds.email}), setting useOAuth=true`);\n extendedCreds.useOAuth = true;\n // If we have no password but useOAuth is true, we need to make sure refresh token exists in Redis\n if (!extendedCreds.password && !extendedCreds.accessToken) {\n // If running in browser edge environment (serverless), try to refresh our tokens from Redis\n try {\n const cachedCreds = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.getCachedEmailCredentials)(userId, accountId);\n if (cachedCreds && cachedCreds.refreshToken) {\n console.log(`Found refresh token in Redis for ${extendedCreds.email}, will use it`);\n extendedCreds.refreshToken = cachedCreds.refreshToken;\n extendedCreds.accessToken = cachedCreds.accessToken;\n extendedCreds.tokenExpiry = cachedCreds.tokenExpiry;\n // Make sure we cache these credentials again with the tokens\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheEmailCredentials)(userId, accountId, extendedCreds);\n } else {\n console.warn(`No refresh token found for ${extendedCreds.email} in Redis cache`);\n }\n } catch (err) {\n console.error(`Error retrieving cached credentials for ${extendedCreds.email}:`, err);\n }\n }\n }\n // If using OAuth, ensure we have a fresh token\n if (extendedCreds.useOAuth) {\n console.log(`Account is configured to use OAuth`);\n if (!extendedCreds.accessToken) {\n console.error(`OAuth is enabled but no access token for account ${extendedCreds.email}`);\n }\n try {\n console.log(`Ensuring fresh token for OAuth account ${extendedCreds.email}`);\n const { accessToken, success } = await (0,_token_refresh__WEBPACK_IMPORTED_MODULE_8__.ensureFreshToken)(userId, extendedCreds.email);\n if (success && accessToken) {\n extendedCreds.accessToken = accessToken;\n console.log(`Successfully refreshed token for ${extendedCreds.email}`);\n } else {\n console.error(`Failed to refresh token for ${extendedCreds.email}`);\n }\n } catch (err) {\n console.error(`Error refreshing token for ${extendedCreds.email}:`, err);\n }\n }\n // Initialize connection tracking\n connectionPool[connectionKey] = {\n client: null,\n lastUsed: Date.now(),\n isConnecting: true,\n connectionAttempts: (connectionPool[connectionKey]?.connectionAttempts || 0) + 1\n };\n // PERFORMANCE FIX: Add connection timeout to prevent hanging connections\n let connectionTimeout = setTimeout(()=>{\n console.error(`[IMAP] Connection for ${connectionKey} timed out after 60 seconds`);\n if (connectionPool[connectionKey]?.isConnecting) {\n delete connectionPool[connectionKey];\n totalConnectionErrors++;\n }\n }, 60 * 1000); // 60 seconds timeout\n // Create connection promise using the extended credentials\n const connectionPromise = createImapConnection(extendedCreds, connectionKey).then((client)=>{\n // Update connection pool entry\n connectionPool[connectionKey].client = client;\n connectionPool[connectionKey].isConnecting = false;\n connectionPool[connectionKey].lastUsed = Date.now();\n // Clear timeout since connection was successful\n if (connectionTimeout) {\n clearTimeout(connectionTimeout);\n connectionTimeout = null;\n }\n // Update session data\n updateSessionData(userId, accountId).catch((err)=>{\n console.error(`Failed to update session data: ${err.message}`);\n });\n totalNewConnections++;\n console.log(`[IMAP] Created new connection for ${connectionKey} in ${Date.now() - startTime}ms (attempt #${connectionPool[connectionKey].connectionAttempts})`);\n return client;\n }).catch((error)=>{\n // Clear timeout to prevent double errors\n if (connectionTimeout) {\n clearTimeout(connectionTimeout);\n connectionTimeout = null;\n }\n // Handle connection error\n console.error(`Failed to create IMAP connection for ${connectionKey}:`, error);\n delete connectionPool[connectionKey];\n totalConnectionErrors++;\n throw error;\n });\n // Save the promise to allow other requests to wait for this connection\n connectionPool[connectionKey].connectionPromise = connectionPromise;\n return connectionPromise;\n}\n/**\n * Helper function to create a new IMAP connection\n */ async function createImapConnection(credentials, connectionKey) {\n // Cast to extended type\n const extendedCreds = credentials;\n console.log(`Creating IMAP connection with credentials:`, {\n email: extendedCreds.email,\n host: extendedCreds.host,\n port: extendedCreds.port,\n hasPassword: !!extendedCreds.password,\n useOAuth: !!extendedCreds.useOAuth,\n hasAccessToken: !!extendedCreds.accessToken,\n hasRefreshToken: !!extendedCreds.refreshToken,\n hasTokenExpiry: !!extendedCreds.tokenExpiry\n });\n let authParams;\n // Check if we have valid OAuth tokens\n if (extendedCreds.useOAuth && extendedCreds.accessToken) {\n console.log(`Using XOAUTH2 authentication for ${connectionKey} (OAuth enabled)`);\n // Set auth parameters for ImapFlow\n authParams = {\n user: extendedCreds.email,\n accessToken: extendedCreds.accessToken\n };\n console.log(`XOAUTH2 auth configured for ${connectionKey}`);\n } else if (extendedCreds.password) {\n // Use regular password authentication\n console.log(`Using password authentication for ${connectionKey} (OAuth not enabled or no token)`);\n authParams = {\n user: extendedCreds.email,\n pass: extendedCreds.password\n };\n } else {\n // No authentication method available\n console.error(`No authentication method found for ${connectionKey}:`, {\n hasPassword: !!extendedCreds.password,\n useOAuth: !!extendedCreds.useOAuth,\n hasAccessToken: !!extendedCreds.accessToken\n });\n throw new Error(`No authentication method available for ${connectionKey} - need either password or OAuth token`);\n }\n console.log(`Creating ImapFlow client for ${connectionKey} with authentication type: ${extendedCreds.useOAuth ? 'OAuth' : 'Password'}`);\n const client = new imapflow__WEBPACK_IMPORTED_MODULE_3__.ImapFlow({\n host: extendedCreds.host,\n port: extendedCreds.port,\n secure: extendedCreds.secure ?? true,\n auth: authParams,\n logger: false,\n emitLogs: false,\n tls: {\n rejectUnauthorized: false\n },\n disableAutoIdle: false\n });\n try {\n console.log(`Connecting to IMAP server: ${extendedCreds.host}:${extendedCreds.port}`);\n await client.connect();\n console.log(`Successfully connected to IMAP server for ${connectionKey}`);\n } catch (error) {\n console.error(`Failed to connect to IMAP server for ${connectionKey}:`, error);\n throw error;\n }\n // Add error handler\n client.on('error', (err)=>{\n console.error(`IMAP connection error for ${connectionKey}:`, err);\n // Remove from pool on error\n if (connectionPool[connectionKey]) {\n delete connectionPool[connectionKey];\n }\n });\n return client;\n}\n/**\n * Update session data in Redis\n */ async function updateSessionData(userId, accountId) {\n const sessionData = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.getCachedImapSession)(userId);\n if (sessionData) {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheImapSession)(userId, {\n ...sessionData,\n lastActive: Date.now(),\n ...accountId && {\n defaultAccountId: accountId\n }\n });\n } else {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheImapSession)(userId, {\n lastActive: Date.now(),\n ...accountId && {\n defaultAccountId: accountId\n }\n });\n }\n}\n/**\n * Get user's email credentials from database\n */ async function getUserEmailCredentials(userId, accountId) {\n const credentials = await _lib_prisma__WEBPACK_IMPORTED_MODULE_5__.prisma.mailCredentials.findFirst({\n where: {\n AND: [\n {\n userId\n },\n accountId ? {\n id: accountId\n } : {}\n ]\n }\n });\n if (!credentials) return null;\n const mailCredentials = credentials;\n return {\n email: mailCredentials.email,\n password: mailCredentials.password,\n host: mailCredentials.host,\n port: mailCredentials.port,\n secure: mailCredentials.secure,\n smtp_host: mailCredentials.smtp_host || undefined,\n smtp_port: mailCredentials.smtp_port || undefined,\n smtp_secure: mailCredentials.smtp_secure ?? false,\n display_name: mailCredentials.display_name || undefined,\n color: mailCredentials.color || undefined\n };\n}\n/**\n * Save or update user's email credentials\n */ async function saveUserEmailCredentials(userId, accountId, credentials) {\n console.log('Saving credentials for user:', userId, 'account:', accountId);\n if (!credentials) {\n throw new Error('No credentials provided');\n }\n // Cast to extended type to access OAuth properties\n const extendedCreds = credentials;\n // Store OAuth information in a separate object for caching\n const oauthData = {\n useOAuth: extendedCreds.useOAuth,\n accessToken: extendedCreds.accessToken,\n refreshToken: extendedCreds.refreshToken,\n tokenExpiry: extendedCreds.tokenExpiry\n };\n // Extract only the fields that exist in the database schema\n // Based on the schema from 'npx prisma db pull', OAuth fields don't exist\n const dbCredentials = {\n email: credentials.email,\n password: credentials.password ?? '',\n host: credentials.host,\n port: credentials.port,\n secure: credentials.secure ?? true,\n smtp_host: credentials.smtp_host || null,\n smtp_port: credentials.smtp_port || null,\n smtp_secure: credentials.smtp_secure ?? false,\n display_name: credentials.display_name || null,\n color: credentials.color || null\n };\n try {\n console.log('Saving credentials to database:', {\n ...dbCredentials,\n password: dbCredentials.password ? '***' : null\n });\n console.log('OAuth data will be saved to Redis cache only:', {\n hasOAuth: !!oauthData.useOAuth,\n hasAccessToken: !!oauthData.accessToken,\n hasRefreshToken: !!oauthData.refreshToken\n });\n // Save to database using the unique constraint on [userId, email]\n await _lib_prisma__WEBPACK_IMPORTED_MODULE_5__.prisma.mailCredentials.upsert({\n where: {\n id: await _lib_prisma__WEBPACK_IMPORTED_MODULE_5__.prisma.mailCredentials.findFirst({\n where: {\n AND: [\n {\n userId\n },\n {\n email: accountId\n }\n ]\n },\n select: {\n id: true\n }\n }).then((result)=>result?.id ?? '')\n },\n update: dbCredentials,\n create: {\n userId,\n ...dbCredentials\n }\n });\n // Create a combined credentials object for caching\n const fullCreds = {\n ...dbCredentials,\n ...oauthData\n }; // Cast to the expected type\n // Cache the full credentials including OAuth tokens\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheEmailCredentials)(userId, accountId, fullCreds);\n console.log('Successfully saved credentials to database and cached full data with OAuth tokens');\n } catch (error) {\n console.error('Error saving credentials:', error);\n throw error;\n }\n}\n/**\n * Get list of emails for a user\n */ async function getEmails(userId, folder, page = 1, perPage = 20, accountId, checkOnly = false) {\n // Normalize folder name and handle account ID\n console.log(`[getEmails] Processing request for folder: ${folder}, normalized to ${folder}, account: ${accountId || 'default'}, checkOnly: ${checkOnly}`);\n try {\n // The getImapConnection function already handles 'default' accountId by finding the first available account\n const client = await getImapConnection(userId, accountId);\n // At this point, accountId has been resolved to an actual account ID by getImapConnection\n // Store the resolved accountId in a variable that is guaranteed to be a string\n const resolvedAccountId = accountId || 'default';\n // Attempt to select the mailbox\n try {\n const mailboxInfo = await client.mailboxOpen(folder);\n console.log(`Opened mailbox ${folder} with ${mailboxInfo.exists} messages`);\n // Get list of all mailboxes for UI\n const mailboxes = await getMailboxes(client, resolvedAccountId);\n // Calculate pagination\n const totalEmails = mailboxInfo.exists || 0;\n const totalPages = Math.ceil(totalEmails / perPage);\n // Check if mailbox is empty\n if (totalEmails === 0) {\n // Cache the empty result\n const emptyResult = {\n emails: [],\n totalEmails: 0,\n page,\n perPage,\n totalPages: 0,\n folder,\n mailboxes,\n newestEmailId: 0\n };\n // Only cache if not in checkOnly mode\n if (!checkOnly) {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheEmailList)(userId, resolvedAccountId, folder, page, perPage, emptyResult);\n }\n return emptyResult;\n }\n // If checkOnly mode, we just fetch the most recent email's ID to compare\n if (checkOnly) {\n console.log(`[getEmails] checkOnly mode: fetching only the most recent email ID`);\n // Get the most recent message (highest sequence number)\n const lastMessageSequence = totalEmails.toString();\n console.log(`[getEmails] Fetching latest message with sequence: ${lastMessageSequence}`);\n const messages = await client.fetch(lastMessageSequence, {\n uid: true\n });\n let newestEmailId = 0;\n for await (const message of messages){\n newestEmailId = message.uid;\n }\n console.log(`[getEmails] Latest email UID: ${newestEmailId}`);\n // Return minimal result with just the newest email ID\n return {\n emails: [],\n totalEmails,\n page,\n perPage,\n totalPages,\n folder,\n mailboxes,\n newestEmailId\n };\n }\n // Calculate message range for pagination\n const start = Math.max(1, totalEmails - page * perPage + 1);\n const end = Math.max(1, totalEmails - (page - 1) * perPage);\n console.log(`Fetching messages ${start}:${end} from ${folder} for account ${resolvedAccountId}`);\n // Fetch messages\n const messages = await client.fetch(`${start}:${end}`, {\n envelope: true,\n flags: true,\n bodyStructure: true,\n uid: true\n });\n const emails = [];\n let newestEmailId = 0;\n for await (const message of messages){\n // Track the newest email ID (highest UID)\n if (message.uid > newestEmailId) {\n newestEmailId = message.uid;\n }\n const email = {\n id: message.uid.toString(),\n from: message.envelope.from?.map((addr)=>({\n name: addr.name || '',\n address: addr.address || ''\n })) || [],\n to: message.envelope.to?.map((addr)=>({\n name: addr.name || '',\n address: addr.address || ''\n })) || [],\n subject: message.envelope.subject || '',\n date: message.envelope.date || new Date(),\n flags: {\n seen: message.flags.has('\\\\Seen'),\n flagged: message.flags.has('\\\\Flagged'),\n answered: message.flags.has('\\\\Answered'),\n draft: message.flags.has('\\\\Draft'),\n deleted: message.flags.has('\\\\Deleted')\n },\n size: message.size || 0,\n hasAttachments: message.bodyStructure?.childNodes?.some((node)=>node.disposition === 'attachment') || false,\n folder: folder,\n contentFetched: false,\n accountId: resolvedAccountId,\n content: {\n text: '',\n html: '',\n isHtml: false,\n direction: 'ltr'\n }\n };\n emails.push(email);\n }\n // Prepare the result\n const result = {\n emails,\n totalEmails,\n page,\n perPage,\n totalPages: Math.ceil(totalEmails / perPage),\n folder,\n mailboxes,\n newestEmailId\n };\n // Cache the result with the effective account ID (only if not in checkOnly mode)\n if (!checkOnly) {\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheEmailList)(userId, resolvedAccountId, folder, page, perPage, result);\n }\n return result;\n } catch (error) {\n console.error('Error fetching emails:', error);\n throw error;\n }\n } catch (error) {\n console.error('Error fetching emails:', error);\n throw error;\n }\n}\n// Map email addresses safely with null checks\nfunction mapAddresses(addresses) {\n if (!addresses || !Array.isArray(addresses)) {\n return [];\n }\n return addresses.map((addr)=>({\n name: addr.name || addr.address || '',\n address: addr.address || ''\n }));\n}\n/**\n * Get a single email with full content\n */ async function getEmailContent(userId, emailId, folder = 'INBOX', accountId) {\n // Validate parameters\n if (!userId || !emailId || !folder) {\n throw new Error('Missing required parameters');\n }\n // Validate UID format\n if (!/^\\d+$/.test(emailId)) {\n throw new Error('Invalid email ID format: must be a numeric UID');\n }\n // Convert to number for IMAP\n const numericId = parseInt(emailId, 10);\n if (isNaN(numericId)) {\n throw new Error('Email ID must be a number');\n }\n // Extract account ID from folder name if present and none was explicitly provided\n const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n // Use the most specific account ID available\n const effectiveAccountId = folderAccountId || accountId || 'default';\n // Normalize folder name by removing account prefix if present\n const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n console.log(`[getEmailContent] Fetching email ${emailId} from folder ${normalizedFolder}, account ${effectiveAccountId}`);\n // Use normalized folder name and effective account ID for cache key\n const cachedEmail = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.getCachedEmailContent)(userId, effectiveAccountId, emailId);\n if (cachedEmail) {\n console.log(`Using cached email content for ${userId}:${effectiveAccountId}:${emailId}`);\n return cachedEmail;\n }\n console.log(`Cache miss for email content ${userId}:${effectiveAccountId}:${emailId}, fetching from IMAP`);\n const client = await getImapConnection(userId, effectiveAccountId);\n try {\n await client.mailboxOpen(normalizedFolder);\n // Log connection details with account context\n console.log(`[DEBUG] Fetching email ${emailId} from folder ${normalizedFolder} for account ${effectiveAccountId}`);\n // Open mailbox with error handling\n const mailbox = await client.mailboxOpen(normalizedFolder);\n if (!mailbox || typeof mailbox === 'boolean') {\n throw new Error(`Failed to open mailbox: ${normalizedFolder} for account ${effectiveAccountId}`);\n }\n // Log mailbox status with account context\n console.log(`[DEBUG] Mailbox ${normalizedFolder} opened for account ${effectiveAccountId}, total messages: ${mailbox.exists}`);\n // Get the UIDVALIDITY and UIDNEXT values\n const uidValidity = mailbox.uidValidity;\n const uidNext = mailbox.uidNext;\n console.log(`[DEBUG] Mailbox UIDVALIDITY: ${uidValidity}, UIDNEXT: ${uidNext} for account ${effectiveAccountId}`);\n // Validate UID exists in mailbox\n if (numericId >= uidNext) {\n throw new Error(`Email ID ${numericId} is greater than or equal to the highest UID in mailbox (${uidNext}) for account ${effectiveAccountId}`);\n }\n // First, try to get the sequence number for this UID\n const searchResult = await client.search({\n uid: numericId.toString()\n });\n if (!searchResult || searchResult.length === 0) {\n throw new Error(`Email with UID ${numericId} not found in folder ${normalizedFolder} for account ${effectiveAccountId}`);\n }\n const sequenceNumber = searchResult[0];\n console.log(`[DEBUG] Found sequence number ${sequenceNumber} for UID ${numericId} in account ${effectiveAccountId}`);\n // Now fetch using the sequence number with error handling\n let message;\n try {\n message = await client.fetchOne(sequenceNumber.toString(), {\n source: true,\n envelope: true,\n flags: true,\n size: true\n });\n } catch (fetchError) {\n console.error(`Error fetching message with sequence ${sequenceNumber}:`, fetchError);\n throw new Error(`Failed to fetch email: ${fetchError instanceof Error ? fetchError.message : 'Unknown error'}`);\n }\n if (!message) {\n throw new Error(`Email not found with sequence number ${sequenceNumber} in folder ${normalizedFolder} for account ${effectiveAccountId}`);\n }\n // Check if message has required fields\n if (!message.source || !message.envelope) {\n throw new Error(`Invalid email data received: missing source or envelope data`);\n }\n const { source, envelope, flags, size } = message;\n // Validate envelope data\n if (!envelope) {\n throw new Error('Email envelope data is missing');\n }\n // Parse the email content, ensuring all styles and structure are preserved\n let parsedEmail;\n try {\n parsedEmail = await (0,mailparser__WEBPACK_IMPORTED_MODULE_6__.simpleParser)(source.toString(), {\n skipHtmlToText: true,\n keepCidLinks: true\n });\n } catch (parseError) {\n console.error(`Error parsing email content for ${emailId}:`, parseError);\n throw new Error(`Failed to parse email content: ${parseError instanceof Error ? parseError.message : 'Unknown error'}`);\n }\n // Convert flags from Set to boolean checks\n const flagsArray = Array.from(flags);\n // Preserve the raw HTML exactly as it was in the original email\n const rawHtml = parsedEmail.html || '';\n const email = {\n id: emailId,\n messageId: envelope.messageId,\n subject: envelope.subject || \"(No Subject)\",\n from: mapAddresses(envelope.from),\n to: mapAddresses(envelope.to),\n cc: mapAddresses(envelope.cc),\n bcc: mapAddresses(envelope.bcc),\n date: envelope.date || new Date(),\n flags: {\n seen: flagsArray.includes(\"\\\\Seen\"),\n flagged: flagsArray.includes(\"\\\\Flagged\"),\n answered: flagsArray.includes(\"\\\\Answered\"),\n deleted: flagsArray.includes(\"\\\\Deleted\"),\n draft: flagsArray.includes(\"\\\\Draft\")\n },\n hasAttachments: parsedEmail.attachments?.length > 0,\n attachments: parsedEmail.attachments?.map((att)=>({\n filename: att.filename || 'attachment',\n contentType: att.contentType,\n size: att.size || 0\n })),\n content: {\n text: parsedEmail.text || '',\n html: rawHtml || '',\n isHtml: !!rawHtml,\n direction: 'ltr' // Default to left-to-right\n },\n folder: normalizedFolder,\n contentFetched: true,\n size: size || 0,\n accountId: effectiveAccountId\n };\n // Cache the email content with effective account ID\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.cacheEmailContent)(userId, effectiveAccountId, emailId, email);\n return email;\n } catch (error) {\n console.error('[ERROR] Email fetch failed:', {\n userId,\n emailId,\n folder: normalizedFolder,\n accountId: effectiveAccountId,\n error: error instanceof Error ? error.message : 'Unknown error',\n details: error instanceof Error ? error.stack : undefined\n });\n throw error;\n } finally{\n try {\n await client.mailboxClose();\n } catch (error) {\n console.error('Error closing mailbox:', error);\n }\n }\n}\n/**\n * Mark an email as read or unread\n */ async function markEmailReadStatus(userId, emailId, isRead, folder = 'INBOX', accountId) {\n // Extract account ID from folder name if present and none was explicitly provided\n const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n // Use the most specific account ID available\n const effectiveAccountId = folderAccountId || accountId || 'default';\n // Normalize folder name by removing account prefix if present\n const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n console.log(`[markEmailReadStatus] Marking email ${emailId} as ${isRead ? 'read' : 'unread'} in folder ${normalizedFolder}, account ${effectiveAccountId}`);\n const client = await getImapConnection(userId, effectiveAccountId);\n try {\n await client.mailboxOpen(normalizedFolder);\n if (isRead) {\n await client.messageFlagsAdd(emailId, [\n '\\\\Seen'\n ]);\n } else {\n await client.messageFlagsRemove(emailId, [\n '\\\\Seen'\n ]);\n }\n // Invalidate content cache since the flags changed\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.invalidateEmailContentCache)(userId, effectiveAccountId, emailId);\n // Also invalidate folder cache because unread counts may have changed\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.invalidateFolderCache)(userId, effectiveAccountId, normalizedFolder);\n return true;\n } catch (error) {\n console.error(`Error marking email ${emailId} as ${isRead ? 'read' : 'unread'} in folder ${normalizedFolder}, account ${effectiveAccountId}:`, error);\n return false;\n } finally{\n try {\n await client.mailboxClose();\n } catch (error) {\n console.error('Error closing mailbox:', error);\n }\n }\n}\n/**\n * Toggle an email's flagged (starred) status\n */ async function toggleEmailFlag(userId, emailId, flagged, folder = 'INBOX', accountId) {\n // Extract account ID from folder name if present and none was explicitly provided\n const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n // Use the most specific account ID available\n const effectiveAccountId = folderAccountId || accountId || 'default';\n // Normalize folder name by removing account prefix if present\n const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n console.log(`[toggleEmailFlag] Marking email ${emailId} as ${flagged ? 'flagged' : 'unflagged'} in folder ${normalizedFolder}, account ${effectiveAccountId}`);\n const client = await getImapConnection(userId, effectiveAccountId);\n try {\n await client.mailboxOpen(normalizedFolder);\n if (flagged) {\n await client.messageFlagsAdd(emailId, [\n '\\\\Flagged'\n ]);\n } else {\n await client.messageFlagsRemove(emailId, [\n '\\\\Flagged'\n ]);\n }\n // Invalidate content cache since the flags changed\n await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_7__.invalidateEmailContentCache)(userId, effectiveAccountId, emailId);\n return true;\n } catch (error) {\n console.error(`Error toggling flag for email ${emailId} in folder ${normalizedFolder}, account ${effectiveAccountId}:`, error);\n return false;\n } finally{\n try {\n await client.mailboxClose();\n } catch (error) {\n console.error('Error closing mailbox:', error);\n }\n }\n}\nasync function sendEmail(userId, emailData) {\n const credentials = await getUserEmailCredentials(userId);\n if (!credentials) {\n return {\n success: false,\n error: 'No email credentials found'\n };\n }\n // Cast to extended type\n const extendedCreds = credentials;\n // Configure SMTP auth based on OAuth or password\n const smtpAuth = extendedCreds.useOAuth && extendedCreds.accessToken ? {\n type: 'OAuth2',\n user: extendedCreds.email,\n accessToken: extendedCreds.accessToken\n } : {\n user: extendedCreds.email,\n pass: extendedCreds.password\n };\n // Create SMTP transporter with user's SMTP settings\n const transporter = nodemailer__WEBPACK_IMPORTED_MODULE_4__.createTransport({\n host: extendedCreds.smtp_host || 'smtp.infomaniak.com',\n port: extendedCreds.smtp_port || 587,\n secure: extendedCreds.smtp_secure || false,\n auth: smtpAuth,\n tls: {\n rejectUnauthorized: false\n }\n });\n try {\n const info = await transporter.sendMail({\n from: extendedCreds.email,\n to: emailData.to,\n cc: emailData.cc,\n bcc: emailData.bcc,\n subject: emailData.subject,\n text: emailData.body,\n html: emailData.body,\n attachments: emailData.attachments?.map((att)=>({\n filename: att.name,\n content: att.content,\n contentType: att.type\n }))\n });\n return {\n success: true,\n messageId: info.messageId\n };\n } catch (error) {\n console.error('Failed to send email:', error);\n return {\n success: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n };\n }\n}\n/**\n * Get list of mailboxes from an IMAP connection\n */ async function getMailboxes(client, accountId) {\n try {\n const mailboxes = await client.list();\n // If we have an accountId, prefix the folder names to prevent namespace collisions\n if (accountId) {\n return mailboxes.map((mailbox)=>`${accountId}:${mailbox.path}`);\n }\n // For backward compatibility, return unprefixed names when no accountId\n return mailboxes.map((mailbox)=>mailbox.path);\n } catch (error) {\n console.error('Error fetching mailboxes:', error);\n // Return empty array on error to avoid showing incorrect folders\n return [];\n }\n}\n/**\n * Test IMAP and SMTP connections for an email account\n */ async function testEmailConnection(credentials) {\n // Cast to extended type to use OAuth properties\n const extendedCreds = credentials;\n console.log('Testing connection with:', {\n ...extendedCreds,\n password: extendedCreds.password ? '***' : undefined,\n accessToken: extendedCreds.accessToken ? '***' : undefined,\n refreshToken: extendedCreds.refreshToken ? '***' : undefined\n });\n // Test IMAP connection\n try {\n console.log(`Testing IMAP connection to ${extendedCreds.host}:${extendedCreds.port} for ${extendedCreds.email}`);\n // Configure auth based on whether we're using OAuth or password\n let authParams;\n if (extendedCreds.useOAuth && extendedCreds.accessToken) {\n console.log('Using XOAUTH2 authentication mechanism');\n // For OAuth, pass the accessToken directly to ImapFlow\n authParams = {\n user: extendedCreds.email,\n accessToken: extendedCreds.accessToken\n };\n // Log the token length to verify it exists\n console.log(`Access token available (length: ${extendedCreds.accessToken.length})`);\n } else {\n console.log('Using password authentication mechanism');\n authParams = {\n user: extendedCreds.email,\n pass: extendedCreds.password\n };\n }\n const client = new imapflow__WEBPACK_IMPORTED_MODULE_3__.ImapFlow({\n host: extendedCreds.host,\n port: extendedCreds.port,\n secure: extendedCreds.secure ?? true,\n auth: authParams,\n logger: false,\n tls: {\n rejectUnauthorized: false\n }\n });\n console.log('Attempting to connect to IMAP server...');\n await client.connect();\n console.log('IMAP connection successful! Getting mailboxes...');\n const folders = await getMailboxes(client);\n await client.logout();\n console.log(`IMAP connection successful for ${extendedCreds.email}`);\n console.log(`Found ${folders.length} folders:`, folders);\n // Test SMTP connection if SMTP settings are provided\n let smtpSuccess = false;\n if (extendedCreds.smtp_host && extendedCreds.smtp_port) {\n try {\n console.log(`Testing SMTP connection to ${extendedCreds.smtp_host}:${extendedCreds.smtp_port}`);\n // Configure SMTP auth based on OAuth or password\n const smtpAuth = extendedCreds.useOAuth && extendedCreds.accessToken ? {\n type: 'OAuth2',\n user: extendedCreds.email,\n accessToken: extendedCreds.accessToken\n } : {\n user: extendedCreds.email,\n pass: extendedCreds.password\n };\n const transporter = nodemailer__WEBPACK_IMPORTED_MODULE_4__.createTransport({\n host: extendedCreds.smtp_host,\n port: extendedCreds.smtp_port,\n secure: extendedCreds.smtp_secure ?? false,\n auth: smtpAuth,\n tls: {\n rejectUnauthorized: false\n }\n });\n await transporter.verify();\n console.log(`SMTP connection successful for ${extendedCreds.email}`);\n smtpSuccess = true;\n } catch (smtpError) {\n console.error(`SMTP connection failed for ${extendedCreds.email}:`, smtpError);\n return {\n imap: true,\n smtp: false,\n error: `SMTP connection failed: ${smtpError instanceof Error ? smtpError.message : 'Unknown error'}`,\n folders\n };\n }\n }\n return {\n imap: true,\n smtp: smtpSuccess,\n folders\n };\n } catch (error) {\n console.error(`IMAP connection failed for ${extendedCreds.email}:`, error);\n return {\n imap: false,\n error: `IMAP connection failed: ${error instanceof Error ? error.message : 'Unknown error'}`\n };\n }\n}\n\n(0,private_next_rsc_action_validate__WEBPACK_IMPORTED_MODULE_9__.ensureServerEntryExports)([\n getImapConnection,\n getUserEmailCredentials,\n saveUserEmailCredentials,\n getEmails,\n getEmailContent,\n markEmailReadStatus,\n toggleEmailFlag,\n sendEmail,\n getMailboxes,\n testEmailConnection\n]);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(getImapConnection, \"60249f33dc41bab8693201a3f19f5e5fb46e641c28\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(getUserEmailCredentials, \"609e97c061f87c9d92e6b4c180de319248e8263787\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(saveUserEmailCredentials, \"70576421f3f1a8e0b47693f06bf0b900d321800592\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(getEmails, \"7e6fee2e8d6b5661c87219a81dea11090773f206cd\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(getEmailContent, \"78725b1539278f6847adbb3f678020efc4d204d6de\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(markEmailReadStatus, \"7c9e86d7555bbe449b808bd2ae7479f82af99409fe\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(toggleEmailFlag, \"7ccc49d190a944aede514b28ad88d4a52971336211\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(sendEmail, \"60852a4e84650a79792bf7dba8eab6e2994fad2674\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(getMailboxes, \"608647185521cdbec25a6e83fd03959d7becd6a6cd\", null);\n(0,private_next_rsc_server_reference__WEBPACK_IMPORTED_MODULE_0__.registerServerReference)(testEmailConnection, \"4010844c55b83dfb1f105bebefe0eefe1e6b431900\", null);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./lib/services/email-service.ts","mappings":"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","sources":["/home/alma/nextgen/Neah-mail/lib/services/email-service.ts"],"sourcesContent":["'use server';\n\nimport 'server-only';\nimport { ImapFlow } from 'imapflow';\nimport nodemailer from 'nodemailer';\nimport { prisma } from '@/lib/prisma';\nimport { simpleParser } from 'mailparser';\nimport { \n  cacheEmailCredentials, \n  getCachedEmailCredentials,\n  cacheEmailList,\n  getCachedEmailList,\n  cacheEmailContent,\n  getCachedEmailContent,\n  cacheImapSession,\n  getCachedImapSession,\n  invalidateFolderCache,\n  invalidateEmailContentCache\n} from '@/lib/redis';\nimport { EmailCredentials, EmailMessage, EmailAddress, EmailAttachment } from '@/lib/types';\nimport { ensureFreshToken } from './token-refresh';\nimport { createXOAuth2Token, refreshAccessToken as refreshMicrosoftAccessToken } from './microsoft-oauth';\nimport { MailCredentials } from '@prisma/client';\nimport Redis from 'ioredis';\nimport { getRedisClient } from '../redis';\n\n// Define EmailCredentials interface with OAuth properties\ninterface EmailCredentialsExtended extends EmailCredentials {\n  useOAuth?: boolean;\n  accessToken?: string;\n  refreshToken?: string;\n  tokenExpiry?: number;\n}\n\n// Define the extended MailCredentials type that includes OAuth fields\ninterface MailCredentialsWithOAuth extends MailCredentials {\n  useOAuth?: boolean;\n  accessToken?: string | null;\n  refreshToken?: string | null;\n  tokenExpiry?: Date | null;\n}\n\n// Types specific to this service\nexport interface EmailListResult {\n  emails: EmailMessage[];\n  totalEmails: number;\n  page: number;\n  perPage: number;\n  totalPages: number;\n  folder: string;\n  mailboxes: string[];\n  newestEmailId: number;\n}\n\n// Connection pool to reuse IMAP clients\nconst connectionPool: Record<string, { \n  client: ImapFlow; \n  lastUsed: number;\n  isConnecting: boolean;\n  connectionPromise?: Promise<ImapFlow>;\n  connectionAttempts?: number;\n}> = {};\n\n// Track overall connection metrics\nlet totalConnectionRequests = 0;\nlet totalNewConnections = 0;\nlet totalReuseConnections = 0;\nlet totalConnectionErrors = 0;\nlet lastMetricsReset = Date.now();\n\n// CRITICAL PERFORMANCE FIX: Increase idle timeout from 15 minutes to 30 minutes\n// This will keep connections alive longer and reduce reconnection delays\nconst CONNECTION_TIMEOUT = 30 * 60 * 1000;  // Increased to 30 minutes (was 15 minutes)\nconst MAX_POOL_SIZE = 20;  // Maximum number of connections to keep in the pool\nconst CONNECTION_CHECK_INTERVAL = 60 * 1000;  // Check every minute\nconst MIN_POOL_SIZE = 2;   // Keep at least this many active connections per user\n\n// Clean up idle connections periodically\nsetInterval(() => {\n  const now = Date.now();\n  const connectionKeys = Object.keys(connectionPool);\n  \n  // If we've been collecting metrics for more than an hour, log and reset\n  if (now - lastMetricsReset > 60 * 60 * 1000) {\n    console.log(`[IMAP METRICS] Total requests: ${totalConnectionRequests}, New connections: ${totalNewConnections}, Reused: ${totalReuseConnections}, Errors: ${totalConnectionErrors}, Success rate: ${((totalReuseConnections + totalNewConnections) / totalConnectionRequests * 100).toFixed(2)}%`);\n    totalConnectionRequests = 0;\n    totalNewConnections = 0;\n    totalReuseConnections = 0;\n    totalConnectionErrors = 0;\n    lastMetricsReset = now;\n  }\n  \n  // PERFORMANCE FIX: Group connections by user for better management\n  const connectionsByUser: Record<string, string[]> = {};\n  \n  connectionKeys.forEach(key => {\n    const userId = key.split(':')[0];\n    if (!connectionsByUser[userId]) {\n      connectionsByUser[userId] = [];\n    }\n    connectionsByUser[userId].push(key);\n  });\n  \n  // PERFORMANCE FIX: Manage pool size per user\n  Object.entries(connectionsByUser).forEach(([userId, userConnections]) => {\n    // Sort connections by last used (oldest first)\n    const sortedConnections = userConnections\n      .map(key => ({ key, lastUsed: connectionPool[key].lastUsed }))\n      .sort((a, b) => a.lastUsed - b.lastUsed);\n    \n    // Keep the most recently used connections up to the min pool size\n    const connectionsToKeep = sortedConnections.slice(-MIN_POOL_SIZE);\n    const keepKeys = new Set(connectionsToKeep.map(conn => conn.key));\n    \n    // Check the rest for idle timeout\n    sortedConnections.forEach(({ key, lastUsed }) => {\n      // Skip connections to keep and those that are in the process of connecting\n      if (keepKeys.has(key) || connectionPool[key].isConnecting) {\n        return;\n      }\n      \n      // Only close connections idle for too long\n      if (now - lastUsed > CONNECTION_TIMEOUT) {\n        console.log(`Closing idle IMAP connection for ${key} (idle for ${Math.round((now - lastUsed)/1000)}s)`);\n        try {\n          if (connectionPool[key].client.usable) {\n            connectionPool[key].client.logout().catch(err => {\n              console.error(`Error closing idle connection for ${key}:`, err);\n            });\n          }\n        } catch (error) {\n          console.error(`Error checking connection status for ${key}:`, error);\n        } finally {\n          delete connectionPool[key];\n          console.log(`Removed idle connection for ${key} from pool (pool size: ${Object.keys(connectionPool).length})`);\n        }\n      }\n    });\n  });\n  \n  // Log connection pool status with more details\n  const activeCount = connectionKeys.filter(key => {\n    const conn = connectionPool[key];\n    return !conn.isConnecting && (conn.client?.usable || false);\n  }).length;\n  \n  const connectingCount = connectionKeys.filter(key => connectionPool[key].isConnecting).length;\n  \n  console.log(`[IMAP POOL] Size: ${connectionKeys.length}, Active: ${activeCount}, Connecting: ${connectingCount}, Max: ${MAX_POOL_SIZE}`);\n}, CONNECTION_CHECK_INTERVAL);\n\n/**\n * Get IMAP connection for a user, reusing existing connections when possible\n * with improved connection handling and error recovery\n */\nexport async function getImapConnection(\n  userId: string,\n  accountId?: string\n): Promise<ImapFlow> {\n  const startTime = Date.now();\n  totalConnectionRequests++;\n  \n  console.log(`Getting IMAP connection for user ${userId}${accountId ? ` account ${accountId}` : ''}`);\n  \n  // Special handling for 'default' accountId - find the first available account\n  if (!accountId || accountId === 'default') {\n    console.log(`No specific account provided or 'default' requested, trying to find first account for user ${userId}`);\n    \n    // Try getting the account ID from cache to avoid database query\n    const sessionData = await getCachedImapSession(userId);\n    if (sessionData && sessionData.defaultAccountId) {\n      accountId = sessionData.defaultAccountId;\n      console.log(`Using cached default account ID: ${accountId}`);\n    } else {\n      // Query to find all accounts for this user\n      const accounts = await prisma.mailCredentials.findMany({\n        where: { userId },\n        orderBy: { createdAt: 'asc' },\n        take: 1\n      });\n      \n      if (accounts && accounts.length > 0) {\n        const firstAccount = accounts[0];\n        console.log(`Using first available account: ${firstAccount.id} (${firstAccount.email})`);\n        accountId = firstAccount.id;\n        \n        // Cache default account ID for future use\n        if (sessionData) {\n          await cacheImapSession(userId, {\n            ...sessionData,\n            defaultAccountId: accountId,\n            lastActive: Date.now()\n          });\n        } else {\n          await cacheImapSession(userId, {\n            lastActive: Date.now(),\n            defaultAccountId: accountId\n          });\n        }\n      } else {\n        totalConnectionErrors++;\n        throw new Error('No email accounts configured for this user');\n      }\n    }\n  }\n  \n  // Use accountId in connection key to ensure different accounts get different connections\n  const connectionKey = `${userId}:${accountId}`;\n  \n  // If we already have a connection for this key\n  if (connectionPool[connectionKey]) {\n    const connection = connectionPool[connectionKey];\n    \n    // If a connection is being established, wait for it\n    if (connection.isConnecting && connection.connectionPromise) {\n      console.log(`Connection in progress for ${connectionKey}, waiting for existing connection`);\n      try {\n        const client = await connection.connectionPromise;\n        connection.lastUsed = Date.now();\n        totalReuseConnections++;\n        console.log(`[IMAP] Reused pending connection for ${connectionKey} in ${Date.now() - startTime}ms`);\n        return client;\n      } catch (error) {\n        console.error(`Error waiting for connection for ${connectionKey}:`, error);\n        // Fall through to create new connection\n      }\n    }\n    \n    // Try to use existing connection if it's usable\n    try {\n      // PERFORMANCE FIX: More robust connection status checking\n      if (connection.client && connection.client.usable) {\n        // Touch the connection to mark it as recently used\n        connection.lastUsed = Date.now();\n        console.log(`Reusing existing IMAP connection for ${connectionKey}`);\n        \n        // Update session data in Redis\n        await updateSessionData(userId, accountId);\n        \n        totalReuseConnections++;\n        console.log(`[IMAP] Successfully reused connection for ${connectionKey} in ${Date.now() - startTime}ms`);\n        return connection.client;\n      } else {\n        console.log(`Existing connection for ${connectionKey} not usable, recreating`);\n        // Will create a new connection below\n      }\n    } catch (error) {\n      console.warn(`Error checking existing connection for ${connectionKey}:`, error);\n      // Will create a new connection below\n    }\n  }\n  \n  // If we get here, we need a new connection\n  console.log(`Creating new IMAP connection for ${connectionKey}`);\n  \n  // First try to get credentials from Redis cache\n  let credentials = await getCachedEmailCredentials(userId, accountId);\n  console.log(`Retrieved credentials from Redis cache for ${userId}:${accountId}:`, credentials ? {\n    email: credentials.email,\n    hasPassword: !!credentials.password,\n    useOAuth: !!credentials.useOAuth,\n    hasAccessToken: !!credentials.accessToken,\n    hasRefreshToken: !!credentials.refreshToken\n  } : 'No credentials found in cache');\n  \n  // If not in cache, get from database and cache them\n  if (!credentials) {\n    console.log(`Credentials not found in cache for ${userId}${accountId ? ` account ${accountId}` : ''}, attempting database lookup`);\n    \n    // Fetch directly from database\n    const dbCredentials = await prisma.mailCredentials.findFirst({\n      where: {\n        AND: [\n          { userId },\n          accountId ? { id: accountId } : {}\n        ]\n      }\n    });\n    \n    if (!dbCredentials) {\n      console.error(`No credentials found for user ${userId}${accountId ? ` account ${accountId}` : ''}`);\n      totalConnectionErrors++;\n      throw new Error('Email account credentials not found');\n    }\n    \n    console.log(`Database lookup returned credentials for ${dbCredentials.email}:`, {\n      email: dbCredentials.email,\n      hasPassword: !!dbCredentials.password,\n      fields: Object.keys(dbCredentials)\n    });\n    \n    // Create our credentials object from database data\n    credentials = {\n      email: dbCredentials.email,\n      password: dbCredentials.password || '',\n      host: dbCredentials.host,\n      port: dbCredentials.port,\n      secure: dbCredentials.secure,\n      smtp_host: dbCredentials.smtp_host || undefined,\n      smtp_port: dbCredentials.smtp_port || undefined,\n      smtp_secure: dbCredentials.smtp_secure ?? false,\n      display_name: dbCredentials.display_name || undefined,\n      color: dbCredentials.color || undefined\n    };\n  }\n  \n  // Cast to extended type\n  const extendedCreds = credentials as EmailCredentialsExtended;\n  \n  // MICROSOFT FIX: Detect Microsoft accounts by hostname and set OAuth flag\n  if (extendedCreds.host === 'outlook.office365.com') {\n    console.log(`Microsoft account detected (${extendedCreds.email}), setting useOAuth=true`);\n    extendedCreds.useOAuth = true;\n\n    // If we have no password but useOAuth is true, we need to make sure refresh token exists in Redis\n    if (!extendedCreds.password && !extendedCreds.accessToken) {\n      // If running in browser edge environment (serverless), try to refresh our tokens from Redis\n      try {\n        const cachedCreds = await getCachedEmailCredentials(userId, accountId);\n        if (cachedCreds && cachedCreds.refreshToken) {\n          console.log(`Found refresh token in Redis for ${extendedCreds.email}, will use it`);\n          extendedCreds.refreshToken = cachedCreds.refreshToken;\n          extendedCreds.accessToken = cachedCreds.accessToken;\n          extendedCreds.tokenExpiry = cachedCreds.tokenExpiry;\n          \n          // Make sure we cache these credentials again with the tokens\n          await cacheEmailCredentials(userId, accountId, extendedCreds);\n        } else {\n          console.warn(`No refresh token found for ${extendedCreds.email} in Redis cache`);\n        }\n      } catch (err) {\n        console.error(`Error retrieving cached credentials for ${extendedCreds.email}:`, err);\n      }\n    }\n  }\n  \n  // If using OAuth, ensure we have a fresh token\n  if (extendedCreds.useOAuth) {\n    console.log(`Account is configured to use OAuth`);\n    \n    if (!extendedCreds.accessToken) {\n      console.error(`OAuth is enabled but no access token for account ${extendedCreds.email}`);\n    }\n    \n    try {\n      console.log(`Ensuring fresh token for OAuth account ${extendedCreds.email}`);\n      const { accessToken, success } = await ensureFreshToken(userId, extendedCreds.email);\n      \n      if (success && accessToken) {\n        extendedCreds.accessToken = accessToken;\n        console.log(`Successfully refreshed token for ${extendedCreds.email}`);\n      } else {\n        console.error(`Failed to refresh token for ${extendedCreds.email}`);\n      }\n    } catch (err) {\n      console.error(`Error refreshing token for ${extendedCreds.email}:`, err);\n    }\n  }\n  \n  // Initialize connection tracking\n  connectionPool[connectionKey] = {\n    client: null as any,\n    lastUsed: Date.now(),\n    isConnecting: true,\n    connectionAttempts: (connectionPool[connectionKey]?.connectionAttempts || 0) + 1\n  };\n  \n  // PERFORMANCE FIX: Add connection timeout to prevent hanging connections\n  let connectionTimeout: NodeJS.Timeout | null = setTimeout(() => {\n    console.error(`[IMAP] Connection for ${connectionKey} timed out after 60 seconds`);\n    if (connectionPool[connectionKey]?.isConnecting) {\n      delete connectionPool[connectionKey];\n      totalConnectionErrors++;\n    }\n  }, 60 * 1000); // 60 seconds timeout\n  \n  // Create connection promise using the extended credentials\n  const connectionPromise = createImapConnection(extendedCreds, connectionKey)\n    .then(client => {\n      // Update connection pool entry\n      connectionPool[connectionKey].client = client;\n      connectionPool[connectionKey].isConnecting = false;\n      connectionPool[connectionKey].lastUsed = Date.now();\n      \n      // Clear timeout since connection was successful\n      if (connectionTimeout) {\n        clearTimeout(connectionTimeout);\n        connectionTimeout = null;\n      }\n      \n      // Update session data\n      updateSessionData(userId, accountId).catch(err => {\n        console.error(`Failed to update session data: ${err.message}`);\n      });\n      \n      totalNewConnections++;\n      console.log(`[IMAP] Created new connection for ${connectionKey} in ${Date.now() - startTime}ms (attempt #${connectionPool[connectionKey].connectionAttempts})`);\n      return client;\n    })\n    .catch(error => {\n      // Clear timeout to prevent double errors\n      if (connectionTimeout) {\n        clearTimeout(connectionTimeout);\n        connectionTimeout = null;\n      }\n      \n      // Handle connection error\n      console.error(`Failed to create IMAP connection for ${connectionKey}:`, error);\n      delete connectionPool[connectionKey];\n      totalConnectionErrors++;\n      throw error;\n    });\n  \n  // Save the promise to allow other requests to wait for this connection\n  connectionPool[connectionKey].connectionPromise = connectionPromise;\n  \n  return connectionPromise;\n}\n\n/**\n * Helper function to create a new IMAP connection\n */\nasync function createImapConnection(credentials: EmailCredentials, connectionKey: string): Promise<ImapFlow> {\n  // Cast to extended type\n  const extendedCreds = credentials as EmailCredentialsExtended;\n  \n  console.log(`Creating IMAP connection with credentials:`, {\n    email: extendedCreds.email,\n    host: extendedCreds.host,\n    port: extendedCreds.port,\n    hasPassword: !!extendedCreds.password,\n    useOAuth: !!extendedCreds.useOAuth,\n    hasAccessToken: !!extendedCreds.accessToken,\n    hasRefreshToken: !!extendedCreds.refreshToken,\n    hasTokenExpiry: !!extendedCreds.tokenExpiry\n  });\n  \n  let authParams: any;\n  \n  // Check if we have valid OAuth tokens\n  if (extendedCreds.useOAuth && extendedCreds.accessToken) {\n    console.log(`Using XOAUTH2 authentication for ${connectionKey} (OAuth enabled)`);\n    \n    // Set auth parameters for ImapFlow\n    authParams = {\n      user: extendedCreds.email,\n      accessToken: extendedCreds.accessToken\n    };\n    \n    console.log(`XOAUTH2 auth configured for ${connectionKey}`);\n  } else if (extendedCreds.password) {\n    // Use regular password authentication\n    console.log(`Using password authentication for ${connectionKey} (OAuth not enabled or no token)`);\n    authParams = {\n      user: extendedCreds.email,\n      pass: extendedCreds.password\n    };\n  } else {\n    // No authentication method available\n    console.error(`No authentication method found for ${connectionKey}:`, {\n      hasPassword: !!extendedCreds.password,\n      useOAuth: !!extendedCreds.useOAuth,\n      hasAccessToken: !!extendedCreds.accessToken\n    });\n    throw new Error(`No authentication method available for ${connectionKey} - need either password or OAuth token`);\n  }\n  \n  console.log(`Creating ImapFlow client for ${connectionKey} with authentication type: ${extendedCreds.useOAuth ? 'OAuth' : 'Password'}`);\n  \n  const client = new ImapFlow({\n    host: extendedCreds.host,\n    port: extendedCreds.port,\n    secure: extendedCreds.secure ?? true,\n    auth: authParams,\n    logger: false,\n    emitLogs: false,\n    tls: {\n      rejectUnauthorized: false\n    },\n    disableAutoIdle: false\n  });\n  \n  try {\n    console.log(`Connecting to IMAP server: ${extendedCreds.host}:${extendedCreds.port}`);\n    await client.connect();\n    console.log(`Successfully connected to IMAP server for ${connectionKey}`);\n  } catch (error) {\n    console.error(`Failed to connect to IMAP server for ${connectionKey}:`, error);\n    throw error;\n  }\n  \n  // Add error handler\n  client.on('error', (err) => {\n    console.error(`IMAP connection error for ${connectionKey}:`, err);\n    // Remove from pool on error\n    if (connectionPool[connectionKey]) {\n      delete connectionPool[connectionKey];\n    }\n  });\n  \n  return client;\n}\n\n/**\n * Update session data in Redis\n */\nasync function updateSessionData(userId: string, accountId?: string): Promise<void> {\n  const sessionData = await getCachedImapSession(userId);\n  \n  if (sessionData) {\n    await cacheImapSession(userId, {\n      ...sessionData,\n      lastActive: Date.now(),\n      ...(accountId && { defaultAccountId: accountId })\n    });\n  } else {\n    await cacheImapSession(userId, {\n      lastActive: Date.now(),\n      ...(accountId && { defaultAccountId: accountId })\n    });\n  }\n}\n\n/**\n * Get user's email credentials from database\n */\nexport async function getUserEmailCredentials(userId: string, accountId?: string): Promise<EmailCredentials | null> {\n  const credentials = await prisma.mailCredentials.findFirst({\n    where: {\n      AND: [\n        { userId },\n        accountId ? { id: accountId } : {}\n      ]\n    }\n  });\n\n  if (!credentials) return null;\n\n  const mailCredentials = credentials as unknown as {\n    email: string;\n    password: string;\n    host: string;\n    port: number;\n    secure: boolean;\n    smtp_host: string | null;\n    smtp_port: number | null;\n    smtp_secure: boolean | null;\n    display_name: string | null;\n    color: string | null;\n  };\n\n  return {\n    email: mailCredentials.email,\n    password: mailCredentials.password,\n    host: mailCredentials.host,\n    port: mailCredentials.port,\n    secure: mailCredentials.secure,\n    smtp_host: mailCredentials.smtp_host || undefined,\n    smtp_port: mailCredentials.smtp_port || undefined,\n    smtp_secure: mailCredentials.smtp_secure ?? false,\n    display_name: mailCredentials.display_name || undefined,\n    color: mailCredentials.color || undefined\n  };\n}\n\n/**\n * Save or update user's email credentials\n */\nexport async function saveUserEmailCredentials(\n  userId: string,\n  accountId: string,\n  credentials: EmailCredentials\n): Promise<void> {\n  console.log('Saving credentials for user:', userId, 'account:', accountId);\n  \n  if (!credentials) {\n    throw new Error('No credentials provided');\n  }\n\n  // Cast to extended type to access OAuth properties\n  const extendedCreds = credentials as EmailCredentialsExtended;\n  \n  // Store OAuth information in a separate object for caching\n  const oauthData = {\n    useOAuth: extendedCreds.useOAuth,\n    accessToken: extendedCreds.accessToken,\n    refreshToken: extendedCreds.refreshToken,\n    tokenExpiry: extendedCreds.tokenExpiry\n  };\n\n  // Extract only the fields that exist in the database schema\n  // Based on the schema from 'npx prisma db pull', OAuth fields don't exist\n  const dbCredentials = {\n    email: credentials.email,\n    password: credentials.password ?? '', // Required field in the DB schema\n    host: credentials.host,\n    port: credentials.port,\n    secure: credentials.secure ?? true,\n    smtp_host: credentials.smtp_host || null,\n    smtp_port: credentials.smtp_port || null,\n    smtp_secure: credentials.smtp_secure ?? false,\n    display_name: credentials.display_name || null,\n    color: credentials.color || null\n  };\n  \n  try {\n    console.log('Saving credentials to database:', {\n      ...dbCredentials,\n      password: dbCredentials.password ? '***' : null,\n    });\n    \n    console.log('OAuth data will be saved to Redis cache only:', {\n      hasOAuth: !!oauthData.useOAuth,\n      hasAccessToken: !!oauthData.accessToken,\n      hasRefreshToken: !!oauthData.refreshToken\n    });\n\n    // Save to database using the unique constraint on [userId, email]\n    await prisma.mailCredentials.upsert({\n      where: {\n        id: await prisma.mailCredentials.findFirst({\n          where: {\n            AND: [\n              { userId },\n              { email: accountId }\n            ]\n          },\n          select: { id: true }\n        }).then(result => result?.id ?? '')\n      },\n      update: dbCredentials,\n      create: {\n        userId,\n        ...dbCredentials\n      }\n    });\n    \n    // Create a combined credentials object for caching\n    const fullCreds = {\n      ...dbCredentials,\n      ...oauthData\n    } as EmailCredentialsExtended;  // Cast to the expected type\n    \n    // Cache the full credentials including OAuth tokens\n    await cacheEmailCredentials(userId, accountId, fullCreds);\n    console.log('Successfully saved credentials to database and cached full data with OAuth tokens');\n  } catch (error) {\n    console.error('Error saving credentials:', error);\n    throw error;\n  }\n}\n\n// Helper type for IMAP fetch options\ninterface FetchOptions {\n  envelope: boolean;\n  flags: boolean;\n  bodyStructure: boolean;\n  internalDate: boolean;\n  size: boolean;\n  bodyParts: { part: string; query: any; limit?: number }[];\n}\n\n/**\n * Get list of emails for a user\n */\nexport async function getEmails(\n  userId: string,\n  folder: string,\n  page: number = 1,\n  perPage: number = 20,\n  accountId?: string,\n  checkOnly: boolean = false\n): Promise<EmailListResult> {\n  // Normalize folder name and handle account ID\n  console.log(`[getEmails] Processing request for folder: ${folder}, normalized to ${folder}, account: ${accountId || 'default'}, checkOnly: ${checkOnly}`);\n  \n  try {\n    // The getImapConnection function already handles 'default' accountId by finding the first available account\n    const client = await getImapConnection(userId, accountId);\n    \n    // At this point, accountId has been resolved to an actual account ID by getImapConnection\n    // Store the resolved accountId in a variable that is guaranteed to be a string\n    const resolvedAccountId = accountId || 'default';\n    \n    // Attempt to select the mailbox\n    try {\n      const mailboxInfo = await client.mailboxOpen(folder);\n      console.log(`Opened mailbox ${folder} with ${mailboxInfo.exists} messages`);\n      \n      // Get list of all mailboxes for UI\n      const mailboxes = await getMailboxes(client, resolvedAccountId);\n      \n      // Calculate pagination\n      const totalEmails = mailboxInfo.exists || 0;\n      const totalPages = Math.ceil(totalEmails / perPage);\n      \n      // Check if mailbox is empty\n      if (totalEmails === 0) {\n        // Cache the empty result\n        const emptyResult = {\n          emails: [],\n          totalEmails: 0,\n          page,\n          perPage,\n          totalPages: 0,\n          folder,\n          mailboxes,\n          newestEmailId: 0\n        };\n        \n        // Only cache if not in checkOnly mode\n        if (!checkOnly) {\n          await cacheEmailList(\n            userId,\n            resolvedAccountId, // Use the guaranteed string account ID\n            folder,\n            page,\n            perPage,\n            emptyResult\n          );\n        }\n        \n        return emptyResult;\n      }\n\n      // If checkOnly mode, we just fetch the most recent email's ID to compare\n      if (checkOnly) {\n        console.log(`[getEmails] checkOnly mode: fetching only the most recent email ID`);\n        \n        // Get the most recent message (highest sequence number)\n        const lastMessageSequence = totalEmails.toString();\n        console.log(`[getEmails] Fetching latest message with sequence: ${lastMessageSequence}`);\n        \n        const messages = await client.fetch(lastMessageSequence, {\n          uid: true\n        });\n        \n        let newestEmailId = 0;\n        for await (const message of messages) {\n          newestEmailId = message.uid;\n        }\n        \n        console.log(`[getEmails] Latest email UID: ${newestEmailId}`);\n        \n        // Return minimal result with just the newest email ID\n        return {\n          emails: [],\n          totalEmails,\n          page,\n          perPage,\n          totalPages,\n          folder,\n          mailboxes,\n          newestEmailId\n        };\n      }\n\n      // Calculate message range for pagination\n      const start = Math.max(1, totalEmails - (page * perPage) + 1);\n      const end = Math.max(1, totalEmails - ((page - 1) * perPage));\n      console.log(`Fetching messages ${start}:${end} from ${folder} for account ${resolvedAccountId}`);\n\n      // Fetch messages\n      const messages = await client.fetch(`${start}:${end}`, {\n        envelope: true,\n        flags: true,\n        bodyStructure: true,\n        uid: true\n      });\n\n      const emails: EmailMessage[] = [];\n      let newestEmailId = 0;\n      \n      for await (const message of messages) {\n        // Track the newest email ID (highest UID)\n        if (message.uid > newestEmailId) {\n          newestEmailId = message.uid;\n        }\n        \n        const email: EmailMessage = {\n          id: message.uid.toString(),\n          from: message.envelope.from?.map(addr => ({\n            name: addr.name || '',\n            address: addr.address || ''\n          })) || [],\n          to: message.envelope.to?.map(addr => ({\n            name: addr.name || '',\n            address: addr.address || ''\n          })) || [],\n          subject: message.envelope.subject || '',\n          date: message.envelope.date || new Date(),\n          flags: {\n            seen: message.flags.has('\\\\Seen'),\n            flagged: message.flags.has('\\\\Flagged'),\n            answered: message.flags.has('\\\\Answered'),\n            draft: message.flags.has('\\\\Draft'),\n            deleted: message.flags.has('\\\\Deleted')\n          },\n          size: message.size || 0,\n          hasAttachments: message.bodyStructure?.childNodes?.some(node => node.disposition === 'attachment') || false,\n          folder: folder,\n          contentFetched: false,\n          accountId: resolvedAccountId,\n          content: {\n            text: '',\n            html: '',\n            isHtml: false,\n            direction: 'ltr'\n          }\n        };\n        emails.push(email);\n      }\n\n      // Prepare the result\n      const result = {\n        emails,\n        totalEmails,\n        page,\n        perPage,\n        totalPages: Math.ceil(totalEmails / perPage),\n        folder,\n        mailboxes,\n        newestEmailId\n      };\n      \n      // Cache the result with the effective account ID (only if not in checkOnly mode)\n      if (!checkOnly) {\n        await cacheEmailList(\n          userId,\n          resolvedAccountId, // Use the guaranteed string account ID\n          folder,\n          page,\n          perPage,\n          result\n        );\n      }\n\n      return result;\n    } catch (error) {\n      console.error('Error fetching emails:', error);\n      throw error;\n    }\n  } catch (error) {\n    console.error('Error fetching emails:', error);\n    throw error;\n  }\n}\n\n// Map email addresses safely with null checks\nfunction mapAddresses(addresses: any[] | undefined): Array<{ name: string; address: string }> {\n  if (!addresses || !Array.isArray(addresses)) {\n    return [];\n  }\n  \n  return addresses.map((addr: any) => ({\n    name: addr.name || addr.address || '',\n    address: addr.address || ''\n  }));\n}\n\n/**\n * Get a single email with full content\n */\nexport async function getEmailContent(\n  userId: string,\n  emailId: string,\n  folder: string = 'INBOX',\n  accountId?: string\n): Promise<EmailMessage> {\n  // Validate parameters\n  if (!userId || !emailId || !folder) {\n    throw new Error('Missing required parameters');\n  }\n\n  // Validate UID format\n  if (!/^\\d+$/.test(emailId)) {\n    throw new Error('Invalid email ID format: must be a numeric UID');\n  }\n\n  // Convert to number for IMAP\n  const numericId = parseInt(emailId, 10);\n  if (isNaN(numericId)) {\n    throw new Error('Email ID must be a number');\n  }\n\n  // Extract account ID from folder name if present and none was explicitly provided\n  const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n  \n  // Use the most specific account ID available\n  const effectiveAccountId = folderAccountId || accountId || 'default';\n  \n  // Normalize folder name by removing account prefix if present\n  const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n  \n  console.log(`[getEmailContent] Fetching email ${emailId} from folder ${normalizedFolder}, account ${effectiveAccountId}`);\n  \n  // Use normalized folder name and effective account ID for cache key\n  const cachedEmail = await getCachedEmailContent(userId, effectiveAccountId, emailId);\n  if (cachedEmail) {\n    console.log(`Using cached email content for ${userId}:${effectiveAccountId}:${emailId}`);\n    return cachedEmail;\n  }\n  \n  console.log(`Cache miss for email content ${userId}:${effectiveAccountId}:${emailId}, fetching from IMAP`);\n  \n  const client = await getImapConnection(userId, effectiveAccountId);\n  \n  try {\n    await client.mailboxOpen(normalizedFolder);\n    \n    // Log connection details with account context\n    console.log(`[DEBUG] Fetching email ${emailId} from folder ${normalizedFolder} for account ${effectiveAccountId}`);\n    \n    // Open mailbox with error handling\n    const mailbox = await client.mailboxOpen(normalizedFolder);\n    if (!mailbox || typeof mailbox === 'boolean') {\n      throw new Error(`Failed to open mailbox: ${normalizedFolder} for account ${effectiveAccountId}`);\n    }\n    \n    // Log mailbox status with account context\n    console.log(`[DEBUG] Mailbox ${normalizedFolder} opened for account ${effectiveAccountId}, total messages: ${mailbox.exists}`);\n    \n    // Get the UIDVALIDITY and UIDNEXT values\n    const uidValidity = mailbox.uidValidity;\n    const uidNext = mailbox.uidNext;\n    \n    console.log(`[DEBUG] Mailbox UIDVALIDITY: ${uidValidity}, UIDNEXT: ${uidNext} for account ${effectiveAccountId}`);\n    \n    // Validate UID exists in mailbox\n    if (numericId >= uidNext) {\n      throw new Error(`Email ID ${numericId} is greater than or equal to the highest UID in mailbox (${uidNext}) for account ${effectiveAccountId}`);\n    }\n    \n    // First, try to get the sequence number for this UID\n    const searchResult = await client.search({ uid: numericId.toString() });\n    if (!searchResult || searchResult.length === 0) {\n      throw new Error(`Email with UID ${numericId} not found in folder ${normalizedFolder} for account ${effectiveAccountId}`);\n    }\n    \n    const sequenceNumber = searchResult[0];\n    console.log(`[DEBUG] Found sequence number ${sequenceNumber} for UID ${numericId} in account ${effectiveAccountId}`);\n    \n    // Now fetch using the sequence number with error handling\n    let message;\n    try {\n      message = await client.fetchOne(sequenceNumber.toString(), {\n        source: true,\n        envelope: true,\n        flags: true,\n        size: true\n      });\n    } catch (fetchError) {\n      console.error(`Error fetching message with sequence ${sequenceNumber}:`, fetchError);\n      throw new Error(`Failed to fetch email: ${fetchError instanceof Error ? fetchError.message : 'Unknown error'}`);\n    }\n    \n    if (!message) {\n      throw new Error(`Email not found with sequence number ${sequenceNumber} in folder ${normalizedFolder} for account ${effectiveAccountId}`);\n    }\n    \n    // Check if message has required fields\n    if (!message.source || !message.envelope) {\n      throw new Error(`Invalid email data received: missing source or envelope data`);\n    }\n    \n    const { source, envelope, flags, size } = message;\n    \n    // Validate envelope data\n    if (!envelope) {\n      throw new Error('Email envelope data is missing');\n    }\n    \n    // Parse the email content, ensuring all styles and structure are preserved\n    let parsedEmail;\n    try {\n      parsedEmail = await simpleParser(source.toString(), {\n        skipHtmlToText: true,\n        keepCidLinks: true\n      });\n    } catch (parseError) {\n      console.error(`Error parsing email content for ${emailId}:`, parseError);\n      throw new Error(`Failed to parse email content: ${parseError instanceof Error ? parseError.message : 'Unknown error'}`);\n    }\n    \n    // Convert flags from Set to boolean checks\n    const flagsArray = Array.from(flags as Set<string>);\n    \n    // Preserve the raw HTML exactly as it was in the original email\n    const rawHtml = parsedEmail.html || '';\n    \n    const email: EmailMessage = {\n      id: emailId,\n      messageId: envelope.messageId,\n      subject: envelope.subject || \"(No Subject)\",\n      from: mapAddresses(envelope.from),\n      to: mapAddresses(envelope.to),\n      cc: mapAddresses(envelope.cc),\n      bcc: mapAddresses(envelope.bcc),\n      date: envelope.date || new Date(),\n      flags: {\n        seen: flagsArray.includes(\"\\\\Seen\"),\n        flagged: flagsArray.includes(\"\\\\Flagged\"),\n        answered: flagsArray.includes(\"\\\\Answered\"),\n        deleted: flagsArray.includes(\"\\\\Deleted\"),\n        draft: flagsArray.includes(\"\\\\Draft\"),\n      },\n      hasAttachments: parsedEmail.attachments?.length > 0,\n      attachments: parsedEmail.attachments?.map(att => ({\n        filename: att.filename || 'attachment',\n        contentType: att.contentType,\n        size: att.size || 0\n      })),\n      content: {\n        text: parsedEmail.text || '',\n        html: rawHtml || '',\n        isHtml: !!rawHtml,\n        direction: 'ltr' // Default to left-to-right\n      },\n      folder: normalizedFolder,\n      contentFetched: true,\n      size: size || 0,\n      accountId: effectiveAccountId\n    };\n    \n    // Cache the email content with effective account ID\n    await cacheEmailContent(userId, effectiveAccountId, emailId, email);\n    \n    return email;\n  } catch (error) {\n    console.error('[ERROR] Email fetch failed:', {\n      userId,\n      emailId,\n      folder: normalizedFolder,\n      accountId: effectiveAccountId,\n      error: error instanceof Error ? error.message : 'Unknown error',\n      details: error instanceof Error ? error.stack : undefined\n    });\n    throw error;\n  } finally {\n    try {\n      await client.mailboxClose();\n    } catch (error) {\n      console.error('Error closing mailbox:', error);\n    }\n  }\n}\n\n/**\n * Mark an email as read or unread\n */\nexport async function markEmailReadStatus(\n  userId: string,\n  emailId: string,\n  isRead: boolean,\n  folder: string = 'INBOX',\n  accountId?: string\n): Promise<boolean> {\n  // Extract account ID from folder name if present and none was explicitly provided\n  const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n  \n  // Use the most specific account ID available\n  const effectiveAccountId = folderAccountId || accountId || 'default';\n  \n  // Normalize folder name by removing account prefix if present\n  const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n  \n  console.log(`[markEmailReadStatus] Marking email ${emailId} as ${isRead ? 'read' : 'unread'} in folder ${normalizedFolder}, account ${effectiveAccountId}`);\n  \n  const client = await getImapConnection(userId, effectiveAccountId);\n  \n  try {\n    await client.mailboxOpen(normalizedFolder);\n    \n    if (isRead) {\n      await client.messageFlagsAdd(emailId, ['\\\\Seen']);\n    } else {\n      await client.messageFlagsRemove(emailId, ['\\\\Seen']);\n    }\n    \n    // Invalidate content cache since the flags changed\n    await invalidateEmailContentCache(userId, effectiveAccountId, emailId);\n    \n    // Also invalidate folder cache because unread counts may have changed\n    await invalidateFolderCache(userId, effectiveAccountId, normalizedFolder);\n    \n    return true;\n  } catch (error) {\n    console.error(`Error marking email ${emailId} as ${isRead ? 'read' : 'unread'} in folder ${normalizedFolder}, account ${effectiveAccountId}:`, error);\n    return false;\n  } finally {\n    try {\n      await client.mailboxClose();\n    } catch (error) {\n      console.error('Error closing mailbox:', error);\n    }\n  }\n}\n\n/**\n * Toggle an email's flagged (starred) status\n */\nexport async function toggleEmailFlag(\n  userId: string,\n  emailId: string,\n  flagged: boolean,\n  folder: string = 'INBOX',\n  accountId?: string\n): Promise<boolean> {\n  // Extract account ID from folder name if present and none was explicitly provided\n  const folderAccountId = folder.includes(':') ? folder.split(':')[0] : accountId;\n  \n  // Use the most specific account ID available\n  const effectiveAccountId = folderAccountId || accountId || 'default';\n  \n  // Normalize folder name by removing account prefix if present\n  const normalizedFolder = folder.includes(':') ? folder.split(':')[1] : folder;\n  \n  console.log(`[toggleEmailFlag] Marking email ${emailId} as ${flagged ? 'flagged' : 'unflagged'} in folder ${normalizedFolder}, account ${effectiveAccountId}`);\n  \n  const client = await getImapConnection(userId, effectiveAccountId);\n  \n  try {\n    await client.mailboxOpen(normalizedFolder);\n    \n    if (flagged) {\n      await client.messageFlagsAdd(emailId, ['\\\\Flagged']);\n    } else {\n      await client.messageFlagsRemove(emailId, ['\\\\Flagged']);\n    }\n    \n    // Invalidate content cache since the flags changed\n    await invalidateEmailContentCache(userId, effectiveAccountId, emailId);\n    \n    return true;\n  } catch (error) {\n    console.error(`Error toggling flag for email ${emailId} in folder ${normalizedFolder}, account ${effectiveAccountId}:`, error);\n    return false;\n  } finally {\n    try {\n      await client.mailboxClose();\n    } catch (error) {\n      console.error('Error closing mailbox:', error);\n    }\n  }\n}\n\n// Define EmailContent interface\ninterface EmailContent {\n  to: string;\n  cc?: string;\n  bcc?: string;\n  subject: string;\n  plainText: string;\n  htmlContent: string;\n  attachments?: Array<{\n    filename: string;\n    content: string;\n    contentType: string;\n  }>;\n}\n\nexport async function sendEmail(\n  userId: string,\n  emailData: {\n    to: string;\n    cc?: string;\n    bcc?: string;\n    subject: string;\n    body: string;\n    attachments?: Array<{\n      name: string;\n      content: string;\n      type: string;\n    }>;\n  }\n): Promise<{ success: boolean; messageId?: string; error?: string }> {\n  const credentials = await getUserEmailCredentials(userId);\n  \n  if (!credentials) {\n    return { \n      success: false, \n      error: 'No email credentials found' \n    };\n  }\n\n  // Cast to extended type\n  const extendedCreds = credentials as EmailCredentialsExtended;\n  \n  // Configure SMTP auth based on OAuth or password\n  const smtpAuth = extendedCreds.useOAuth && extendedCreds.accessToken\n    ? {\n        type: 'OAuth2',\n        user: extendedCreds.email,\n        accessToken: extendedCreds.accessToken\n      }\n    : {\n        user: extendedCreds.email,\n        pass: extendedCreds.password\n      };\n  \n  // Create SMTP transporter with user's SMTP settings\n  const transporter = nodemailer.createTransport({\n    host: extendedCreds.smtp_host || 'smtp.infomaniak.com',\n    port: extendedCreds.smtp_port || 587,\n    secure: extendedCreds.smtp_secure || false,\n    auth: smtpAuth,\n    tls: {\n      rejectUnauthorized: false\n    }\n  } as nodemailer.TransportOptions);\n\n  try {\n    const info = await transporter.sendMail({\n      from: extendedCreds.email,\n      to: emailData.to,\n      cc: emailData.cc,\n      bcc: emailData.bcc,\n      subject: emailData.subject,\n      text: emailData.body,\n      html: emailData.body,\n      attachments: emailData.attachments?.map(att => ({\n        filename: att.name,\n        content: att.content,\n        contentType: att.type\n      })),\n    });\n\n    return {\n      success: true,\n      messageId: info.messageId\n    };\n  } catch (error) {\n    console.error('Failed to send email:', error);\n    return {\n      success: false,\n      error: error instanceof Error ? error.message : 'Unknown error'\n    };\n  }\n}\n\n/**\n * Get list of mailboxes from an IMAP connection\n */\nexport async function getMailboxes(client: ImapFlow, accountId?: string): Promise<string[]> {\n  try {\n    const mailboxes = await client.list();\n    \n    // If we have an accountId, prefix the folder names to prevent namespace collisions\n    if (accountId) {\n      return mailboxes.map(mailbox => `${accountId}:${mailbox.path}`);\n    }\n    \n    // For backward compatibility, return unprefixed names when no accountId\n    return mailboxes.map(mailbox => mailbox.path);\n  } catch (error) {\n    console.error('Error fetching mailboxes:', error);\n    // Return empty array on error to avoid showing incorrect folders\n    return [];\n  }\n}\n\n/**\n * Test IMAP and SMTP connections for an email account\n */\nexport async function testEmailConnection(credentials: EmailCredentials): Promise<{\n  imap: boolean;\n  smtp?: boolean;\n  error?: string;\n  folders?: string[];\n}> {\n  // Cast to extended type to use OAuth properties\n  const extendedCreds = credentials as EmailCredentialsExtended;\n  \n  console.log('Testing connection with:', {\n    ...extendedCreds,\n    password: extendedCreds.password ? '***' : undefined,\n    accessToken: extendedCreds.accessToken ? '***' : undefined,\n    refreshToken: extendedCreds.refreshToken ? '***' : undefined\n  });\n\n  // Test IMAP connection\n  try {\n    console.log(`Testing IMAP connection to ${extendedCreds.host}:${extendedCreds.port} for ${extendedCreds.email}`);\n    \n    // Configure auth based on whether we're using OAuth or password\n    let authParams: any;\n    \n    if (extendedCreds.useOAuth && extendedCreds.accessToken) {\n      console.log('Using XOAUTH2 authentication mechanism');\n      \n      // For OAuth, pass the accessToken directly to ImapFlow\n      authParams = {\n        user: extendedCreds.email,\n        accessToken: extendedCreds.accessToken\n      };\n      \n      // Log the token length to verify it exists\n      console.log(`Access token available (length: ${extendedCreds.accessToken.length})`);\n    } else {\n      console.log('Using password authentication mechanism');\n      authParams = {\n        user: extendedCreds.email,\n        pass: extendedCreds.password\n      };\n    }\n    \n    const client = new ImapFlow({\n      host: extendedCreds.host,\n      port: extendedCreds.port,\n      secure: extendedCreds.secure ?? true,\n      auth: authParams,\n      logger: false,\n      tls: {\n        rejectUnauthorized: false\n      }\n    });\n\n    console.log('Attempting to connect to IMAP server...');\n    await client.connect();\n    console.log('IMAP connection successful! Getting mailboxes...');\n    \n    const folders = await getMailboxes(client);\n    await client.logout();\n\n    console.log(`IMAP connection successful for ${extendedCreds.email}`);\n    console.log(`Found ${folders.length} folders:`, folders);\n\n    // Test SMTP connection if SMTP settings are provided\n    let smtpSuccess = false;\n    if (extendedCreds.smtp_host && extendedCreds.smtp_port) {\n      try {\n        console.log(`Testing SMTP connection to ${extendedCreds.smtp_host}:${extendedCreds.smtp_port}`);\n        \n        // Configure SMTP auth based on OAuth or password\n        const smtpAuth = extendedCreds.useOAuth && extendedCreds.accessToken\n          ? {\n              type: 'OAuth2',\n              user: extendedCreds.email,\n              accessToken: extendedCreds.accessToken\n            }\n          : {\n              user: extendedCreds.email,\n              pass: extendedCreds.password,\n            };\n        \n        const transporter = nodemailer.createTransport({\n          host: extendedCreds.smtp_host,\n          port: extendedCreds.smtp_port,\n          secure: extendedCreds.smtp_secure ?? false,\n          auth: smtpAuth,\n          tls: {\n            rejectUnauthorized: false\n          }\n        } as nodemailer.TransportOptions);\n\n        await transporter.verify();\n        console.log(`SMTP connection successful for ${extendedCreds.email}`);\n        smtpSuccess = true;\n      } catch (smtpError) {\n        console.error(`SMTP connection failed for ${extendedCreds.email}:`, smtpError);\n        return {\n          imap: true,\n          smtp: false,\n          error: `SMTP connection failed: ${smtpError instanceof Error ? smtpError.message : 'Unknown error'}`,\n          folders\n        };\n      }\n    }\n\n    return {\n      imap: true,\n      smtp: smtpSuccess,\n      folders\n    };\n  } catch (error) {\n    console.error(`IMAP connection failed for ${extendedCreds.email}:`, error);\n    return {\n      imap: false,\n      error: `IMAP connection failed: ${error instanceof Error ? error.message : 'Unknown error'}`\n    };\n  }\n}"],"names":["ImapFlow","nodemailer","prisma","simpleParser","cacheEmailCredentials","getCachedEmailCredentials","cacheEmailList","cacheEmailContent","getCachedEmailContent","cacheImapSession","getCachedImapSession","invalidateFolderCache","invalidateEmailContentCache","ensureFreshToken","connectionPool","totalConnectionRequests","totalNewConnections","totalReuseConnections","totalConnectionErrors","lastMetricsReset","Date","now","CONNECTION_TIMEOUT","MAX_POOL_SIZE","CONNECTION_CHECK_INTERVAL","MIN_POOL_SIZE","setInterval","connectionKeys","Object","keys","console","log","toFixed","connectionsByUser","forEach","key","userId","split","push","entries","userConnections","sortedConnections","map","lastUsed","sort","a","b","connectionsToKeep","slice","keepKeys","Set","conn","has","isConnecting","Math","round","client","usable","logout","catch","err","error","length","activeCount","filter","connectingCount","getImapConnection","accountId","startTime","sessionData","defaultAccountId","accounts","mailCredentials","findMany","where","orderBy","createdAt","take","firstAccount","id","email","lastActive","Error","connectionKey","connection","connectionPromise","updateSessionData","warn","credentials","hasPassword","password","useOAuth","hasAccessToken","accessToken","hasRefreshToken","refreshToken","dbCredentials","findFirst","AND","fields","host","port","secure","smtp_host","undefined","smtp_port","smtp_secure","display_name","color","extendedCreds","cachedCreds","tokenExpiry","success","connectionAttempts","connectionTimeout","setTimeout","createImapConnection","then","clearTimeout","message","hasTokenExpiry","authParams","user","pass","auth","logger","emitLogs","tls","rejectUnauthorized","disableAutoIdle","connect","on","getUserEmailCredentials","saveUserEmailCredentials","oauthData","hasOAuth","upsert","select","result","update","create","fullCreds","getEmails","folder","page","perPage","checkOnly","resolvedAccountId","mailboxInfo","mailboxOpen","exists","mailboxes","getMailboxes","totalEmails","totalPages","ceil","emptyResult","emails","newestEmailId","lastMessageSequence","toString","messages","fetch","uid","start","max","end","envelope","flags","bodyStructure","from","addr","name","address","to","subject","date","seen","flagged","answered","draft","deleted","size","hasAttachments","childNodes","some","node","disposition","contentFetched","content","text","html","isHtml","direction","mapAddresses","addresses","Array","isArray","getEmailContent","emailId","test","numericId","parseInt","isNaN","folderAccountId","includes","effectiveAccountId","normalizedFolder","cachedEmail","mailbox","uidValidity","uidNext","searchResult","search","sequenceNumber","fetchOne","source","fetchError","parsedEmail","skipHtmlToText","keepCidLinks","parseError","flagsArray","rawHtml","messageId","cc","bcc","attachments","att","filename","contentType","details","stack","mailboxClose","markEmailReadStatus","isRead","messageFlagsAdd","messageFlagsRemove","toggleEmailFlag","sendEmail","emailData","smtpAuth","type","transporter","createTransport","info","sendMail","body","list","path","testEmailConnection","folders","smtpSuccess","verify","smtpError","imap","smtp"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./lib/services/email-service.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/services/microsoft-oauth.ts":
|
|
/*!*****************************************!*\
|
|
!*** ./lib/services/microsoft-oauth.ts ***!
|
|
\*****************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ createXOAuth2Token: () => (/* binding */ createXOAuth2Token),\n/* harmony export */ exchangeCodeForTokens: () => (/* binding */ exchangeCodeForTokens),\n/* harmony export */ getMicrosoftAuthUrl: () => (/* binding */ getMicrosoftAuthUrl),\n/* harmony export */ refreshAccessToken: () => (/* binding */ refreshAccessToken)\n/* harmony export */ });\n/* harmony import */ var axios__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! axios */ \"(rsc)/./node_modules/axios/lib/axios.js\");\n\n// Get tenant ID from env var or use a default\nconst tenantId = process.env.MICROSOFT_TENANT_ID || 'common'; // Use 'organizations' or actual tenant ID\n// Microsoft OAuth URLs with configurable tenant\nconst MICROSOFT_AUTHORIZE_URL = `https://login.microsoftonline.com/${tenantId}/oauth2/v2.0/authorize`;\nconst MICROSOFT_TOKEN_URL = `https://login.microsoftonline.com/${tenantId}/oauth2/v2.0/token`;\n// Client configuration from environment variables\nconst clientId = process.env.MICROSOFT_CLIENT_ID;\nconst clientSecret = process.env.MICROSOFT_CLIENT_SECRET;\nconst redirectUri = process.env.MICROSOFT_REDIRECT_URI;\n// Log configuration for debugging\nconsole.log('Microsoft OAuth Configuration:', {\n tenantId,\n authorizeUrl: MICROSOFT_AUTHORIZE_URL,\n tokenUrl: MICROSOFT_TOKEN_URL,\n clientIdFirstChars: clientId ? clientId.substring(0, 5) + '...' : 'undefined',\n redirectUri\n});\n// Required scopes for IMAP and SMTP access\nconst REQUIRED_SCOPES = [\n 'offline_access',\n 'https://outlook.office.com/IMAP.AccessAsUser.All',\n 'https://outlook.office.com/SMTP.Send'\n].join(' ');\n/**\n * Generates the authorization URL for Microsoft OAuth\n */ function getMicrosoftAuthUrl(state) {\n const params = new URLSearchParams({\n client_id: clientId,\n response_type: 'code',\n redirect_uri: redirectUri,\n scope: REQUIRED_SCOPES,\n state,\n response_mode: 'query'\n });\n return `${MICROSOFT_AUTHORIZE_URL}?${params.toString()}`;\n}\n/**\n * Exchange authorization code for tokens\n */ async function exchangeCodeForTokens(code) {\n const params = new URLSearchParams({\n client_id: clientId,\n client_secret: clientSecret,\n code,\n redirect_uri: redirectUri,\n grant_type: 'authorization_code'\n });\n try {\n console.log(`Exchanging code for tokens. URL: ${MICROSOFT_TOKEN_URL}`);\n const response = await axios__WEBPACK_IMPORTED_MODULE_0__[\"default\"].post(MICROSOFT_TOKEN_URL, params.toString(), {\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded'\n }\n });\n console.log('Token exchange successful!');\n return {\n access_token: response.data.access_token,\n refresh_token: response.data.refresh_token,\n expires_in: response.data.expires_in\n };\n } catch (error) {\n console.error('Error exchanging code for tokens:', error);\n // Enhanced error logging\n if (error.response) {\n console.error('Response data:', error.response.data);\n console.error('Response status:', error.response.status);\n console.error('Response headers:', error.response.headers);\n // Extract the error message from Microsoft's response format\n const errorData = error.response.data;\n if (errorData && errorData.error_description) {\n throw new Error(`Token exchange failed: ${errorData.error_description}`);\n }\n }\n throw new Error('Failed to exchange authorization code for tokens');\n }\n}\n/**\n * Refresh an access token using a refresh token\n */ async function refreshAccessToken(refreshToken) {\n const params = new URLSearchParams({\n client_id: clientId,\n client_secret: clientSecret,\n refresh_token: refreshToken,\n grant_type: 'refresh_token',\n scope: REQUIRED_SCOPES\n });\n try {\n console.log(`Refreshing access token. URL: ${MICROSOFT_TOKEN_URL}`);\n const response = await axios__WEBPACK_IMPORTED_MODULE_0__[\"default\"].post(MICROSOFT_TOKEN_URL, params.toString(), {\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded'\n }\n });\n console.log('Token refresh successful!');\n return {\n access_token: response.data.access_token,\n refresh_token: response.data.refresh_token,\n expires_in: response.data.expires_in\n };\n } catch (error) {\n console.error('Error refreshing token:', error);\n // Enhanced error logging\n if (error.response) {\n console.error('Response data:', error.response.data);\n console.error('Response status:', error.response.status);\n console.error('Response headers:', error.response.headers);\n // Extract the error message from Microsoft's response format\n const errorData = error.response.data;\n if (errorData && errorData.error_description) {\n throw new Error(`Token refresh failed: ${errorData.error_description}`);\n }\n }\n throw new Error('Failed to refresh access token');\n }\n}\n/**\n * Create special XOAUTH2 string for IMAP authentication\n */ function createXOAuth2Token(email, accessToken) {\n // This creates the XOAUTH2 token in the required format for ImapFlow\n // Format: user=<email>\\x01auth=Bearer <token>\\x01\\x01\n const auth = `user=${email}\\x01auth=Bearer ${accessToken}\\x01\\x01`;\n const base64Auth = Buffer.from(auth).toString('base64');\n console.log('Generated XOAUTH2 token (length):', base64Auth.length);\n return base64Auth;\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"(rsc)/./lib/services/microsoft-oauth.ts","mappings":"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","sources":["/home/alma/nextgen/Neah-mail/lib/services/microsoft-oauth.ts"],"sourcesContent":["import axios from 'axios';\n\n// Get tenant ID from env var or use a default\nconst tenantId = process.env.MICROSOFT_TENANT_ID || 'common'; // Use 'organizations' or actual tenant ID\n\n// Microsoft OAuth URLs with configurable tenant\nconst MICROSOFT_AUTHORIZE_URL = `https://login.microsoftonline.com/${tenantId}/oauth2/v2.0/authorize`;\nconst MICROSOFT_TOKEN_URL = `https://login.microsoftonline.com/${tenantId}/oauth2/v2.0/token`;\n\n// Client configuration from environment variables\nconst clientId = process.env.MICROSOFT_CLIENT_ID;\nconst clientSecret = process.env.MICROSOFT_CLIENT_SECRET;\nconst redirectUri = process.env.MICROSOFT_REDIRECT_URI;\n\n// Log configuration for debugging\nconsole.log('Microsoft OAuth Configuration:', {\n  tenantId,\n  authorizeUrl: MICROSOFT_AUTHORIZE_URL,\n  tokenUrl: MICROSOFT_TOKEN_URL,\n  clientIdFirstChars: clientId ? clientId.substring(0, 5) + '...' : 'undefined',\n  redirectUri\n});\n\n// Required scopes for IMAP and SMTP access\nconst REQUIRED_SCOPES = [\n  'offline_access',\n  'https://outlook.office.com/IMAP.AccessAsUser.All',\n  'https://outlook.office.com/SMTP.Send'\n].join(' ');\n\n/**\n * Generates the authorization URL for Microsoft OAuth\n */\nexport function getMicrosoftAuthUrl(state: string): string {\n  const params = new URLSearchParams({\n    client_id: clientId!,\n    response_type: 'code',\n    redirect_uri: redirectUri!,\n    scope: REQUIRED_SCOPES,\n    state,\n    response_mode: 'query'\n  });\n\n  return `${MICROSOFT_AUTHORIZE_URL}?${params.toString()}`;\n}\n\n/**\n * Exchange authorization code for tokens\n */\nexport async function exchangeCodeForTokens(code: string): Promise<{\n  access_token: string;\n  refresh_token: string;\n  expires_in: number;\n}> {\n  const params = new URLSearchParams({\n    client_id: clientId!,\n    client_secret: clientSecret!,\n    code,\n    redirect_uri: redirectUri!,\n    grant_type: 'authorization_code'\n  });\n\n  try {\n    console.log(`Exchanging code for tokens. URL: ${MICROSOFT_TOKEN_URL}`);\n    \n    const response = await axios.post(MICROSOFT_TOKEN_URL, params.toString(), {\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded'\n      }\n    });\n\n    console.log('Token exchange successful!');\n    return {\n      access_token: response.data.access_token,\n      refresh_token: response.data.refresh_token,\n      expires_in: response.data.expires_in\n    };\n  } catch (error: any) {\n    console.error('Error exchanging code for tokens:', error);\n    \n    // Enhanced error logging\n    if (error.response) {\n      console.error('Response data:', error.response.data);\n      console.error('Response status:', error.response.status);\n      console.error('Response headers:', error.response.headers);\n      \n      // Extract the error message from Microsoft's response format\n      const errorData = error.response.data;\n      if (errorData && errorData.error_description) {\n        throw new Error(`Token exchange failed: ${errorData.error_description}`);\n      }\n    }\n    \n    throw new Error('Failed to exchange authorization code for tokens');\n  }\n}\n\n/**\n * Refresh an access token using a refresh token\n */\nexport async function refreshAccessToken(refreshToken: string): Promise<{\n  access_token: string;\n  refresh_token?: string;\n  expires_in: number;\n}> {\n  const params = new URLSearchParams({\n    client_id: clientId!,\n    client_secret: clientSecret!,\n    refresh_token: refreshToken,\n    grant_type: 'refresh_token',\n    scope: REQUIRED_SCOPES\n  });\n\n  try {\n    console.log(`Refreshing access token. URL: ${MICROSOFT_TOKEN_URL}`);\n    \n    const response = await axios.post(MICROSOFT_TOKEN_URL, params.toString(), {\n      headers: {\n        'Content-Type': 'application/x-www-form-urlencoded'\n      }\n    });\n\n    console.log('Token refresh successful!');\n    return {\n      access_token: response.data.access_token,\n      refresh_token: response.data.refresh_token,\n      expires_in: response.data.expires_in\n    };\n  } catch (error: any) {\n    console.error('Error refreshing token:', error);\n    \n    // Enhanced error logging\n    if (error.response) {\n      console.error('Response data:', error.response.data);\n      console.error('Response status:', error.response.status);\n      console.error('Response headers:', error.response.headers);\n      \n      // Extract the error message from Microsoft's response format\n      const errorData = error.response.data;\n      if (errorData && errorData.error_description) {\n        throw new Error(`Token refresh failed: ${errorData.error_description}`);\n      }\n    }\n    \n    throw new Error('Failed to refresh access token');\n  }\n}\n\n/**\n * Create special XOAUTH2 string for IMAP authentication\n */\nexport function createXOAuth2Token(email: string, accessToken: string): string {\n  // This creates the XOAUTH2 token in the required format for ImapFlow\n  // Format: user=<email>\\x01auth=Bearer <token>\\x01\\x01\n  const auth = `user=${email}\\x01auth=Bearer ${accessToken}\\x01\\x01`;\n  const base64Auth = Buffer.from(auth).toString('base64');\n  \n  console.log('Generated XOAUTH2 token (length):', base64Auth.length);\n  return base64Auth;\n} "],"names":["axios","tenantId","process","env","MICROSOFT_TENANT_ID","MICROSOFT_AUTHORIZE_URL","MICROSOFT_TOKEN_URL","clientId","MICROSOFT_CLIENT_ID","clientSecret","MICROSOFT_CLIENT_SECRET","redirectUri","MICROSOFT_REDIRECT_URI","console","log","authorizeUrl","tokenUrl","clientIdFirstChars","substring","REQUIRED_SCOPES","join","getMicrosoftAuthUrl","state","params","URLSearchParams","client_id","response_type","redirect_uri","scope","response_mode","toString","exchangeCodeForTokens","code","client_secret","grant_type","response","post","headers","access_token","data","refresh_token","expires_in","error","status","errorData","error_description","Error","refreshAccessToken","refreshToken","createXOAuth2Token","email","accessToken","auth","base64Auth","Buffer","from","length"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./lib/services/microsoft-oauth.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./lib/services/token-refresh.ts":
|
|
/*!***************************************!*\
|
|
!*** ./lib/services/token-refresh.ts ***!
|
|
\***************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ ensureFreshToken: () => (/* binding */ ensureFreshToken),\n/* harmony export */ isTokenExpired: () => (/* binding */ isTokenExpired)\n/* harmony export */ });\n/* harmony import */ var _microsoft_oauth__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./microsoft-oauth */ \"(rsc)/./lib/services/microsoft-oauth.ts\");\n/* harmony import */ var _lib_redis__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @/lib/redis */ \"(rsc)/./lib/redis.ts\");\n\n\n/**\n * Check if a token is expired or about to expire (within 5 minutes)\n */ function isTokenExpired(expiryTimestamp) {\n const fiveMinutesInMs = 5 * 60 * 1000;\n return Date.now() + fiveMinutesInMs >= expiryTimestamp;\n}\n/**\n * Refresh an access token if it's expired or about to expire\n */ async function ensureFreshToken(userId, email) {\n try {\n // Use Redis to get the tokens (no database lookup needed)\n console.log(`Checking if token refresh is needed for ${email}`);\n const redis = (0,_lib_redis__WEBPACK_IMPORTED_MODULE_1__.getRedisClient)();\n const key = _lib_redis__WEBPACK_IMPORTED_MODULE_1__.KEYS.CREDENTIALS(userId, email);\n const credStr = await redis.get(key);\n if (!credStr) {\n console.log(`No credentials found in Redis for ${email}`);\n return {\n accessToken: '',\n success: false\n };\n }\n const creds = JSON.parse(credStr);\n // If not OAuth or missing refresh token, return failure\n if (!creds.useOAuth || !creds.refreshToken) {\n console.log(`Account ${email} is not using OAuth or missing refresh token`);\n return {\n accessToken: '',\n success: false\n };\n }\n // If token is still valid, return current token\n if (creds.tokenExpiry && creds.accessToken && creds.tokenExpiry > Date.now() + 5 * 60 * 1000) {\n console.log(`Token for ${email} is still valid, no refresh needed`);\n return {\n accessToken: creds.accessToken,\n success: true\n };\n }\n // Token is expired or about to expire, refresh it\n console.log(`Refreshing token for ${email}`);\n const tokens = await (0,_microsoft_oauth__WEBPACK_IMPORTED_MODULE_0__.refreshAccessToken)(creds.refreshToken);\n // Update Redis cache with new tokens\n creds.accessToken = tokens.access_token;\n if (tokens.refresh_token) {\n creds.refreshToken = tokens.refresh_token;\n }\n creds.tokenExpiry = Date.now() + tokens.expires_in * 1000;\n await redis.set(key, JSON.stringify(creds), 'EX', 86400); // 24 hours\n console.log(`Token for ${email} refreshed and cached in Redis`);\n return {\n accessToken: tokens.access_token,\n success: true\n };\n } catch (error) {\n console.error(`Error refreshing token for ${email}:`, error);\n return {\n accessToken: '',\n success: false\n };\n }\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9saWIvc2VydmljZXMvdG9rZW4tcmVmcmVzaC50cyIsIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQXVEO0FBQ0o7QUFFbkQ7O0NBRUMsR0FDTSxTQUFTRyxlQUFlQyxlQUF1QjtJQUNwRCxNQUFNQyxrQkFBa0IsSUFBSSxLQUFLO0lBQ2pDLE9BQU9DLEtBQUtDLEdBQUcsS0FBS0YsbUJBQW1CRDtBQUN6QztBQUVBOztDQUVDLEdBQ00sZUFBZUksaUJBQ3BCQyxNQUFjLEVBQ2RDLEtBQWE7SUFFYixJQUFJO1FBQ0YsMERBQTBEO1FBQzFEQyxRQUFRQyxHQUFHLENBQUMsQ0FBQyx3Q0FBd0MsRUFBRUYsT0FBTztRQUM5RCxNQUFNRyxRQUFRWiwwREFBY0E7UUFDNUIsTUFBTWEsTUFBTVosNENBQUlBLENBQUNhLFdBQVcsQ0FBQ04sUUFBUUM7UUFDckMsTUFBTU0sVUFBVSxNQUFNSCxNQUFNSSxHQUFHLENBQUNIO1FBRWhDLElBQUksQ0FBQ0UsU0FBUztZQUNaTCxRQUFRQyxHQUFHLENBQUMsQ0FBQyxrQ0FBa0MsRUFBRUYsT0FBTztZQUN4RCxPQUFPO2dCQUFFUSxhQUFhO2dCQUFJQyxTQUFTO1lBQU07UUFDM0M7UUFFQSxNQUFNQyxRQUFRQyxLQUFLQyxLQUFLLENBQUNOO1FBRXpCLHdEQUF3RDtRQUN4RCxJQUFJLENBQUNJLE1BQU1HLFFBQVEsSUFBSSxDQUFDSCxNQUFNSSxZQUFZLEVBQUU7WUFDMUNiLFFBQVFDLEdBQUcsQ0FBQyxDQUFDLFFBQVEsRUFBRUYsTUFBTSw0Q0FBNEMsQ0FBQztZQUMxRSxPQUFPO2dCQUFFUSxhQUFhO2dCQUFJQyxTQUFTO1lBQU07UUFDM0M7UUFFQSxnREFBZ0Q7UUFDaEQsSUFBSUMsTUFBTUssV0FBVyxJQUFJTCxNQUFNRixXQUFXLElBQ3RDRSxNQUFNSyxXQUFXLEdBQUduQixLQUFLQyxHQUFHLEtBQUssSUFBSSxLQUFLLE1BQU07WUFDbERJLFFBQVFDLEdBQUcsQ0FBQyxDQUFDLFVBQVUsRUFBRUYsTUFBTSxrQ0FBa0MsQ0FBQztZQUNsRSxPQUFPO2dCQUFFUSxhQUFhRSxNQUFNRixXQUFXO2dCQUFFQyxTQUFTO1lBQUs7UUFDekQ7UUFFQSxrREFBa0Q7UUFDbERSLFFBQVFDLEdBQUcsQ0FBQyxDQUFDLHFCQUFxQixFQUFFRixPQUFPO1FBQzNDLE1BQU1nQixTQUFTLE1BQU0xQixvRUFBa0JBLENBQUNvQixNQUFNSSxZQUFZO1FBRTFELHFDQUFxQztRQUNyQ0osTUFBTUYsV0FBVyxHQUFHUSxPQUFPQyxZQUFZO1FBQ3ZDLElBQUlELE9BQU9FLGFBQWEsRUFBRTtZQUN4QlIsTUFBTUksWUFBWSxHQUFHRSxPQUFPRSxhQUFhO1FBQzNDO1FBQ0FSLE1BQU1LLFdBQVcsR0FBR25CLEtBQUtDLEdBQUcsS0FBTW1CLE9BQU9HLFVBQVUsR0FBRztRQUV0RCxNQUFNaEIsTUFBTWlCLEdBQUcsQ0FBQ2hCLEtBQUtPLEtBQUtVLFNBQVMsQ0FBQ1gsUUFBUSxNQUFNLFFBQVEsV0FBVztRQUNyRVQsUUFBUUMsR0FBRyxDQUFDLENBQUMsVUFBVSxFQUFFRixNQUFNLDhCQUE4QixDQUFDO1FBRTlELE9BQU87WUFBRVEsYUFBYVEsT0FBT0MsWUFBWTtZQUFFUixTQUFTO1FBQUs7SUFDM0QsRUFBRSxPQUFPYSxPQUFPO1FBQ2RyQixRQUFRcUIsS0FBSyxDQUFDLENBQUMsMkJBQTJCLEVBQUV0QixNQUFNLENBQUMsQ0FBQyxFQUFFc0I7UUFDdEQsT0FBTztZQUFFZCxhQUFhO1lBQUlDLFNBQVM7UUFBTTtJQUMzQztBQUNGIiwic291cmNlcyI6WyIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2xpYi9zZXJ2aWNlcy90b2tlbi1yZWZyZXNoLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IHJlZnJlc2hBY2Nlc3NUb2tlbiB9IGZyb20gJy4vbWljcm9zb2Z0LW9hdXRoJztcbmltcG9ydCB7IGdldFJlZGlzQ2xpZW50LCBLRVlTIH0gZnJvbSAnQC9saWIvcmVkaXMnO1xuXG4vKipcbiAqIENoZWNrIGlmIGEgdG9rZW4gaXMgZXhwaXJlZCBvciBhYm91dCB0byBleHBpcmUgKHdpdGhpbiA1IG1pbnV0ZXMpXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBpc1Rva2VuRXhwaXJlZChleHBpcnlUaW1lc3RhbXA6IG51bWJlcik6IGJvb2xlYW4ge1xuICBjb25zdCBmaXZlTWludXRlc0luTXMgPSA1ICogNjAgKiAxMDAwO1xuICByZXR1cm4gRGF0ZS5ub3coKSArIGZpdmVNaW51dGVzSW5NcyA+PSBleHBpcnlUaW1lc3RhbXA7XG59XG5cbi8qKlxuICogUmVmcmVzaCBhbiBhY2Nlc3MgdG9rZW4gaWYgaXQncyBleHBpcmVkIG9yIGFib3V0IHRvIGV4cGlyZVxuICovXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gZW5zdXJlRnJlc2hUb2tlbihcbiAgdXNlcklkOiBzdHJpbmcsXG4gIGVtYWlsOiBzdHJpbmdcbik6IFByb21pc2U8eyBhY2Nlc3NUb2tlbjogc3RyaW5nOyBzdWNjZXNzOiBib29sZWFuIH0+IHtcbiAgdHJ5IHtcbiAgICAvLyBVc2UgUmVkaXMgdG8gZ2V0IHRoZSB0b2tlbnMgKG5vIGRhdGFiYXNlIGxvb2t1cCBuZWVkZWQpXG4gICAgY29uc29sZS5sb2coYENoZWNraW5nIGlmIHRva2VuIHJlZnJlc2ggaXMgbmVlZGVkIGZvciAke2VtYWlsfWApO1xuICAgIGNvbnN0IHJlZGlzID0gZ2V0UmVkaXNDbGllbnQoKTtcbiAgICBjb25zdCBrZXkgPSBLRVlTLkNSRURFTlRJQUxTKHVzZXJJZCwgZW1haWwpO1xuICAgIGNvbnN0IGNyZWRTdHIgPSBhd2FpdCByZWRpcy5nZXQoa2V5KTtcbiAgICBcbiAgICBpZiAoIWNyZWRTdHIpIHtcbiAgICAgIGNvbnNvbGUubG9nKGBObyBjcmVkZW50aWFscyBmb3VuZCBpbiBSZWRpcyBmb3IgJHtlbWFpbH1gKTtcbiAgICAgIHJldHVybiB7IGFjY2Vzc1Rva2VuOiAnJywgc3VjY2VzczogZmFsc2UgfTtcbiAgICB9XG4gICAgXG4gICAgY29uc3QgY3JlZHMgPSBKU09OLnBhcnNlKGNyZWRTdHIpO1xuICAgIFxuICAgIC8vIElmIG5vdCBPQXV0aCBvciBtaXNzaW5nIHJlZnJlc2ggdG9rZW4sIHJldHVybiBmYWlsdXJlXG4gICAgaWYgKCFjcmVkcy51c2VPQXV0aCB8fCAhY3JlZHMucmVmcmVzaFRva2VuKSB7XG4gICAgICBjb25zb2xlLmxvZyhgQWNjb3VudCAke2VtYWlsfSBpcyBub3QgdXNpbmcgT0F1dGggb3IgbWlzc2luZyByZWZyZXNoIHRva2VuYCk7XG4gICAgICByZXR1cm4geyBhY2Nlc3NUb2tlbjogJycsIHN1Y2Nlc3M6IGZhbHNlIH07XG4gICAgfVxuXG4gICAgLy8gSWYgdG9rZW4gaXMgc3RpbGwgdmFsaWQsIHJldHVybiBjdXJyZW50IHRva2VuXG4gICAgaWYgKGNyZWRzLnRva2VuRXhwaXJ5ICYmIGNyZWRzLmFjY2Vzc1Rva2VuICYmIFxuICAgICAgICBjcmVkcy50b2tlbkV4cGlyeSA+IERhdGUubm93KCkgKyA1ICogNjAgKiAxMDAwKSB7XG4gICAgICBjb25zb2xlLmxvZyhgVG9rZW4gZm9yICR7ZW1haWx9IGlzIHN0aWxsIHZhbGlkLCBubyByZWZyZXNoIG5lZWRlZGApO1xuICAgICAgcmV0dXJuIHsgYWNjZXNzVG9rZW46IGNyZWRzLmFjY2Vzc1Rva2VuLCBzdWNjZXNzOiB0cnVlIH07XG4gICAgfVxuXG4gICAgLy8gVG9rZW4gaXMgZXhwaXJlZCBvciBhYm91dCB0byBleHBpcmUsIHJlZnJlc2ggaXRcbiAgICBjb25zb2xlLmxvZyhgUmVmcmVzaGluZyB0b2tlbiBmb3IgJHtlbWFpbH1gKTtcbiAgICBjb25zdCB0b2tlbnMgPSBhd2FpdCByZWZyZXNoQWNjZXNzVG9rZW4oY3JlZHMucmVmcmVzaFRva2VuKTtcbiAgICBcbiAgICAvLyBVcGRhdGUgUmVkaXMgY2FjaGUgd2l0aCBuZXcgdG9rZW5zXG4gICAgY3JlZHMuYWNjZXNzVG9rZW4gPSB0b2tlbnMuYWNjZXNzX3Rva2VuO1xuICAgIGlmICh0b2tlbnMucmVmcmVzaF90b2tlbikge1xuICAgICAgY3JlZHMucmVmcmVzaFRva2VuID0gdG9rZW5zLnJlZnJlc2hfdG9rZW47XG4gICAgfVxuICAgIGNyZWRzLnRva2VuRXhwaXJ5ID0gRGF0ZS5ub3coKSArICh0b2tlbnMuZXhwaXJlc19pbiAqIDEwMDApO1xuICAgIFxuICAgIGF3YWl0IHJlZGlzLnNldChrZXksIEpTT04uc3RyaW5naWZ5KGNyZWRzKSwgJ0VYJywgODY0MDApOyAvLyAyNCBob3Vyc1xuICAgIGNvbnNvbGUubG9nKGBUb2tlbiBmb3IgJHtlbWFpbH0gcmVmcmVzaGVkIGFuZCBjYWNoZWQgaW4gUmVkaXNgKTtcblxuICAgIHJldHVybiB7IGFjY2Vzc1Rva2VuOiB0b2tlbnMuYWNjZXNzX3Rva2VuLCBzdWNjZXNzOiB0cnVlIH07XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgY29uc29sZS5lcnJvcihgRXJyb3IgcmVmcmVzaGluZyB0b2tlbiBmb3IgJHtlbWFpbH06YCwgZXJyb3IpO1xuICAgIHJldHVybiB7IGFjY2Vzc1Rva2VuOiAnJywgc3VjY2VzczogZmFsc2UgfTtcbiAgfVxufSAiXSwibmFtZXMiOlsicmVmcmVzaEFjY2Vzc1Rva2VuIiwiZ2V0UmVkaXNDbGllbnQiLCJLRVlTIiwiaXNUb2tlbkV4cGlyZWQiLCJleHBpcnlUaW1lc3RhbXAiLCJmaXZlTWludXRlc0luTXMiLCJEYXRlIiwibm93IiwiZW5zdXJlRnJlc2hUb2tlbiIsInVzZXJJZCIsImVtYWlsIiwiY29uc29sZSIsImxvZyIsInJlZGlzIiwia2V5IiwiQ1JFREVOVElBTFMiLCJjcmVkU3RyIiwiZ2V0IiwiYWNjZXNzVG9rZW4iLCJzdWNjZXNzIiwiY3JlZHMiLCJKU09OIiwicGFyc2UiLCJ1c2VPQXV0aCIsInJlZnJlc2hUb2tlbiIsInRva2VuRXhwaXJ5IiwidG9rZW5zIiwiYWNjZXNzX3Rva2VuIiwicmVmcmVzaF90b2tlbiIsImV4cGlyZXNfaW4iLCJzZXQiLCJzdHJpbmdpZnkiLCJlcnJvciJdLCJpZ25vcmVMaXN0IjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./lib/services/token-refresh.ts\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fcourrier%2Femails%2Froute&page=%2Fapi%2Fcourrier%2Femails%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fcourrier%2Femails%2Froute.ts&appDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Fapp&pageExtensions=tsx&pageExtensions=ts&pageExtensions=jsx&pageExtensions=js&rootDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail&isDev=true&tsconfigPath=tsconfig.json&basePath=&assetPrefix=&nextConfigOutput=&preferredRegion=&middlewareConfig=e30%3D!":
|
|
/*!*************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************!*\
|
|
!*** ./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fcourrier%2Femails%2Froute&page=%2Fapi%2Fcourrier%2Femails%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fcourrier%2Femails%2Froute.ts&appDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Fapp&pageExtensions=tsx&pageExtensions=ts&pageExtensions=jsx&pageExtensions=js&rootDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail&isDev=true&tsconfigPath=tsconfig.json&basePath=&assetPrefix=&nextConfigOutput=&preferredRegion=&middlewareConfig=e30%3D! ***!
|
|
\*************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ patchFetch: () => (/* binding */ patchFetch),\n/* harmony export */ routeModule: () => (/* binding */ routeModule),\n/* harmony export */ serverHooks: () => (/* binding */ serverHooks),\n/* harmony export */ workAsyncStorage: () => (/* binding */ workAsyncStorage),\n/* harmony export */ workUnitAsyncStorage: () => (/* binding */ workUnitAsyncStorage)\n/* harmony export */ });\n/* harmony import */ var next_dist_server_route_modules_app_route_module_compiled__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! next/dist/server/route-modules/app-route/module.compiled */ \"(rsc)/./node_modules/next/dist/server/route-modules/app-route/module.compiled.js\");\n/* harmony import */ var next_dist_server_route_modules_app_route_module_compiled__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(next_dist_server_route_modules_app_route_module_compiled__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var next_dist_server_route_kind__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! next/dist/server/route-kind */ \"(rsc)/./node_modules/next/dist/server/route-kind.js\");\n/* harmony import */ var next_dist_server_lib_patch_fetch__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! next/dist/server/lib/patch-fetch */ \"(rsc)/./node_modules/next/dist/server/lib/patch-fetch.js\");\n/* harmony import */ var next_dist_server_lib_patch_fetch__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(next_dist_server_lib_patch_fetch__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var _home_alma_nextgen_Neah_mail_app_api_courrier_emails_route_ts__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./app/api/courrier/emails/route.ts */ \"(rsc)/./app/api/courrier/emails/route.ts\");\n\n\n\n\n// We inject the nextConfigOutput here so that we can use them in the route\n// module.\nconst nextConfigOutput = \"\"\nconst routeModule = new next_dist_server_route_modules_app_route_module_compiled__WEBPACK_IMPORTED_MODULE_0__.AppRouteRouteModule({\n definition: {\n kind: next_dist_server_route_kind__WEBPACK_IMPORTED_MODULE_1__.RouteKind.APP_ROUTE,\n page: \"/api/courrier/emails/route\",\n pathname: \"/api/courrier/emails\",\n filename: \"route\",\n bundlePath: \"app/api/courrier/emails/route\"\n },\n resolvedPagePath: \"/home/alma/nextgen/Neah-mail/app/api/courrier/emails/route.ts\",\n nextConfigOutput,\n userland: _home_alma_nextgen_Neah_mail_app_api_courrier_emails_route_ts__WEBPACK_IMPORTED_MODULE_3__\n});\n// Pull out the exports that we need to expose from the module. This should\n// be eliminated when we've moved the other routes to the new format. These\n// are used to hook into the route.\nconst { workAsyncStorage, workUnitAsyncStorage, serverHooks } = routeModule;\nfunction patchFetch() {\n return (0,next_dist_server_lib_patch_fetch__WEBPACK_IMPORTED_MODULE_2__.patchFetch)({\n workAsyncStorage,\n workUnitAsyncStorage\n });\n}\n\n\n//# sourceMappingURL=app-route.js.map//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9ub2RlX21vZHVsZXMvbmV4dC9kaXN0L2J1aWxkL3dlYnBhY2svbG9hZGVycy9uZXh0LWFwcC1sb2FkZXIvaW5kZXguanM/bmFtZT1hcHAlMkZhcGklMkZjb3VycmllciUyRmVtYWlscyUyRnJvdXRlJnBhZ2U9JTJGYXBpJTJGY291cnJpZXIlMkZlbWFpbHMlMkZyb3V0ZSZhcHBQYXRocz0mcGFnZVBhdGg9cHJpdmF0ZS1uZXh0LWFwcC1kaXIlMkZhcGklMkZjb3VycmllciUyRmVtYWlscyUyRnJvdXRlLnRzJmFwcERpcj0lMkZob21lJTJGYWxtYSUyRm5leHRnZW4lMkZOZWFoLW1haWwlMkZhcHAmcGFnZUV4dGVuc2lvbnM9dHN4JnBhZ2VFeHRlbnNpb25zPXRzJnBhZ2VFeHRlbnNpb25zPWpzeCZwYWdlRXh0ZW5zaW9ucz1qcyZyb290RGlyPSUyRmhvbWUlMkZhbG1hJTJGbmV4dGdlbiUyRk5lYWgtbWFpbCZpc0Rldj10cnVlJnRzY29uZmlnUGF0aD10c2NvbmZpZy5qc29uJmJhc2VQYXRoPSZhc3NldFByZWZpeD0mbmV4dENvbmZpZ091dHB1dD0mcHJlZmVycmVkUmVnaW9uPSZtaWRkbGV3YXJlQ29uZmlnPWUzMCUzRCEiLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7QUFBK0Y7QUFDdkM7QUFDcUI7QUFDYTtBQUMxRjtBQUNBO0FBQ0E7QUFDQSx3QkFBd0IseUdBQW1CO0FBQzNDO0FBQ0EsY0FBYyxrRUFBUztBQUN2QjtBQUNBO0FBQ0E7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUNBO0FBQ0EsWUFBWTtBQUNaLENBQUM7QUFDRDtBQUNBO0FBQ0E7QUFDQSxRQUFRLHNEQUFzRDtBQUM5RDtBQUNBLFdBQVcsNEVBQVc7QUFDdEI7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUMwRjs7QUFFMUYiLCJzb3VyY2VzIjpbIiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcHBSb3V0ZVJvdXRlTW9kdWxlIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvcm91dGUtbW9kdWxlcy9hcHAtcm91dGUvbW9kdWxlLmNvbXBpbGVkXCI7XG5pbXBvcnQgeyBSb3V0ZUtpbmQgfSBmcm9tIFwibmV4dC9kaXN0L3NlcnZlci9yb3V0ZS1raW5kXCI7XG5pbXBvcnQgeyBwYXRjaEZldGNoIGFzIF9wYXRjaEZldGNoIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvbGliL3BhdGNoLWZldGNoXCI7XG5pbXBvcnQgKiBhcyB1c2VybGFuZCBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9hcHAvYXBpL2NvdXJyaWVyL2VtYWlscy9yb3V0ZS50c1wiO1xuLy8gV2UgaW5qZWN0IHRoZSBuZXh0Q29uZmlnT3V0cHV0IGhlcmUgc28gdGhhdCB3ZSBjYW4gdXNlIHRoZW0gaW4gdGhlIHJvdXRlXG4vLyBtb2R1bGUuXG5jb25zdCBuZXh0Q29uZmlnT3V0cHV0ID0gXCJcIlxuY29uc3Qgcm91dGVNb2R1bGUgPSBuZXcgQXBwUm91dGVSb3V0ZU1vZHVsZSh7XG4gICAgZGVmaW5pdGlvbjoge1xuICAgICAgICBraW5kOiBSb3V0ZUtpbmQuQVBQX1JPVVRFLFxuICAgICAgICBwYWdlOiBcIi9hcGkvY291cnJpZXIvZW1haWxzL3JvdXRlXCIsXG4gICAgICAgIHBhdGhuYW1lOiBcIi9hcGkvY291cnJpZXIvZW1haWxzXCIsXG4gICAgICAgIGZpbGVuYW1lOiBcInJvdXRlXCIsXG4gICAgICAgIGJ1bmRsZVBhdGg6IFwiYXBwL2FwaS9jb3Vycmllci9lbWFpbHMvcm91dGVcIlxuICAgIH0sXG4gICAgcmVzb2x2ZWRQYWdlUGF0aDogXCIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2FwcC9hcGkvY291cnJpZXIvZW1haWxzL3JvdXRlLnRzXCIsXG4gICAgbmV4dENvbmZpZ091dHB1dCxcbiAgICB1c2VybGFuZFxufSk7XG4vLyBQdWxsIG91dCB0aGUgZXhwb3J0cyB0aGF0IHdlIG5lZWQgdG8gZXhwb3NlIGZyb20gdGhlIG1vZHVsZS4gVGhpcyBzaG91bGRcbi8vIGJlIGVsaW1pbmF0ZWQgd2hlbiB3ZSd2ZSBtb3ZlZCB0aGUgb3RoZXIgcm91dGVzIHRvIHRoZSBuZXcgZm9ybWF0LiBUaGVzZVxuLy8gYXJlIHVzZWQgdG8gaG9vayBpbnRvIHRoZSByb3V0ZS5cbmNvbnN0IHsgd29ya0FzeW5jU3RvcmFnZSwgd29ya1VuaXRBc3luY1N0b3JhZ2UsIHNlcnZlckhvb2tzIH0gPSByb3V0ZU1vZHVsZTtcbmZ1bmN0aW9uIHBhdGNoRmV0Y2goKSB7XG4gICAgcmV0dXJuIF9wYXRjaEZldGNoKHtcbiAgICAgICAgd29ya0FzeW5jU3RvcmFnZSxcbiAgICAgICAgd29ya1VuaXRBc3luY1N0b3JhZ2VcbiAgICB9KTtcbn1cbmV4cG9ydCB7IHJvdXRlTW9kdWxlLCB3b3JrQXN5bmNTdG9yYWdlLCB3b3JrVW5pdEFzeW5jU3RvcmFnZSwgc2VydmVySG9va3MsIHBhdGNoRmV0Y2gsICB9O1xuXG4vLyMgc291cmNlTWFwcGluZ1VSTD1hcHAtcm91dGUuanMubWFwIl0sIm5hbWVzIjpbXSwiaWdub3JlTGlzdCI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fcourrier%2Femails%2Froute&page=%2Fapi%2Fcourrier%2Femails%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fcourrier%2Femails%2Froute.ts&appDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Fapp&pageExtensions=tsx&pageExtensions=ts&pageExtensions=jsx&pageExtensions=js&rootDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail&isDev=true&tsconfigPath=tsconfig.json&basePath=&assetPrefix=&nextConfigOutput=&preferredRegion=&middlewareConfig=e30%3D!\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/next/dist/build/webpack/loaders/next-flight-action-entry-loader.js?actions=%5B%5B%22%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Flib%2Fservices%2Femail-service.ts%22%2C%5B%7B%22id%22%3A%224010844c55b83dfb1f105bebefe0eefe1e6b431900%22%2C%22exportedName%22%3A%22testEmailConnection%22%7D%2C%7B%22id%22%3A%2260249f33dc41bab8693201a3f19f5e5fb46e641c28%22%2C%22exportedName%22%3A%22getImapConnection%22%7D%2C%7B%22id%22%3A%2260852a4e84650a79792bf7dba8eab6e2994fad2674%22%2C%22exportedName%22%3A%22sendEmail%22%7D%2C%7B%22id%22%3A%22608647185521cdbec25a6e83fd03959d7becd6a6cd%22%2C%22exportedName%22%3A%22getMailboxes%22%7D%2C%7B%22id%22%3A%22609e97c061f87c9d92e6b4c180de319248e8263787%22%2C%22exportedName%22%3A%22getUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2270576421f3f1a8e0b47693f06bf0b900d321800592%22%2C%22exportedName%22%3A%22saveUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2278725b1539278f6847adbb3f678020efc4d204d6de%22%2C%22exportedName%22%3A%22getEmailContent%22%7D%2C%7B%22id%22%3A%227c9e86d7555bbe449b808bd2ae7479f82af99409fe%22%2C%22exportedName%22%3A%22markEmailReadStatus%22%7D%2C%7B%22id%22%3A%227ccc49d190a944aede514b28ad88d4a52971336211%22%2C%22exportedName%22%3A%22toggleEmailFlag%22%7D%2C%7B%22id%22%3A%227e6fee2e8d6b5661c87219a81dea11090773f206cd%22%2C%22exportedName%22%3A%22getEmails%22%7D%5D%5D%5D&__client_imported__=!":
|
|
/*!**************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************!*\
|
|
!*** ./node_modules/next/dist/build/webpack/loaders/next-flight-action-entry-loader.js?actions=%5B%5B%22%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Flib%2Fservices%2Femail-service.ts%22%2C%5B%7B%22id%22%3A%224010844c55b83dfb1f105bebefe0eefe1e6b431900%22%2C%22exportedName%22%3A%22testEmailConnection%22%7D%2C%7B%22id%22%3A%2260249f33dc41bab8693201a3f19f5e5fb46e641c28%22%2C%22exportedName%22%3A%22getImapConnection%22%7D%2C%7B%22id%22%3A%2260852a4e84650a79792bf7dba8eab6e2994fad2674%22%2C%22exportedName%22%3A%22sendEmail%22%7D%2C%7B%22id%22%3A%22608647185521cdbec25a6e83fd03959d7becd6a6cd%22%2C%22exportedName%22%3A%22getMailboxes%22%7D%2C%7B%22id%22%3A%22609e97c061f87c9d92e6b4c180de319248e8263787%22%2C%22exportedName%22%3A%22getUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2270576421f3f1a8e0b47693f06bf0b900d321800592%22%2C%22exportedName%22%3A%22saveUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2278725b1539278f6847adbb3f678020efc4d204d6de%22%2C%22exportedName%22%3A%22getEmailContent%22%7D%2C%7B%22id%22%3A%227c9e86d7555bbe449b808bd2ae7479f82af99409fe%22%2C%22exportedName%22%3A%22markEmailReadStatus%22%7D%2C%7B%22id%22%3A%227ccc49d190a944aede514b28ad88d4a52971336211%22%2C%22exportedName%22%3A%22toggleEmailFlag%22%7D%2C%7B%22id%22%3A%227e6fee2e8d6b5661c87219a81dea11090773f206cd%22%2C%22exportedName%22%3A%22getEmails%22%7D%5D%5D%5D&__client_imported__=! ***!
|
|
\**************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************************/
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
"use strict";
|
|
eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"4010844c55b83dfb1f105bebefe0eefe1e6b431900\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.testEmailConnection),\n/* harmony export */ \"60249f33dc41bab8693201a3f19f5e5fb46e641c28\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.getImapConnection),\n/* harmony export */ \"60852a4e84650a79792bf7dba8eab6e2994fad2674\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.sendEmail),\n/* harmony export */ \"608647185521cdbec25a6e83fd03959d7becd6a6cd\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.getMailboxes),\n/* harmony export */ \"609e97c061f87c9d92e6b4c180de319248e8263787\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.getUserEmailCredentials),\n/* harmony export */ \"70576421f3f1a8e0b47693f06bf0b900d321800592\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.saveUserEmailCredentials),\n/* harmony export */ \"78725b1539278f6847adbb3f678020efc4d204d6de\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.getEmailContent),\n/* harmony export */ \"7c9e86d7555bbe449b808bd2ae7479f82af99409fe\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.markEmailReadStatus),\n/* harmony export */ \"7ccc49d190a944aede514b28ad88d4a52971336211\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.toggleEmailFlag),\n/* harmony export */ \"7e6fee2e8d6b5661c87219a81dea11090773f206cd\": () => (/* reexport safe */ _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__.getEmails)\n/* harmony export */ });\n/* harmony import */ var _home_alma_nextgen_Neah_mail_lib_services_email_service_ts__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./lib/services/email-service.ts */ \"(rsc)/./lib/services/email-service.ts\");\n\n\n\n\n\n\n\n\n\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9ub2RlX21vZHVsZXMvbmV4dC9kaXN0L2J1aWxkL3dlYnBhY2svbG9hZGVycy9uZXh0LWZsaWdodC1hY3Rpb24tZW50cnktbG9hZGVyLmpzP2FjdGlvbnM9JTVCJTVCJTIyJTJGaG9tZSUyRmFsbWElMkZuZXh0Z2VuJTJGTmVhaC1tYWlsJTJGbGliJTJGc2VydmljZXMlMkZlbWFpbC1zZXJ2aWNlLnRzJTIyJTJDJTVCJTdCJTIyaWQlMjIlM0ElMjI0MDEwODQ0YzU1YjgzZGZiMWYxMDViZWJlZmUwZWVmZTFlNmI0MzE5MDAlMjIlMkMlMjJleHBvcnRlZE5hbWUlMjIlM0ElMjJ0ZXN0RW1haWxDb25uZWN0aW9uJTIyJTdEJTJDJTdCJTIyaWQlMjIlM0ElMjI2MDI0OWYzM2RjNDFiYWI4NjkzMjAxYTNmMTlmNWU1ZmI0NmU2NDFjMjglMjIlMkMlMjJleHBvcnRlZE5hbWUlMjIlM0ElMjJnZXRJbWFwQ29ubmVjdGlvbiUyMiU3RCUyQyU3QiUyMmlkJTIyJTNBJTIyNjA4NTJhNGU4NDY1MGE3OTc5MmJmN2RiYThlYWI2ZTI5OTRmYWQyNjc0JTIyJTJDJTIyZXhwb3J0ZWROYW1lJTIyJTNBJTIyc2VuZEVtYWlsJTIyJTdEJTJDJTdCJTIyaWQlMjIlM0ElMjI2MDg2NDcxODU1MjFjZGJlYzI1YTZlODNmZDAzOTU5ZDdiZWNkNmE2Y2QlMjIlMkMlMjJleHBvcnRlZE5hbWUlMjIlM0ElMjJnZXRNYWlsYm94ZXMlMjIlN0QlMkMlN0IlMjJpZCUyMiUzQSUyMjYwOWU5N2MwNjFmODdjOWQ5MmU2YjRjMTgwZGUzMTkyNDhlODI2Mzc4NyUyMiUyQyUyMmV4cG9ydGVkTmFtZSUyMiUzQSUyMmdldFVzZXJFbWFpbENyZWRlbnRpYWxzJTIyJTdEJTJDJTdCJTIyaWQlMjIlM0ElMjI3MDU3NjQyMWYzZjFhOGUwYjQ3NjkzZjA2YmYwYjkwMGQzMjE4MDA1OTIlMjIlMkMlMjJleHBvcnRlZE5hbWUlMjIlM0ElMjJzYXZlVXNlckVtYWlsQ3JlZGVudGlhbHMlMjIlN0QlMkMlN0IlMjJpZCUyMiUzQSUyMjc4NzI1YjE1MzkyNzhmNjg0N2FkYmIzZjY3ODAyMGVmYzRkMjA0ZDZkZSUyMiUyQyUyMmV4cG9ydGVkTmFtZSUyMiUzQSUyMmdldEVtYWlsQ29udGVudCUyMiU3RCUyQyU3QiUyMmlkJTIyJTNBJTIyN2M5ZTg2ZDc1NTViYmU0NDliODA4YmQyYWU3NDc5ZjgyYWY5OTQwOWZlJTIyJTJDJTIyZXhwb3J0ZWROYW1lJTIyJTNBJTIybWFya0VtYWlsUmVhZFN0YXR1cyUyMiU3RCUyQyU3QiUyMmlkJTIyJTNBJTIyN2NjYzQ5ZDE5MGE5NDRhZWRlNTE0YjI4YWQ4OGQ0YTUyOTcxMzM2MjExJTIyJTJDJTIyZXhwb3J0ZWROYW1lJTIyJTNBJTIydG9nZ2xlRW1haWxGbGFnJTIyJTdEJTJDJTdCJTIyaWQlMjIlM0ElMjI3ZTZmZWUyZThkNmI1NjYxYzg3MjE5YTgxZGVhMTEwOTA3NzNmMjA2Y2QlMjIlMkMlMjJleHBvcnRlZE5hbWUlMjIlM0ElMjJnZXRFbWFpbHMlMjIlN0QlNUQlNUQlNUQmX19jbGllbnRfaW1wb3J0ZWRfXz0hIiwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7OztBQUNnSjtBQUNGO0FBQ1I7QUFDRztBQUNXO0FBQ0M7QUFDVDtBQUNJO0FBQ0o7QUFDTiIsInNvdXJjZXMiOlsiIl0sInNvdXJjZXNDb250ZW50IjpbIlxuZXhwb3J0IHsgdGVzdEVtYWlsQ29ubmVjdGlvbiBhcyBcIjQwMTA4NDRjNTViODNkZmIxZjEwNWJlYmVmZTBlZWZlMWU2YjQzMTkwMFwiIH0gZnJvbSBcIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbGliL3NlcnZpY2VzL2VtYWlsLXNlcnZpY2UudHNcIlxuZXhwb3J0IHsgZ2V0SW1hcENvbm5lY3Rpb24gYXMgXCI2MDI0OWYzM2RjNDFiYWI4NjkzMjAxYTNmMTlmNWU1ZmI0NmU2NDFjMjhcIiB9IGZyb20gXCIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2xpYi9zZXJ2aWNlcy9lbWFpbC1zZXJ2aWNlLnRzXCJcbmV4cG9ydCB7IHNlbmRFbWFpbCBhcyBcIjYwODUyYTRlODQ2NTBhNzk3OTJiZjdkYmE4ZWFiNmUyOTk0ZmFkMjY3NFwiIH0gZnJvbSBcIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbGliL3NlcnZpY2VzL2VtYWlsLXNlcnZpY2UudHNcIlxuZXhwb3J0IHsgZ2V0TWFpbGJveGVzIGFzIFwiNjA4NjQ3MTg1NTIxY2RiZWMyNWE2ZTgzZmQwMzk1OWQ3YmVjZDZhNmNkXCIgfSBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9saWIvc2VydmljZXMvZW1haWwtc2VydmljZS50c1wiXG5leHBvcnQgeyBnZXRVc2VyRW1haWxDcmVkZW50aWFscyBhcyBcIjYwOWU5N2MwNjFmODdjOWQ5MmU2YjRjMTgwZGUzMTkyNDhlODI2Mzc4N1wiIH0gZnJvbSBcIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbGliL3NlcnZpY2VzL2VtYWlsLXNlcnZpY2UudHNcIlxuZXhwb3J0IHsgc2F2ZVVzZXJFbWFpbENyZWRlbnRpYWxzIGFzIFwiNzA1NzY0MjFmM2YxYThlMGI0NzY5M2YwNmJmMGI5MDBkMzIxODAwNTkyXCIgfSBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9saWIvc2VydmljZXMvZW1haWwtc2VydmljZS50c1wiXG5leHBvcnQgeyBnZXRFbWFpbENvbnRlbnQgYXMgXCI3ODcyNWIxNTM5Mjc4ZjY4NDdhZGJiM2Y2NzgwMjBlZmM0ZDIwNGQ2ZGVcIiB9IGZyb20gXCIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2xpYi9zZXJ2aWNlcy9lbWFpbC1zZXJ2aWNlLnRzXCJcbmV4cG9ydCB7IG1hcmtFbWFpbFJlYWRTdGF0dXMgYXMgXCI3YzllODZkNzU1NWJiZTQ0OWI4MDhiZDJhZTc0NzlmODJhZjk5NDA5ZmVcIiB9IGZyb20gXCIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2xpYi9zZXJ2aWNlcy9lbWFpbC1zZXJ2aWNlLnRzXCJcbmV4cG9ydCB7IHRvZ2dsZUVtYWlsRmxhZyBhcyBcIjdjY2M0OWQxOTBhOTQ0YWVkZTUxNGIyOGFkODhkNGE1Mjk3MTMzNjIxMVwiIH0gZnJvbSBcIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvbGliL3NlcnZpY2VzL2VtYWlsLXNlcnZpY2UudHNcIlxuZXhwb3J0IHsgZ2V0RW1haWxzIGFzIFwiN2U2ZmVlMmU4ZDZiNTY2MWM4NzIxOWE4MWRlYTExMDkwNzczZjIwNmNkXCIgfSBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9saWIvc2VydmljZXMvZW1haWwtc2VydmljZS50c1wiXG4iXSwibmFtZXMiOltdLCJpZ25vcmVMaXN0IjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/next/dist/build/webpack/loaders/next-flight-action-entry-loader.js?actions=%5B%5B%22%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Flib%2Fservices%2Femail-service.ts%22%2C%5B%7B%22id%22%3A%224010844c55b83dfb1f105bebefe0eefe1e6b431900%22%2C%22exportedName%22%3A%22testEmailConnection%22%7D%2C%7B%22id%22%3A%2260249f33dc41bab8693201a3f19f5e5fb46e641c28%22%2C%22exportedName%22%3A%22getImapConnection%22%7D%2C%7B%22id%22%3A%2260852a4e84650a79792bf7dba8eab6e2994fad2674%22%2C%22exportedName%22%3A%22sendEmail%22%7D%2C%7B%22id%22%3A%22608647185521cdbec25a6e83fd03959d7becd6a6cd%22%2C%22exportedName%22%3A%22getMailboxes%22%7D%2C%7B%22id%22%3A%22609e97c061f87c9d92e6b4c180de319248e8263787%22%2C%22exportedName%22%3A%22getUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2270576421f3f1a8e0b47693f06bf0b900d321800592%22%2C%22exportedName%22%3A%22saveUserEmailCredentials%22%7D%2C%7B%22id%22%3A%2278725b1539278f6847adbb3f678020efc4d204d6de%22%2C%22exportedName%22%3A%22getEmailContent%22%7D%2C%7B%22id%22%3A%227c9e86d7555bbe449b808bd2ae7479f82af99409fe%22%2C%22exportedName%22%3A%22markEmailReadStatus%22%7D%2C%7B%22id%22%3A%227ccc49d190a944aede514b28ad88d4a52971336211%22%2C%22exportedName%22%3A%22toggleEmailFlag%22%7D%2C%7B%22id%22%3A%227e6fee2e8d6b5661c87219a81dea11090773f206cd%22%2C%22exportedName%22%3A%22getEmails%22%7D%5D%5D%5D&__client_imported__=!\n");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(rsc)/./node_modules/next/dist/build/webpack/loaders/next-flight-client-entry-loader.js?server=true!":
|
|
/*!******************************************************************************************************!*\
|
|
!*** ./node_modules/next/dist/build/webpack/loaders/next-flight-client-entry-loader.js?server=true! ***!
|
|
\******************************************************************************************************/
|
|
/***/ (() => {
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "(ssr)/./node_modules/next/dist/build/webpack/loaders/next-flight-client-entry-loader.js?server=true!":
|
|
/*!******************************************************************************************************!*\
|
|
!*** ./node_modules/next/dist/build/webpack/loaders/next-flight-client-entry-loader.js?server=true! ***!
|
|
\******************************************************************************************************/
|
|
/***/ (() => {
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "../app-render/after-task-async-storage.external":
|
|
/*!***********************************************************************************!*\
|
|
!*** external "next/dist/server/app-render/after-task-async-storage.external.js" ***!
|
|
\***********************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("next/dist/server/app-render/after-task-async-storage.external.js");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "../app-render/work-async-storage.external":
|
|
/*!*****************************************************************************!*\
|
|
!*** external "next/dist/server/app-render/work-async-storage.external.js" ***!
|
|
\*****************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("next/dist/server/app-render/work-async-storage.external.js");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./work-unit-async-storage.external":
|
|
/*!**********************************************************************************!*\
|
|
!*** external "next/dist/server/app-render/work-unit-async-storage.external.js" ***!
|
|
\**********************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("next/dist/server/app-render/work-unit-async-storage.external.js");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "?d272":
|
|
/*!********************************!*\
|
|
!*** supports-color (ignored) ***!
|
|
\********************************/
|
|
/***/ (() => {
|
|
|
|
/* (ignored) */
|
|
|
|
/***/ }),
|
|
|
|
/***/ "@prisma/client":
|
|
/*!*********************************!*\
|
|
!*** external "@prisma/client" ***!
|
|
\*********************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("@prisma/client");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "assert":
|
|
/*!*************************!*\
|
|
!*** external "assert" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("assert");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "buffer":
|
|
/*!*************************!*\
|
|
!*** external "buffer" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("buffer");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "child_process":
|
|
/*!********************************!*\
|
|
!*** external "child_process" ***!
|
|
\********************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("child_process");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "crypto":
|
|
/*!*************************!*\
|
|
!*** external "crypto" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("crypto");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "dns":
|
|
/*!**********************!*\
|
|
!*** external "dns" ***!
|
|
\**********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("dns");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "events":
|
|
/*!*************************!*\
|
|
!*** external "events" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("events");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "fs":
|
|
/*!*********************!*\
|
|
!*** external "fs" ***!
|
|
\*********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("fs");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "http":
|
|
/*!***********************!*\
|
|
!*** external "http" ***!
|
|
\***********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("http");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "https":
|
|
/*!************************!*\
|
|
!*** external "https" ***!
|
|
\************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("https");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "module":
|
|
/*!*************************!*\
|
|
!*** external "module" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("module");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "net":
|
|
/*!**********************!*\
|
|
!*** external "net" ***!
|
|
\**********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("net");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "next/dist/compiled/next-server/app-page.runtime.dev.js":
|
|
/*!*************************************************************************!*\
|
|
!*** external "next/dist/compiled/next-server/app-page.runtime.dev.js" ***!
|
|
\*************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("next/dist/compiled/next-server/app-page.runtime.dev.js");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "next/dist/compiled/next-server/app-route.runtime.dev.js":
|
|
/*!**************************************************************************!*\
|
|
!*** external "next/dist/compiled/next-server/app-route.runtime.dev.js" ***!
|
|
\**************************************************************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("next/dist/compiled/next-server/app-route.runtime.dev.js");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "node:buffer":
|
|
/*!******************************!*\
|
|
!*** external "node:buffer" ***!
|
|
\******************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("node:buffer");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "node:events":
|
|
/*!******************************!*\
|
|
!*** external "node:events" ***!
|
|
\******************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("node:events");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "node:os":
|
|
/*!**************************!*\
|
|
!*** external "node:os" ***!
|
|
\**************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("node:os");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "node:path":
|
|
/*!****************************!*\
|
|
!*** external "node:path" ***!
|
|
\****************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("node:path");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "node:stream":
|
|
/*!******************************!*\
|
|
!*** external "node:stream" ***!
|
|
\******************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("node:stream");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "os":
|
|
/*!*********************!*\
|
|
!*** external "os" ***!
|
|
\*********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("os");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "path":
|
|
/*!***********************!*\
|
|
!*** external "path" ***!
|
|
\***********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("path");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "querystring":
|
|
/*!******************************!*\
|
|
!*** external "querystring" ***!
|
|
\******************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("querystring");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "stream":
|
|
/*!*************************!*\
|
|
!*** external "stream" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("stream");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "string_decoder":
|
|
/*!*********************************!*\
|
|
!*** external "string_decoder" ***!
|
|
\*********************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("string_decoder");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "tls":
|
|
/*!**********************!*\
|
|
!*** external "tls" ***!
|
|
\**********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("tls");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "tty":
|
|
/*!**********************!*\
|
|
!*** external "tty" ***!
|
|
\**********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("tty");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "url":
|
|
/*!**********************!*\
|
|
!*** external "url" ***!
|
|
\**********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("url");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "util":
|
|
/*!***********************!*\
|
|
!*** external "util" ***!
|
|
\***********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("util");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "worker_threads":
|
|
/*!*********************************!*\
|
|
!*** external "worker_threads" ***!
|
|
\*********************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("worker_threads");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "zlib":
|
|
/*!***********************!*\
|
|
!*** external "zlib" ***!
|
|
\***********************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = require("zlib");
|
|
|
|
/***/ })
|
|
|
|
};
|
|
;
|
|
|
|
// load runtime
|
|
var __webpack_require__ = require("../../../../webpack-runtime.js");
|
|
__webpack_require__.C(exports);
|
|
var __webpack_exec__ = (moduleId) => (__webpack_require__(__webpack_require__.s = moduleId))
|
|
var __webpack_exports__ = __webpack_require__.X(0, ["vendor-chunks/next","vendor-chunks/jose","vendor-chunks/next-auth","vendor-chunks/openid-client","vendor-chunks/@babel","vendor-chunks/oauth","vendor-chunks/preact","vendor-chunks/uuid","vendor-chunks/yallist","vendor-chunks/preact-render-to-string","vendor-chunks/cookie","vendor-chunks/oidc-token-hash","vendor-chunks/@panva","vendor-chunks/jwt-decode","vendor-chunks/crypto-js","vendor-chunks/ioredis","vendor-chunks/@ioredis","vendor-chunks/debug","vendor-chunks/lodash.defaults","vendor-chunks/redis-parser","vendor-chunks/denque","vendor-chunks/cluster-key-slot","vendor-chunks/lodash.isarguments","vendor-chunks/redis-errors","vendor-chunks/ms","vendor-chunks/standard-as-callback","vendor-chunks/nodemailer","vendor-chunks/imapflow","vendor-chunks/encoding-japanese","vendor-chunks/iconv-lite","vendor-chunks/form-data","vendor-chunks/zod","vendor-chunks/axios","vendor-chunks/libmime","vendor-chunks/entities","vendor-chunks/he","vendor-chunks/html-to-text","vendor-chunks/htmlparser2","vendor-chunks/ip-address","vendor-chunks/mailparser","vendor-chunks/socks","vendor-chunks/smart-buffer","vendor-chunks/pino","vendor-chunks/mailsplit","vendor-chunks/jsbn","vendor-chunks/domutils","vendor-chunks/linkify-it","vendor-chunks/domhandler","vendor-chunks/follow-redirects","vendor-chunks/fast-redact","vendor-chunks/safe-stable-stringify","vendor-chunks/sonic-boom","vendor-chunks/thread-stream","vendor-chunks/selderee","vendor-chunks/get-intrinsic","vendor-chunks/parseley","vendor-chunks/punycode.js","vendor-chunks/peberminta","vendor-chunks/tlds","vendor-chunks/pino-std-serializers","vendor-chunks/dom-serializer","vendor-chunks/sprintf-js","vendor-chunks/libqp","vendor-chunks/asynckit","vendor-chunks/libbase64","vendor-chunks/uc.micro","vendor-chunks/combined-stream","vendor-chunks/deepmerge","vendor-chunks/proxy-from-env","vendor-chunks/@selderee","vendor-chunks/quick-format-unescaped","vendor-chunks/has-symbols","vendor-chunks/delayed-stream","vendor-chunks/function-bind","vendor-chunks/safer-buffer","vendor-chunks/domelementtype","vendor-chunks/on-exit-leak-free","vendor-chunks/leac","vendor-chunks/es-set-tostringtag","vendor-chunks/atomic-sleep","vendor-chunks/get-proto","vendor-chunks/call-bind-apply-helpers","vendor-chunks/dunder-proto","vendor-chunks/math-intrinsics","vendor-chunks/es-errors","vendor-chunks/gopd","vendor-chunks/es-define-property","vendor-chunks/hasown","vendor-chunks/has-tostringtag","vendor-chunks/es-object-atoms"], () => (__webpack_exec__("(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fcourrier%2Femails%2Froute&page=%2Fapi%2Fcourrier%2Femails%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fcourrier%2Femails%2Froute.ts&appDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail%2Fapp&pageExtensions=tsx&pageExtensions=ts&pageExtensions=jsx&pageExtensions=js&rootDir=%2Fhome%2Falma%2Fnextgen%2FNeah-mail&isDev=true&tsconfigPath=tsconfig.json&basePath=&assetPrefix=&nextConfigOutput=&preferredRegion=&middlewareConfig=e30%3D!")));
|
|
module.exports = __webpack_exports__;
|
|
|
|
})(); |