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

243 lines
99 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/users/route";
exports.ids = ["app/api/users/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/users/route.ts":
/*!********************************!*\
!*** ./app/api/users/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 */ DELETE: () => (/* binding */ DELETE),\n/* harmony export */ GET: () => (/* binding */ GET),\n/* harmony export */ POST: () => (/* binding */ POST)\n/* harmony export */ });\n/* harmony import */ var next_auth_next__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! next-auth/next */ \"(rsc)/./node_modules/next-auth/next/index.js\");\n/* harmony import */ var _app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! @/app/api/auth/[...nextauth]/route */ \"(rsc)/./app/api/auth/[...nextauth]/route.ts\");\n/* harmony import */ var next_server__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! next/server */ \"(rsc)/./node_modules/next/dist/api/server.js\");\n\n\n\nasync function GET() {\n const session = await (0,next_auth_next__WEBPACK_IMPORTED_MODULE_0__.getServerSession)(_app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_1__.authOptions);\n if (!session) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Non autorisé\"\n }, {\n status: 401\n });\n }\n console.log(\"Session:\", {\n accessToken: session.accessToken?.substring(0, 20) + \"...\",\n user: session.user\n });\n try {\n // Get client credentials token\n const tokenResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/realms/${process.env.KEYCLOAK_REALM}/protocol/openid-connect/token`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded'\n },\n body: new URLSearchParams({\n grant_type: 'client_credentials',\n client_id: process.env.KEYCLOAK_CLIENT_ID,\n client_secret: process.env.KEYCLOAK_CLIENT_SECRET\n })\n });\n const tokenData = await tokenResponse.json();\n console.log(\"Token response:\", {\n ok: tokenResponse.ok,\n status: tokenResponse.status,\n data: tokenData.access_token ? \"Token received\" : tokenData\n });\n if (!tokenResponse.ok) {\n console.error(\"Failed to get token:\", tokenData);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json([\n getCurrentUser(session)\n ]);\n }\n // Get users list with brief=false to get full user details\n const usersResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users?briefRepresentation=false`, {\n headers: {\n Authorization: `Bearer ${tokenData.access_token}`,\n 'Content-Type': 'application/json'\n }\n });\n if (!usersResponse.ok) {\n console.error(\"Failed to fetch users:\", await usersResponse.text());\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json([\n getCurrentUser(session)\n ]);\n }\n const users = await usersResponse.json();\n console.log(\"Raw users data:\", users.map((u)=>({\n id: u.id,\n username: u.username,\n realm: u.realm,\n serviceAccountClientId: u.serviceAccountClientId\n })));\n // Filter out service accounts and users from other realms\n const filteredUsers = users.filter((user)=>!user.serviceAccountClientId && // Remove service accounts\n (!user.realm || user.realm === process.env.KEYCLOAK_REALM // Only users from our realm\n ));\n console.log(\"Filtered users count:\", filteredUsers.length);\n // Fetch roles for each user\n const usersWithRoles = await Promise.all(filteredUsers.map(async (user)=>{\n try {\n const rolesResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${user.id}/role-mappings/realm`, {\n headers: {\n Authorization: `Bearer ${tokenData.access_token}`,\n 'Content-Type': 'application/json'\n }\n });\n let roles = [];\n if (rolesResponse.ok) {\n const rolesData = await rolesResponse.json();\n roles = rolesData.filter((role)=>!role.name.startsWith('default-roles-') && ![\n 'offline_access',\n 'uma_authorization'\n ].includes(role.name)).map((role)=>role.name);\n console.log(`Roles for user ${user.username}:`, roles);\n }\n return {\n id: user.id,\n username: user.username,\n firstName: user.firstName || '',\n lastName: user.lastName || '',\n email: user.email,\n createdTimestamp: user.createdTimestamp,\n enabled: user.enabled,\n roles: roles\n };\n } catch (error) {\n console.error(`Error fetching roles for user ${user.id}:`, error);\n return {\n id: user.id,\n username: user.username,\n firstName: user.firstName || '',\n lastName: user.lastName || '',\n email: user.email,\n createdTimestamp: user.createdTimestamp,\n enabled: user.enabled,\n roles: []\n };\n }\n }));\n console.log(\"Final users data:\", usersWithRoles.map((u)=>({\n username: u.username,\n roles: u.roles\n })));\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json(usersWithRoles);\n } catch (error) {\n console.error(\"Error:\", error);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json([\n getCurrentUser(session)\n ]);\n }\n}\n// Helper function to get current user data\nfunction getCurrentUser(session) {\n return {\n id: session.user.id,\n username: session.user.username,\n firstName: session.user.first_name,\n lastName: session.user.last_name,\n email: session.user.email,\n createdTimestamp: Date.now(),\n roles: session.user.role || []\n };\n}\nasync function getAdminToken() {\n try {\n const tokenResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/realms/${process.env.KEYCLOAK_REALM}/protocol/openid-connect/token`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/x-www-form-urlencoded'\n },\n body: new URLSearchParams({\n grant_type: 'client_credentials',\n client_id: process.env.KEYCLOAK_CLIENT_ID,\n client_secret: process.env.KEYCLOAK_CLIENT_SECRET\n })\n });\n const data = await tokenResponse.json();\n if (!tokenResponse.ok || !data.access_token) {\n console.error('Token Error:', data);\n return null;\n }\n return data.access_token;\n } catch (error) {\n console.error('Token Error:', error);\n return null;\n }\n}\n// Validate username according to Keycloak requirements\nfunction validateUsername(username) {\n // Keycloak username requirements:\n // - Only alphanumeric characters, dots (.), hyphens (-), and underscores (_)\n // - Must start with a letter or number\n // - Must be between 3 and 255 characters\n const usernameRegex = /^[a-zA-Z0-9][a-zA-Z0-9._-]{2,254}$/;\n if (!usernameRegex.test(username)) {\n return {\n isValid: false,\n error: \"Le nom d'utilisateur doit commencer par une lettre ou un chiffre, ne contenir que des lettres, chiffres, points, tirets et underscores, et faire entre 3 et 255 caractères\"\n };\n }\n return {\n isValid: true\n };\n}\n// Helper function to create user in Leantime\nasync function createLeantimeUser(userData) {\n try {\n const response = await fetch('https://agilite.slm-lab.net/api/jsonrpc', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-API-Key': process.env.LEANTIME_TOKEN || ''\n },\n body: JSON.stringify({\n method: 'leantime.rpc.Users.Users.addUser',\n jsonrpc: '2.0',\n id: 1,\n params: {\n values: {\n '0': 0,\n '1': userData.lastName,\n '2': userData.firstName,\n '3': '20',\n '4': '',\n '5': 'a',\n '6': userData.email,\n '7': 0,\n '8': 0,\n '9': null,\n '10': '',\n '11': '',\n '12': '',\n '13': new Date().toISOString(),\n lastname: userData.lastName,\n firstname: userData.firstName,\n role: '20',\n profileId: '',\n status: 'a',\n username: userData.email,\n password: userData.password,\n twoFAEnabled: 0,\n clientId: 0,\n clientName: null,\n jobTitle: '',\n jobLevel: '',\n department: '',\n modified: new Date().toISOString(),\n createdOn: new Date().toISOString(),\n source: 'keycloak',\n notifications: 1,\n settings: '{}'\n }\n }\n })\n });\n const data = await response.json();\n console.log('Leantime response:', data);\n if (!response.ok || !data.result) {\n console.error('Leantime user creation failed:', data);\n return {\n success: false,\n error: data.error?.message || 'Failed to create user in Leantime'\n };\n }\n return {\n success: true\n };\n } catch (error) {\n console.error('Error creating Leantime user:', error);\n return {\n success: false,\n error: 'Error creating user in Leantime'\n };\n }\n}\nasync function POST(req) {\n const session = await (0,next_auth_next__WEBPACK_IMPORTED_MODULE_0__.getServerSession)(_app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_1__.authOptions);\n if (!session) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Non autorisé\"\n }, {\n status: 401\n });\n }\n try {\n const data = await req.json();\n console.log(\"Creating user:\", data);\n // Validate username\n const usernameValidation = validateUsername(data.username);\n if (!usernameValidation.isValid) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: usernameValidation.error\n }, {\n status: 400\n });\n }\n const token = await getAdminToken();\n if (!token) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur d'authentification\"\n }, {\n status: 401\n });\n }\n // First, get all available roles from Keycloak\n const rolesResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/roles`, {\n headers: {\n Authorization: `Bearer ${token}`\n }\n });\n if (!rolesResponse.ok) {\n const errorData = await rolesResponse.json();\n console.error(\"Failed to fetch roles:\", errorData);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur lors de la récupération des rôles\"\n }, {\n status: rolesResponse.status\n });\n }\n const availableRoles = await rolesResponse.json();\n console.log(\"Available roles:\", availableRoles);\n // Verify that the requested roles exist\n const requestedRoles = data.roles || [];\n const validRoles = requestedRoles.filter((roleName)=>availableRoles.some((r)=>r.name === roleName));\n if (validRoles.length === 0) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Aucun rôle valide n'a été spécifié\"\n }, {\n status: 400\n });\n }\n // Create the user in Keycloak\n const createResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${token}`,\n \"Content-Type\": \"application/json\"\n },\n body: JSON.stringify({\n username: data.username,\n enabled: true,\n emailVerified: true,\n firstName: data.firstName,\n lastName: data.lastName,\n email: data.email,\n credentials: [\n {\n type: \"password\",\n value: data.password,\n temporary: false\n }\n ]\n })\n });\n console.log(\"Keycloak create response:\", {\n status: createResponse.status,\n ok: createResponse.ok\n });\n if (!createResponse.ok) {\n const errorData = await createResponse.json();\n console.error(\"Keycloak error:\", errorData);\n if (errorData.errorMessage?.includes(\"User exists with same username\")) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Un utilisateur existe déjà avec ce nom d'utilisateur\"\n }, {\n status: 400\n });\n } else if (errorData.errorMessage?.includes(\"User exists with same email\")) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Un utilisateur existe déjà avec cet email\"\n }, {\n status: 400\n });\n }\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur création utilisateur\",\n details: errorData\n }, {\n status: 400\n });\n }\n // Get the created user\n const userResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users?username=${data.username}`, {\n headers: {\n Authorization: `Bearer ${token}`\n }\n });\n const users = await userResponse.json();\n const user = users[0];\n if (!user) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Utilisateur créé mais impossible de le récupérer\"\n }, {\n status: 500\n });\n }\n // Add roles to the user\n const roleObjects = validRoles.map((roleName)=>availableRoles.find((r)=>r.name === roleName));\n const roleResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${user.id}/role-mappings/realm`, {\n method: \"POST\",\n headers: {\n Authorization: `Bearer ${token}`,\n \"Content-Type\": \"application/json\"\n },\n body: JSON.stringify(roleObjects)\n });\n if (!roleResponse.ok) {\n const errorData = await roleResponse.json();\n console.error(\"Failed to add roles:\", errorData);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur lors de l'ajout des rôles\",\n details: errorData\n }, {\n status: 500\n });\n }\n // Create user in Leantime\n const leantimeResult = await createLeantimeUser({\n username: data.username,\n firstName: data.firstName,\n lastName: data.lastName,\n email: data.email,\n password: data.password\n });\n if (!leantimeResult.success) {\n console.error(\"Leantime user creation failed:\", leantimeResult.error);\n // We don't return an error here since Keycloak user was created successfully\n // We just log the error and continue\n }\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n success: true,\n user: {\n ...user,\n roles: validRoles\n }\n });\n } catch (error) {\n console.error(\"Error creating user:\", error);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur serveur\",\n details: error\n }, {\n status: 500\n });\n }\n}\n// Helper function to delete user from Leantime\nasync function deleteLeantimeUser(email) {\n try {\n const response = await fetch('https://agilite.slm-lab.net/api/jsonrpc', {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'X-API-Key': process.env.LEANTIME_TOKEN || ''\n },\n body: JSON.stringify({\n method: 'leantime.rpc.Users.Users.deleteUser',\n jsonrpc: '2.0',\n id: 1,\n params: {\n email: email\n }\n })\n });\n const data = await response.json();\n console.log('Leantime delete response:', data);\n if (!response.ok || !data.result) {\n console.error('Leantime user deletion failed:', data);\n return {\n success: false,\n error: data.error?.message || 'Failed to delete user in Leantime'\n };\n }\n return {\n success: true\n };\n } catch (error) {\n console.error('Error deleting Leantime user:', error);\n return {\n success: false,\n error: 'Error deleting user in Leantime'\n };\n }\n}\nasync function DELETE(req) {\n const session = await (0,next_auth_next__WEBPACK_IMPORTED_MODULE_0__.getServerSession)(_app_api_auth_nextauth_route__WEBPACK_IMPORTED_MODULE_1__.authOptions);\n if (!session) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Non autorisé\"\n }, {\n status: 401\n });\n }\n try {\n const { searchParams } = new URL(req.url);\n const userId = searchParams.get('id');\n const email = searchParams.get('email');\n if (!userId || !email) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"ID utilisateur et email requis\"\n }, {\n status: 400\n });\n }\n const token = await getAdminToken();\n if (!token) {\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur d'authentification\"\n }, {\n status: 401\n });\n }\n // Delete user from Keycloak\n const deleteResponse = await fetch(`${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${userId}`, {\n method: \"DELETE\",\n headers: {\n Authorization: `Bearer ${token}`\n }\n });\n if (!deleteResponse.ok) {\n const errorData = await deleteResponse.json();\n console.error(\"Keycloak delete error:\", errorData);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur lors de la suppression de l'utilisateur\",\n details: errorData\n }, {\n status: deleteResponse.status\n });\n }\n // Delete user from Leantime\n const leantimeResult = await deleteLeantimeUser(email);\n if (!leantimeResult.success) {\n console.error(\"Leantime user deletion failed:\", leantimeResult.error);\n // We don't return an error here since Keycloak user was deleted successfully\n // We just log the error and continue\n }\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n success: true\n });\n } catch (error) {\n console.error(\"Error deleting user:\", error);\n return next_server__WEBPACK_IMPORTED_MODULE_2__.NextResponse.json({\n error: \"Erreur serveur\",\n details: error\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/users/route.ts","mappings":"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","sources":["/home/alma/nextgen/Neah-mail/app/api/users/route.ts"],"sourcesContent":["import { getServerSession } from \"next-auth/next\";\nimport { authOptions } from \"@/app/api/auth/[...nextauth]/route\";\nimport { NextResponse } from \"next/server\";\n\nexport async function GET() {\n  const session = await getServerSession(authOptions);\n\n  if (!session) {\n    return NextResponse.json({ error: \"Non autorisé\" }, { status: 401 });\n  }\n\n  console.log(\"Session:\", {\n    accessToken: session.accessToken?.substring(0, 20) + \"...\",\n    user: session.user,\n  });\n\n  try {\n    // Get client credentials token\n    const tokenResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/realms/${process.env.KEYCLOAK_REALM}/protocol/openid-connect/token`,\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/x-www-form-urlencoded',\n        },\n        body: new URLSearchParams({\n          grant_type: 'client_credentials',\n          client_id: process.env.KEYCLOAK_CLIENT_ID!,\n          client_secret: process.env.KEYCLOAK_CLIENT_SECRET!,\n        }),\n      }\n    );\n\n    const tokenData = await tokenResponse.json();\n    console.log(\"Token response:\", {\n      ok: tokenResponse.ok,\n      status: tokenResponse.status,\n      data: tokenData.access_token ? \"Token received\" : tokenData,\n    });\n\n    if (!tokenResponse.ok) {\n      console.error(\"Failed to get token:\", tokenData);\n      return NextResponse.json([getCurrentUser(session)]);\n    }\n\n    // Get users list with brief=false to get full user details\n    const usersResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users?briefRepresentation=false`,\n      {\n        headers: {\n          Authorization: `Bearer ${tokenData.access_token}`,\n          'Content-Type': 'application/json',\n        },\n      }\n    );\n\n    if (!usersResponse.ok) {\n      console.error(\"Failed to fetch users:\", await usersResponse.text());\n      return NextResponse.json([getCurrentUser(session)]);\n    }\n\n    const users = await usersResponse.json();\n    console.log(\"Raw users data:\", users.map((u: any) => ({\n      id: u.id,\n      username: u.username,\n      realm: u.realm,\n      serviceAccountClientId: u.serviceAccountClientId,\n    })));\n\n    // Filter out service accounts and users from other realms\n    const filteredUsers = users.filter((user: any) => \n      !user.serviceAccountClientId && // Remove service accounts\n      (!user.realm || user.realm === process.env.KEYCLOAK_REALM) // Only users from our realm\n    );\n\n    console.log(\"Filtered users count:\", filteredUsers.length);\n\n    // Fetch roles for each user\n    const usersWithRoles = await Promise.all(filteredUsers.map(async (user: any) => {\n      try {\n        const rolesResponse = await fetch(\n          `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${user.id}/role-mappings/realm`,\n          {\n            headers: {\n              Authorization: `Bearer ${tokenData.access_token}`,\n              'Content-Type': 'application/json',\n            },\n          }\n        );\n\n        let roles = [];\n        if (rolesResponse.ok) {\n          const rolesData = await rolesResponse.json();\n          roles = rolesData\n            .filter((role: any) => \n              !role.name.startsWith('default-roles-') && \n              !['offline_access', 'uma_authorization'].includes(role.name)\n            )\n            .map((role: any) => role.name);\n          console.log(`Roles for user ${user.username}:`, roles);\n        }\n\n        return {\n          id: user.id,\n          username: user.username,\n          firstName: user.firstName || '',\n          lastName: user.lastName || '',\n          email: user.email,\n          createdTimestamp: user.createdTimestamp,\n          enabled: user.enabled,\n          roles: roles,\n        };\n      } catch (error) {\n        console.error(`Error fetching roles for user ${user.id}:`, error);\n        return {\n          id: user.id,\n          username: user.username,\n          firstName: user.firstName || '',\n          lastName: user.lastName || '',\n          email: user.email,\n          createdTimestamp: user.createdTimestamp,\n          enabled: user.enabled,\n          roles: [],\n        };\n      }\n    }));\n\n    console.log(\"Final users data:\", usersWithRoles.map(u => ({\n      username: u.username,\n      roles: u.roles,\n    })));\n\n    return NextResponse.json(usersWithRoles);\n\n  } catch (error) {\n    console.error(\"Error:\", error);\n    return NextResponse.json([getCurrentUser(session)]);\n  }\n}\n\n// Helper function to get current user data\nfunction getCurrentUser(session: any) {\n  return {\n    id: session.user.id,\n    username: session.user.username,\n    firstName: session.user.first_name,\n    lastName: session.user.last_name,\n    email: session.user.email,\n    createdTimestamp: Date.now(),\n    roles: session.user.role || [],\n  };\n}\n\nasync function getAdminToken() {\n  try {\n    const tokenResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/realms/${process.env.KEYCLOAK_REALM}/protocol/openid-connect/token`,\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/x-www-form-urlencoded',\n        },\n        body: new URLSearchParams({\n          grant_type: 'client_credentials',\n          client_id: process.env.KEYCLOAK_CLIENT_ID!,\n          client_secret: process.env.KEYCLOAK_CLIENT_SECRET!,\n        }),\n      }\n    );\n\n    const data = await tokenResponse.json();\n\n    if (!tokenResponse.ok || !data.access_token) {\n      console.error('Token Error:', data);\n      return null;\n    }\n\n    return data.access_token;\n  } catch (error) {\n    console.error('Token Error:', error);\n    return null;\n  }\n}\n\n// Validate username according to Keycloak requirements\nfunction validateUsername(username: string): { isValid: boolean; error?: string } {\n  // Keycloak username requirements:\n  // - Only alphanumeric characters, dots (.), hyphens (-), and underscores (_)\n  // - Must start with a letter or number\n  // - Must be between 3 and 255 characters\n  const usernameRegex = /^[a-zA-Z0-9][a-zA-Z0-9._-]{2,254}$/;\n  \n  if (!usernameRegex.test(username)) {\n    return {\n      isValid: false,\n      error: \"Le nom d'utilisateur doit commencer par une lettre ou un chiffre, ne contenir que des lettres, chiffres, points, tirets et underscores, et faire entre 3 et 255 caractères\"\n    };\n  }\n  \n  return { isValid: true };\n}\n\n// Helper function to create user in Leantime\nasync function createLeantimeUser(userData: {\n  username: string;\n  firstName: string;\n  lastName: string;\n  email: string;\n  password: string;\n}): Promise<{ success: boolean; error?: string }> {\n  try {\n    const response = await fetch('https://agilite.slm-lab.net/api/jsonrpc', {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'X-API-Key': process.env.LEANTIME_TOKEN || '',\n      },\n      body: JSON.stringify({\n        method: 'leantime.rpc.Users.Users.addUser',\n        jsonrpc: '2.0',\n        id: 1,\n        params: {\n          values: {\n            '0': 0, // This will be set by Leantime\n            '1': userData.lastName,\n            '2': userData.firstName,\n            '3': '20', // Default role\n            '4': '', // profileId\n            '5': 'a', // status\n            '6': userData.email,\n            '7': 0, // twoFAEnabled\n            '8': 0, // clientId\n            '9': null, // clientName\n            '10': '', // jobTitle\n            '11': '', // jobLevel\n            '12': '', // department\n            '13': new Date().toISOString(), // modified\n            lastname: userData.lastName,\n            firstname: userData.firstName,\n            role: '20', // Default role\n            profileId: '',\n            status: 'a',\n            username: userData.email,\n            password: userData.password,\n            twoFAEnabled: 0,\n            clientId: 0,\n            clientName: null,\n            jobTitle: '',\n            jobLevel: '',\n            department: '',\n            modified: new Date().toISOString(),\n            createdOn: new Date().toISOString(),\n            source: 'keycloak',\n            notifications: 1,\n            settings: '{}'\n          }\n        }\n      })\n    });\n\n    const data = await response.json();\n    console.log('Leantime response:', data);\n\n    if (!response.ok || !data.result) {\n      console.error('Leantime user creation failed:', data);\n      return { \n        success: false, \n        error: data.error?.message || 'Failed to create user in Leantime' \n      };\n    }\n\n    return { success: true };\n  } catch (error) {\n    console.error('Error creating Leantime user:', error);\n    return { \n      success: false, \n      error: 'Error creating user in Leantime' \n    };\n  }\n}\n\nexport async function POST(req: Request) {\n  const session = await getServerSession(authOptions);\n\n  if (!session) {\n    return NextResponse.json({ error: \"Non autorisé\" }, { status: 401 });\n  }\n\n  try {\n    const data = await req.json();\n    console.log(\"Creating user:\", data);\n\n    // Validate username\n    const usernameValidation = validateUsername(data.username);\n    if (!usernameValidation.isValid) {\n      return NextResponse.json(\n        { error: usernameValidation.error },\n        { status: 400 }\n      );\n    }\n\n    const token = await getAdminToken();\n    if (!token) {\n      return NextResponse.json({ error: \"Erreur d'authentification\" }, { status: 401 });\n    }\n\n    // First, get all available roles from Keycloak\n    const rolesResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/roles`,\n      {\n        headers: {\n          Authorization: `Bearer ${token}`,\n        },\n      }\n    );\n\n    if (!rolesResponse.ok) {\n      const errorData = await rolesResponse.json();\n      console.error(\"Failed to fetch roles:\", errorData);\n      return NextResponse.json({ error: \"Erreur lors de la récupération des rôles\" }, { status: rolesResponse.status });\n    }\n\n    const availableRoles = await rolesResponse.json();\n    console.log(\"Available roles:\", availableRoles);\n\n    // Verify that the requested roles exist\n    const requestedRoles = data.roles || [];\n    const validRoles = requestedRoles.filter((roleName: string) => \n      availableRoles.some((r: any) => r.name === roleName)\n    );\n\n    if (validRoles.length === 0) {\n      return NextResponse.json(\n        { error: \"Aucun rôle valide n'a été spécifié\" },\n        { status: 400 }\n      );\n    }\n\n    // Create the user in Keycloak\n    const createResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users`,\n      {\n        method: \"POST\",\n        headers: {\n          Authorization: `Bearer ${token}`,\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify({\n          username: data.username,\n          enabled: true,\n          emailVerified: true,\n          firstName: data.firstName,\n          lastName: data.lastName,\n          email: data.email,\n          credentials: [\n            {\n              type: \"password\",\n              value: data.password,\n              temporary: false,\n            },\n          ],\n        }),\n      }\n    );\n\n    console.log(\"Keycloak create response:\", {\n      status: createResponse.status,\n      ok: createResponse.ok\n    });\n\n    if (!createResponse.ok) {\n      const errorData = await createResponse.json();\n      console.error(\"Keycloak error:\", errorData);\n      \n      if (errorData.errorMessage?.includes(\"User exists with same username\")) {\n        return NextResponse.json(\n          { error: \"Un utilisateur existe déjà avec ce nom d'utilisateur\" },\n          { status: 400 }\n        );\n      } else if (errorData.errorMessage?.includes(\"User exists with same email\")) {\n        return NextResponse.json(\n          { error: \"Un utilisateur existe déjà avec cet email\" },\n          { status: 400 }\n        );\n      }\n      return NextResponse.json(\n        { error: \"Erreur création utilisateur\", details: errorData },\n        { status: 400 }\n      );\n    }\n\n    // Get the created user\n    const userResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users?username=${data.username}`,\n      {\n        headers: {\n          Authorization: `Bearer ${token}`,\n        },\n      }\n    );\n\n    const users = await userResponse.json();\n    const user = users[0];\n\n    if (!user) {\n      return NextResponse.json(\n        { error: \"Utilisateur créé mais impossible de le récupérer\" },\n        { status: 500 }\n      );\n    }\n\n    // Add roles to the user\n    const roleObjects = validRoles.map((roleName: string) => \n      availableRoles.find((r: any) => r.name === roleName)\n    );\n\n    const roleResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${user.id}/role-mappings/realm`,\n      {\n        method: \"POST\",\n        headers: {\n          Authorization: `Bearer ${token}`,\n          \"Content-Type\": \"application/json\",\n        },\n        body: JSON.stringify(roleObjects),\n      }\n    );\n\n    if (!roleResponse.ok) {\n      const errorData = await roleResponse.json();\n      console.error(\"Failed to add roles:\", errorData);\n      return NextResponse.json(\n        { error: \"Erreur lors de l'ajout des rôles\", details: errorData },\n        { status: 500 }\n      );\n    }\n\n    // Create user in Leantime\n    const leantimeResult = await createLeantimeUser({\n      username: data.username,\n      firstName: data.firstName,\n      lastName: data.lastName,\n      email: data.email,\n      password: data.password,\n    });\n\n    if (!leantimeResult.success) {\n      console.error(\"Leantime user creation failed:\", leantimeResult.error);\n      // We don't return an error here since Keycloak user was created successfully\n      // We just log the error and continue\n    }\n\n    return NextResponse.json({\n      success: true,\n      user: {\n        ...user,\n        roles: validRoles,\n      },\n    });\n\n  } catch (error) {\n    console.error(\"Error creating user:\", error);\n    return NextResponse.json(\n      { error: \"Erreur serveur\", details: error },\n      { status: 500 }\n    );\n  }\n}\n\n// Helper function to delete user from Leantime\nasync function deleteLeantimeUser(email: string): Promise<{ success: boolean; error?: string }> {\n  try {\n    const response = await fetch('https://agilite.slm-lab.net/api/jsonrpc', {\n      method: 'POST',\n      headers: {\n        'Content-Type': 'application/json',\n        'X-API-Key': process.env.LEANTIME_TOKEN || '',\n      },\n      body: JSON.stringify({\n        method: 'leantime.rpc.Users.Users.deleteUser',\n        jsonrpc: '2.0',\n        id: 1,\n        params: {\n          email: email\n        }\n      })\n    });\n\n    const data = await response.json();\n    console.log('Leantime delete response:', data);\n\n    if (!response.ok || !data.result) {\n      console.error('Leantime user deletion failed:', data);\n      return { \n        success: false, \n        error: data.error?.message || 'Failed to delete user in Leantime' \n      };\n    }\n\n    return { success: true };\n  } catch (error) {\n    console.error('Error deleting Leantime user:', error);\n    return { \n      success: false, \n      error: 'Error deleting user in Leantime' \n    };\n  }\n}\n\nexport async function DELETE(req: Request) {\n  const session = await getServerSession(authOptions);\n\n  if (!session) {\n    return NextResponse.json({ error: \"Non autorisé\" }, { status: 401 });\n  }\n\n  try {\n    const { searchParams } = new URL(req.url);\n    const userId = searchParams.get('id');\n    const email = searchParams.get('email');\n\n    if (!userId || !email) {\n      return NextResponse.json(\n        { error: \"ID utilisateur et email requis\" },\n        { status: 400 }\n      );\n    }\n\n    const token = await getAdminToken();\n    if (!token) {\n      return NextResponse.json({ error: \"Erreur d'authentification\" }, { status: 401 });\n    }\n\n    // Delete user from Keycloak\n    const deleteResponse = await fetch(\n      `${process.env.KEYCLOAK_BASE_URL}/admin/realms/${process.env.KEYCLOAK_REALM}/users/${userId}`,\n      {\n        method: \"DELETE\",\n        headers: {\n          Authorization: `Bearer ${token}`,\n        },\n      }\n    );\n\n    if (!deleteResponse.ok) {\n      const errorData = await deleteResponse.json();\n      console.error(\"Keycloak delete error:\", errorData);\n      return NextResponse.json(\n        { error: \"Erreur lors de la suppression de l'utilisateur\", details: errorData },\n        { status: deleteResponse.status }\n      );\n    }\n\n    // Delete user from Leantime\n    const leantimeResult = await deleteLeantimeUser(email);\n\n    if (!leantimeResult.success) {\n      console.error(\"Leantime user deletion failed:\", leantimeResult.error);\n      // We don't return an error here since Keycloak user was deleted successfully\n      // We just log the error and continue\n    }\n\n    return NextResponse.json({ success: true });\n\n  } catch (error) {\n    console.error(\"Error deleting user:\", error);\n    return NextResponse.json(\n      { error: \"Erreur serveur\", details: error },\n      { status: 500 }\n    );\n  }\n}\n"],"names":["getServerSession","authOptions","NextResponse","GET","session","json","error","status","console","log","accessToken","substring","user","tokenResponse","fetch","process","env","KEYCLOAK_BASE_URL","KEYCLOAK_REALM","method","headers","body","URLSearchParams","grant_type","client_id","KEYCLOAK_CLIENT_ID","client_secret","KEYCLOAK_CLIENT_SECRET","tokenData","ok","data","access_token","getCurrentUser","usersResponse","Authorization","text","users","map","u","id","username","realm","serviceAccountClientId","filteredUsers","filter","length","usersWithRoles","Promise","all","rolesResponse","roles","rolesData","role","name","startsWith","includes","firstName","lastName","email","createdTimestamp","enabled","first_name","last_name","Date","now","getAdminToken","validateUsername","usernameRegex","test","isValid","createLeantimeUser","userData","response","LEANTIME_TOKEN","JSON","stringify","jsonrpc","params","values","toISOString","lastname","firstname","profileId","password","twoFAEnabled","clientId","clientName","jobTitle","jobLevel","department","modified","createdOn","source","notifications","settings","result","success","message","POST","req","usernameValidation","token","errorData","availableRoles","requestedRoles","validRoles","roleName","some","r","createResponse","emailVerified","credentials","type","value","temporary","errorMessage","details","userResponse","roleObjects","find","roleResponse","leantimeResult","deleteLeantimeUser","DELETE","searchParams","URL","url","userId","get","deleteResponse"],"ignoreList":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///(rsc)/./app/api/users/route.ts\n");
/***/ }),
/***/ "(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fusers%2Froute&page=%2Fapi%2Fusers%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fusers%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%2Fusers%2Froute&page=%2Fapi%2Fusers%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fusers%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_users_route_ts__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./app/api/users/route.ts */ \"(rsc)/./app/api/users/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/users/route\",\n pathname: \"/api/users\",\n filename: \"route\",\n bundlePath: \"app/api/users/route\"\n },\n resolvedPagePath: \"/home/alma/nextgen/Neah-mail/app/api/users/route.ts\",\n nextConfigOutput,\n userland: _home_alma_nextgen_Neah_mail_app_api_users_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/bmFtZT1hcHAlMkZhcGklMkZ1c2VycyUyRnJvdXRlJnBhZ2U9JTJGYXBpJTJGdXNlcnMlMkZyb3V0ZSZhcHBQYXRocz0mcGFnZVBhdGg9cHJpdmF0ZS1uZXh0LWFwcC1kaXIlMkZhcGklMkZ1c2VycyUyRnJvdXRlLnRzJmFwcERpcj0lMkZob21lJTJGYWxtYSUyRm5leHRnZW4lMkZOZWFoLW1haWwlMkZhcHAmcGFnZUV4dGVuc2lvbnM9dHN4JnBhZ2VFeHRlbnNpb25zPXRzJnBhZ2VFeHRlbnNpb25zPWpzeCZwYWdlRXh0ZW5zaW9ucz1qcyZyb290RGlyPSUyRmhvbWUlMkZhbG1hJTJGbmV4dGdlbiUyRk5lYWgtbWFpbCZpc0Rldj10cnVlJnRzY29uZmlnUGF0aD10c2NvbmZpZy5qc29uJmJhc2VQYXRoPSZhc3NldFByZWZpeD0mbmV4dENvbmZpZ091dHB1dD0mcHJlZmVycmVkUmVnaW9uPSZtaWRkbGV3YXJlQ29uZmlnPWUzMCUzRCEiLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7QUFBK0Y7QUFDdkM7QUFDcUI7QUFDRztBQUNoRjtBQUNBO0FBQ0E7QUFDQSx3QkFBd0IseUdBQW1CO0FBQzNDO0FBQ0EsY0FBYyxrRUFBUztBQUN2QjtBQUNBO0FBQ0E7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUNBO0FBQ0EsWUFBWTtBQUNaLENBQUM7QUFDRDtBQUNBO0FBQ0E7QUFDQSxRQUFRLHNEQUFzRDtBQUM5RDtBQUNBLFdBQVcsNEVBQVc7QUFDdEI7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUMwRjs7QUFFMUYiLCJzb3VyY2VzIjpbIiJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBcHBSb3V0ZVJvdXRlTW9kdWxlIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvcm91dGUtbW9kdWxlcy9hcHAtcm91dGUvbW9kdWxlLmNvbXBpbGVkXCI7XG5pbXBvcnQgeyBSb3V0ZUtpbmQgfSBmcm9tIFwibmV4dC9kaXN0L3NlcnZlci9yb3V0ZS1raW5kXCI7XG5pbXBvcnQgeyBwYXRjaEZldGNoIGFzIF9wYXRjaEZldGNoIH0gZnJvbSBcIm5leHQvZGlzdC9zZXJ2ZXIvbGliL3BhdGNoLWZldGNoXCI7XG5pbXBvcnQgKiBhcyB1c2VybGFuZCBmcm9tIFwiL2hvbWUvYWxtYS9uZXh0Z2VuL05lYWgtbWFpbC9hcHAvYXBpL3VzZXJzL3JvdXRlLnRzXCI7XG4vLyBXZSBpbmplY3QgdGhlIG5leHRDb25maWdPdXRwdXQgaGVyZSBzbyB0aGF0IHdlIGNhbiB1c2UgdGhlbSBpbiB0aGUgcm91dGVcbi8vIG1vZHVsZS5cbmNvbnN0IG5leHRDb25maWdPdXRwdXQgPSBcIlwiXG5jb25zdCByb3V0ZU1vZHVsZSA9IG5ldyBBcHBSb3V0ZVJvdXRlTW9kdWxlKHtcbiAgICBkZWZpbml0aW9uOiB7XG4gICAgICAgIGtpbmQ6IFJvdXRlS2luZC5BUFBfUk9VVEUsXG4gICAgICAgIHBhZ2U6IFwiL2FwaS91c2Vycy9yb3V0ZVwiLFxuICAgICAgICBwYXRobmFtZTogXCIvYXBpL3VzZXJzXCIsXG4gICAgICAgIGZpbGVuYW1lOiBcInJvdXRlXCIsXG4gICAgICAgIGJ1bmRsZVBhdGg6IFwiYXBwL2FwaS91c2Vycy9yb3V0ZVwiXG4gICAgfSxcbiAgICByZXNvbHZlZFBhZ2VQYXRoOiBcIi9ob21lL2FsbWEvbmV4dGdlbi9OZWFoLW1haWwvYXBwL2FwaS91c2Vycy9yb3V0ZS50c1wiLFxuICAgIG5leHRDb25maWdPdXRwdXQsXG4gICAgdXNlcmxhbmRcbn0pO1xuLy8gUHVsbCBvdXQgdGhlIGV4cG9ydHMgdGhhdCB3ZSBuZWVkIHRvIGV4cG9zZSBmcm9tIHRoZSBtb2R1bGUuIFRoaXMgc2hvdWxkXG4vLyBiZSBlbGltaW5hdGVkIHdoZW4gd2UndmUgbW92ZWQgdGhlIG90aGVyIHJvdXRlcyB0byB0aGUgbmV3IGZvcm1hdC4gVGhlc2Vcbi8vIGFyZSB1c2VkIHRvIGhvb2sgaW50byB0aGUgcm91dGUuXG5jb25zdCB7IHdvcmtBc3luY1N0b3JhZ2UsIHdvcmtVbml0QXN5bmNTdG9yYWdlLCBzZXJ2ZXJIb29rcyB9ID0gcm91dGVNb2R1bGU7XG5mdW5jdGlvbiBwYXRjaEZldGNoKCkge1xuICAgIHJldHVybiBfcGF0Y2hGZXRjaCh7XG4gICAgICAgIHdvcmtBc3luY1N0b3JhZ2UsXG4gICAgICAgIHdvcmtVbml0QXN5bmNTdG9yYWdlXG4gICAgfSk7XG59XG5leHBvcnQgeyByb3V0ZU1vZHVsZSwgd29ya0FzeW5jU3RvcmFnZSwgd29ya1VuaXRBc3luY1N0b3JhZ2UsIHNlcnZlckhvb2tzLCBwYXRjaEZldGNoLCAgfTtcblxuLy8jIHNvdXJjZU1hcHBpbmdVUkw9YXBwLXJvdXRlLmpzLm1hcCJdLCJuYW1lcyI6W10sImlnbm9yZUxpc3QiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fusers%2Froute&page=%2Fapi%2Fusers%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fusers%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");
/***/ }),
/***/ "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");
/***/ }),
/***/ "events":
/*!*************************!*\
!*** external "events" ***!
\*************************/
/***/ ((module) => {
"use strict";
module.exports = require("events");
/***/ }),
/***/ "http":
/*!***********************!*\
!*** external "http" ***!
\***********************/
/***/ ((module) => {
"use strict";
module.exports = require("http");
/***/ }),
/***/ "https":
/*!************************!*\
!*** external "https" ***!
\************************/
/***/ ((module) => {
"use strict";
module.exports = require("https");
/***/ }),
/***/ "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");
/***/ }),
/***/ "querystring":
/*!******************************!*\
!*** external "querystring" ***!
\******************************/
/***/ ((module) => {
"use strict";
module.exports = require("querystring");
/***/ }),
/***/ "url":
/*!**********************!*\
!*** external "url" ***!
\**********************/
/***/ ((module) => {
"use strict";
module.exports = require("url");
/***/ }),
/***/ "util":
/*!***********************!*\
!*** external "util" ***!
\***********************/
/***/ ((module) => {
"use strict";
module.exports = require("util");
/***/ }),
/***/ "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"], () => (__webpack_exec__("(rsc)/./node_modules/next/dist/build/webpack/loaders/next-app-loader/index.js?name=app%2Fapi%2Fusers%2Froute&page=%2Fapi%2Fusers%2Froute&appPaths=&pagePath=private-next-app-dir%2Fapi%2Fusers%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__;
})();