NeahNew/.next/server/app/api/redis/status/route.js
2025-05-03 14:17:46 +02:00

275 lines
68 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/redis/status/route";
exports.ids = ["app/api/redis/status/route"];
exports.modules = {
/***/ "(rsc)/./app/api/redis/status/route.ts":
/*!***************************************!*\
!*** ./app/api/redis/status/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 _lib_redis__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @/lib/redis */ \"(rsc)/./lib/redis.ts\");\n\n\n/**\n * API route to check Redis connection status\n * Used for monitoring and debugging\n */ async function GET() {\n try {\n const status = await (0,_lib_redis__WEBPACK_IMPORTED_MODULE_1__.getRedisStatus)();\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json({\n ready: status.status === 'connected',\n status: status.status,\n ping: status.ping,\n error: status.error\n });\n } catch (error) {\n return next_server__WEBPACK_IMPORTED_MODULE_0__.NextResponse.json({\n ready: false,\n error: error instanceof Error ? error.message : 'Unknown error'\n }, {\n status: 500\n });\n }\n}\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiKHJzYykvLi9hcHAvYXBpL3JlZGlzL3N0YXR1cy9yb3V0ZS50cyIsIm1hcHBpbmdzIjoiOzs7Ozs7QUFBMkM7QUFDRTtBQUU3Qzs7O0NBR0MsR0FDTSxlQUFlRTtJQUNwQixJQUFJO1FBQ0YsTUFBTUMsU0FBUyxNQUFNRiwwREFBY0E7UUFFbkMsT0FBT0QscURBQVlBLENBQUNJLElBQUksQ0FBQztZQUN2QkMsT0FBT0YsT0FBT0EsTUFBTSxLQUFLO1lBQ3pCQSxRQUFRQSxPQUFPQSxNQUFNO1lBQ3JCRyxNQUFNSCxPQUFPRyxJQUFJO1lBQ2pCQyxPQUFPSixPQUFPSSxLQUFLO1FBQ3JCO0lBQ0YsRUFBRSxPQUFPQSxPQUFPO1FBQ2QsT0FBT1AscURBQVlBLENBQUNJLElBQUksQ0FBQztZQUN2QkMsT0FBTztZQUNQRSxPQUFPQSxpQkFBaUJDLFFBQVFELE1BQU1FLE9BQU8sR0FBRztRQUNsRCxHQUFHO1lBQUVOLFFBQVE7UUFBSTtJQUNuQjtBQUNGIiwic291cmNlcyI6WyIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2FwcC9hcGkvcmVkaXMvc3RhdHVzL3JvdXRlLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE5leHRSZXNwb25zZSB9IGZyb20gJ25leHQvc2VydmVyJztcbmltcG9ydCB7IGdldFJlZGlzU3RhdHVzIH0gZnJvbSAnQC9saWIvcmVkaXMnO1xuXG4vKipcbiAqIEFQSSByb3V0ZSB0byBjaGVjayBSZWRpcyBjb25uZWN0aW9uIHN0YXR1c1xuICogVXNlZCBmb3IgbW9uaXRvcmluZyBhbmQgZGVidWdnaW5nXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBHRVQoKSB7XG4gIHRyeSB7XG4gICAgY29uc3Qgc3RhdHVzID0gYXdhaXQgZ2V0UmVkaXNTdGF0dXMoKTtcbiAgICBcbiAgICByZXR1cm4gTmV4dFJlc3BvbnNlLmpzb24oe1xuICAgICAgcmVhZHk6IHN0YXR1cy5zdGF0dXMgPT09ICdjb25uZWN0ZWQnLFxuICAgICAgc3RhdHVzOiBzdGF0dXMuc3RhdHVzLFxuICAgICAgcGluZzogc3RhdHVzLnBpbmcsXG4gICAgICBlcnJvcjogc3RhdHVzLmVycm9yXG4gICAgfSk7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgcmV0dXJuIE5leHRSZXNwb25zZS5qc29uKHtcbiAgICAgIHJlYWR5OiBmYWxzZSxcbiAgICAgIGVycm9yOiBlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6ICdVbmtub3duIGVycm9yJ1xuICAgIH0sIHsgc3RhdHVzOiA1MDAgfSk7XG4gIH1cbn0gIl0sIm5hbWVzIjpbIk5leHRSZXNwb25zZSIsImdldFJlZGlzU3RhdHVzIiwiR0VUIiwic3RhdHVzIiwianNvbiIsInJlYWR5IiwicGluZyIsImVycm9yIiwiRXJyb3IiLCJtZXNzYWdlIl0sImlnbm9yZUxpc3QiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///(rsc)/./app/api/redis/status/route.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":"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","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)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fredis%2Fstatus%2Froute&page=%2Fapi%2Fredis%2Fstatus%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fredis%2Fstatus%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%2Fredis%2Fstatus%2Froute&page=%2Fapi%2Fredis%2Fstatus%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fredis%2Fstatus%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_redis_status_route_ts__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./app/api/redis/status/route.ts */ \"(rsc)/./app/api/redis/status/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/redis/status/route\",\n pathname: \"/api/redis/status\",\n filename: \"route\",\n bundlePath: \"app/api/redis/status/route\"\n },\n resolvedPagePath: \"/home/alma/nextgen/Neah-mail/app/api/redis/status/route.ts\",\n nextConfigOutput,\n userland: _home_alma_nextgen_Neah_mail_app_api_redis_status_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/bmFtZT1hcHAlMkZhcGklMkZyZWRpcyUyRnN0YXR1cyUyRnJvdXRlJnBhZ2U9JTJGYXBpJTJGcmVkaXMlMkZzdGF0dXMlMkZyb3V0ZSZhcHBQYXRocz0mcGFnZVBhdGg9cHJpdmF0ZS1uZXh0LWFwcC1kaXIlMkZhcGklMkZyZWRpcyUyRnN0YXR1cyUyRnJvdXRlLnRzJmFwcERpcj0lMkZob21lJTJGYWxtYSUyRm5leHRnZW4lMkZOZWFoLW1haWwlMkZhcHAmcGFnZUV4dGVuc2lvbnM9dHN4JnBhZ2VFeHRlbnNpb25zPXRzJnBhZ2VFeHRlbnNpb25zPWpzeCZwYWdlRXh0ZW5zaW9ucz1qcyZyb290RGlyPSUyRmhvbWUlMkZhbG1hJTJGbmV4dGdlbiUyRk5lYWgtbWFpbCZpc0Rldj10cnVlJnRzY29uZmlnUGF0aD10c2NvbmZpZy5qc29uJmJhc2VQYXRoPSZhc3NldFByZWZpeD0mbmV4dENvbmZpZ091dHB1dD0mcHJlZmVycmVkUmVnaW9uPSZtaWRkbGV3YXJlQ29uZmlnPWUzMCUzRCEiLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7QUFBK0Y7QUFDdkM7QUFDcUI7QUFDVTtBQUN2RjtBQUNBO0FBQ0E7QUFDQSx3QkFBd0IseUdBQW1CO0FBQzNDO0FBQ0EsY0FBYyxrRUFBUztBQUN2QjtBQUNBO0FBQ0E7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUNBO0FBQ0EsWUFBWTtBQUNaLENBQUM7QUFDRDtBQUNBO0FBQ0E7QUFDQSxRQUFRLHNEQUFzRDtBQUM5RDtBQUNBLFdBQVcsNEVBQVc7QUFDdEI7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUMwRjs7QUFFMUYiLCJzb3VyY2VzIjpbIiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcHBSb3V0ZVJvdXRlTW9kdWxlIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvcm91dGUtbW9kdWxlcy9hcHAtcm91dGUvbW9kdWxlLmNvbXBpbGVkXCI7XG5pbXBvcnQgeyBSb3V0ZUtpbmQgfSBmcm9tIFwibmV4dC9kaXN0L3NlcnZlci9yb3V0ZS1raW5kXCI7XG5pbXBvcnQgeyBwYXRjaEZldGNoIGFzIF9wYXRjaEZldGNoIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvbGliL3BhdGNoLWZldGNoXCI7XG5pbXBvcnQgKiBhcyB1c2VybGFuZCBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9hcHAvYXBpL3JlZGlzL3N0YXR1cy9yb3V0ZS50c1wiO1xuLy8gV2UgaW5qZWN0IHRoZSBuZXh0Q29uZmlnT3V0cHV0IGhlcmUgc28gdGhhdCB3ZSBjYW4gdXNlIHRoZW0gaW4gdGhlIHJvdXRlXG4vLyBtb2R1bGUuXG5jb25zdCBuZXh0Q29uZmlnT3V0cHV0ID0gXCJcIlxuY29uc3Qgcm91dGVNb2R1bGUgPSBuZXcgQXBwUm91dGVSb3V0ZU1vZHVsZSh7XG4gICAgZGVmaW5pdGlvbjoge1xuICAgICAgICBraW5kOiBSb3V0ZUtpbmQuQVBQX1JPVVRFLFxuICAgICAgICBwYWdlOiBcIi9hcGkvcmVkaXMvc3RhdHVzL3JvdXRlXCIsXG4gICAgICAgIHBhdGhuYW1lOiBcIi9hcGkvcmVkaXMvc3RhdHVzXCIsXG4gICAgICAgIGZpbGVuYW1lOiBcInJvdXRlXCIsXG4gICAgICAgIGJ1bmRsZVBhdGg6IFwiYXBwL2FwaS9yZWRpcy9zdGF0dXMvcm91dGVcIlxuICAgIH0sXG4gICAgcmVzb2x2ZWRQYWdlUGF0aDogXCIvaG9tZS9hbG1hL25leHRnZW4vTmVhaC1tYWlsL2FwcC9hcGkvcmVkaXMvc3RhdHVzL3JvdXRlLnRzXCIsXG4gICAgbmV4dENvbmZpZ091dHB1dCxcbiAgICB1c2VybGFuZFxufSk7XG4vLyBQdWxsIG91dCB0aGUgZXhwb3J0cyB0aGF0IHdlIG5lZWQgdG8gZXhwb3NlIGZyb20gdGhlIG1vZHVsZS4gVGhpcyBzaG91bGRcbi8vIGJlIGVsaW1pbmF0ZWQgd2hlbiB3ZSd2ZSBtb3ZlZCB0aGUgb3RoZXIgcm91dGVzIHRvIHRoZSBuZXcgZm9ybWF0LiBUaGVzZVxuLy8gYXJlIHVzZWQgdG8gaG9vayBpbnRvIHRoZSByb3V0ZS5cbmNvbnN0IHsgd29ya0FzeW5jU3RvcmFnZSwgd29ya1VuaXRBc3luY1N0b3JhZ2UsIHNlcnZlckhvb2tzIH0gPSByb3V0ZU1vZHVsZTtcbmZ1bmN0aW9uIHBhdGNoRmV0Y2goKSB7XG4gICAgcmV0dXJuIF9wYXRjaEZldGNoKHtcbiAgICAgICAgd29ya0FzeW5jU3RvcmFnZSxcbiAgICAgICAgd29ya1VuaXRBc3luY1N0b3JhZ2VcbiAgICB9KTtcbn1cbmV4cG9ydCB7IHJvdXRlTW9kdWxlLCB3b3JrQXN5bmNTdG9yYWdlLCB3b3JrVW5pdEFzeW5jU3RvcmFnZSwgc2VydmVySG9va3MsIHBhdGNoRmV0Y2gsICB9O1xuXG4vLyMgc291cmNlTWFwcGluZ1VSTD1hcHAtcm91dGUuanMubWFwIl0sIm5hbWVzIjpbXSwiaWdub3JlTGlzdCI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fredis%2Fstatus%2Froute&page=%2Fapi%2Fredis%2Fstatus%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fredis%2Fstatus%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-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) */
/***/ }),
/***/ "assert":
/*!*************************!*\
!*** external "assert" ***!
\*************************/
/***/ ((module) => {
"use strict";
module.exports = require("assert");
/***/ }),
/***/ "buffer":
/*!*************************!*\
!*** external "buffer" ***!
\*************************/
/***/ ((module) => {
"use strict";
module.exports = require("buffer");
/***/ }),
/***/ "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");
/***/ }),
/***/ "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");
/***/ }),
/***/ "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");
/***/ })
};
;
// 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/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"], () => (__webpack_exec__("(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fredis%2Fstatus%2Froute&page=%2Fapi%2Fredis%2Fstatus%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fredis%2Fstatus%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__;
})();