najm-auth 0.1.6 → 0.1.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of najm-auth might be problematic. Click here for more details.

package/dist/index.mjs CHANGED
@@ -1,12 +1,2998 @@
1
- var ns=Object.defineProperty;var os=(e,t)=>{for(var a in t)ns(e,a,{get:t[a],enumerable:!0,configurable:!0,set:(i)=>t[a]=()=>i})};var n=function(e,t,a,i){var l=arguments.length,c=l<3?t:i===null?i=Object.getOwnPropertyDescriptor(t,a):i,d;if(typeof Reflect==="object"&&typeof Reflect.decorate==="function")c=Reflect.decorate(e,t,a,i);else for(var E=e.length-1;E>=0;E--)if(d=e[E])c=(l<3?d(c):l>3?d(t,a,c):d(t,a))||c;return l>3&&c&&Object.defineProperty(t,a,c),c},u=(e,t)=>(a,i)=>t(a,i,e),r=(e,t)=>{if(typeof Reflect==="object"&&typeof Reflect.metadata==="function")return Reflect.metadata(e,t)};import{Injectable as cs}from"najm-api";import Fe from"bcrypt";class J{constructor(){}async hashPassword(e){if(!e)return null;if(typeof e!=="string"||e.trim().length===0)return null;return Fe.hash(e,10)}async comparePassword(e,t){return Fe.compare(e,t)}}J=n([cs(),r("design:paramtypes",[])],J);import{setCookie as us,Injectable as ps,deleteCookie as ls}from"najm-api";import ms from"timestring";class G{setRefreshCookie(e){let t=ms(process.env.REFRESH_EXPIRES_IN,"s");us("refreshToken",e,{httpOnly:!1,sameSite:"Lax",maxAge:t,path:"/api/auth/refresh"})}clearRefreshCookie(){ls("refreshToken",{httpOnly:!1,sameSite:"Lax",path:"/api/auth/refresh",maxAge:0})}}G=n([ps()],G);import{Controller as Ds,Get as He,Post as We,Params as Qs,Body as Xe,User as Zs,t as ne}from"najm-api";import{t as ct}from"najm-api";import{Injectable as Ws,getCurrentLanguage as Xs}from"najm-api";import{rolesTable as U,usersTable as y,permissionsTable as Ve,rolePermissionsTable as be}from"@/database/schema";import{eq as T,ne as ys}from"drizzle-orm";import{Repository as ds}from"najm-api";class M{getUser(){return{id:y.id,email:y.email,emailVerified:y.emailVerified,image:y.image,status:y.status,roleId:y.roleId,role:U.name,createdAt:y.createdAt,updatedAt:y.updatedAt}}async getAll(){let e=await this.db.select(this.getUser()).from(y).leftJoin(U,T(y.roleId,U.id));return Promise.all(e.map(async(t)=>({...t,permissions:await this.getUserPermissions(t.id)})))}async getById(e){let[t]=await this.db.select(this.getUser()).from(y).leftJoin(U,T(y.roleId,U.id)).where(T(y.id,e)).limit(1);if(!t)return t;return{...t,permissions:await this.getUserPermissions(t.id)}}async getByEmail(e){let[t]=await this.db.select(this.getUser()).from(y).leftJoin(U,T(y.roleId,U.id)).where(T(y.email,e));return t}async create(e){let[t]=await this.db.insert(y).values(e).returning();return t}async update(e,t){let[a]=await this.db.update(y).set(t).where(T(y.id,e)).returning();return a}async delete(e){let[t]=await this.db.delete(y).where(T(y.id,e)).returning();return t}async deleteAll(){let e=await this.db.select({id:U.id}).from(U).where(T(U.name,"admin")).limit(1);if(e.length===0)return await this.db.delete(y).returning();return await this.db.delete(y).where(ys(y.roleId,e[0].id)).returning()}async getRoleNameById(e){let[t]=await this.db.select({roleName:U.name}).from(y).leftJoin(U,T(y.roleId,U.id)).where(T(y.id,e));return t.roleName}async getUserPassword(e){let[t]=await this.db.select({id:y.id,email:y.email,password:y.password}).from(y).where(T(y.email,e)).limit(1);return t.password}async getUserPermissions(e){let[t]=await this.db.select({roleId:y.roleId}).from(y).where(T(y.id,e)).limit(1);if(!t||!t.roleId)return[];return(await this.db.select({name:Ve.name}).from(be).leftJoin(Ve,T(be.permissionId,Ve.id)).where(T(be.roleId,t.roleId))).map((i)=>i.name).filter((i)=>i)}}M=n([ds()],M);import{Injectable as hs,t as j}from"najm-api";import{parseSchema as gs}from"@/shared";import{userSchema as fs}from"@/lib/validations";class C{userRepository;encryptionService;constructor(e,t){this.userRepository=e;this.encryptionService=t}async validateCreateUser(e){return gs(fs,e)}async isEmailExists(e){return!!await this.userRepository.getByEmail(e)}async isPasswordValid(e,t){return!!await this.encryptionService.comparePassword(e,t)}async isUserExist(e){return!!await this.userRepository.getById(e)}async checkUserIdIsUnique(e){if(!e)return;if(await this.userRepository.getById(e))throw new Error(j("users.errors.idExists"))}async isCorrectPass(e){return e&&typeof e==="string"&&e.trim().length>0}async hasRole(e,t){let a=await this.userRepository.getRoleNameById(e);if(!a)throw Error(j("auth.errors.accessDenied"));if(!t.some((l)=>a.toLowerCase()===l.toLowerCase()))throw Error(j("auth.errors.accessDenied"));return!0}async checkUserExistsByEmail(e){let t=await this.userRepository.getByEmail(e);if(!t)throw new Error(j("auth.errors.invalidCredentials"));return t}async checkUserExists(e){let t=await this.isUserExist(e);if(!t)throw new Error(j("users.errors.notFound"));return t}async checkEmailUnique(e,t=null){if(!e)return;let a=await this.userRepository.getByEmail(e);if(a&&a.id!==t)throw new Error(j("auth.errors.emailExists"))}async checkEmailExists(e){let t=await this.userRepository.getByEmail(e);if(!t)throw new Error(j("users.errors.notFound"));return t}async checkPasswordValid(e,t){if(!await this.isPasswordValid(e,t))throw new Error(j("auth.errors.invalidCredentials"))}}C=n([hs(),r("design:paramtypes",[typeof M==="undefined"?Object:M,typeof J==="undefined"?Object:J])],C);import{Injectable as Vs,setLanguage as bs,getCurrentLanguage as _s,Transactional as $s}from"najm-api";import{rolesTable as P}from"@/database/schema";import{eq as ke}from"drizzle-orm";import{Repository as ws}from"najm-api";class H{async getAll(){return await this.db.select().from(P)}async getById(e){let[t]=await this.db.select().from(P).where(ke(P.id,e));return t}async getByName(e){let[t]=await this.db.select().from(P).where(ke(P.name,e));return t}async create(e){let[t]=await this.db.insert(P).values(e).returning();return t}async update(e,t){let[a]=await this.db.update(P).set(t).where(ke(P.id,e)).returning();return a}async delete(e){let[t]=await this.db.delete(P).where(ke(P.id,e)).returning();return t}}H=n([ws()],H);import{Injectable as xs,t as Te}from"najm-api";import{parseSchema as vs}from"@/shared";import{roleSchema as ks}from"@/lib/validations";class K{roleRepository;constructor(e){this.roleRepository=e}async validateCreateRole(e){return vs(ks,e)}async isRoleNameExists(e){return!!await this.roleRepository.getByName(e)}async isRoleIdExists(e){return!!await this.roleRepository.getById(e)}async checkNameUnique(e,t=null){if(!e)return;let a=await this.roleRepository.getByName(e);if(a&&a.id!==t)throw new Error(Te("roles.errors.exists"))}async checkRoleExists(e){if(!await this.isRoleIdExists(e))throw new Error(Te("roles.errors.notFound"))}async checkRoleExistsByName(e){if(!await this.isRoleNameExists(e))throw new Error(Te("roles.errors.notFound"))}async checkAdminRoleExists(){let e=await this.roleRepository.getByName("admin");if(!e)throw new Error(Te("users.errors.adminRoleNotFound"));return e}}K=n([xs(),r("design:paramtypes",[typeof H==="undefined"?Object:H])],K);import{Injectable as st,Headers as ze,createGuard as rt,GuardParams as Ns,Ctx as et}from"najm-api";import{tokensTable as se,usersTable as g,rolesTable as re,permissionsTable as _e,rolePermissionsTable as $e}from"@/database/schema";import{eq as I}from"drizzle-orm";import{Repository as Ts}from"najm-api";class O{async storeRefreshToken(e){return await this.db.insert(se).values(e).onConflictDoUpdate({target:se.userId,set:{token:e.token,expiresAt:e.expiresAt}}).returning()}async getRefreshToken(e){let[t]=await this.db.select().from(se).where(I(se.userId,e));return t?.token}async revokeToken(e){let[t]=await this.db.delete(se).where(I(se.userId,e)).returning();return t}async isUserExists(e){let[t]=await this.db.select({id:g.id}).from(g).where(I(g.id,e)).limit(1);return!!t}async getRoleNameById(e){let[t]=await this.db.select({roleName:re.name}).from(g).leftJoin(re,I(g.roleId,re.id)).where(I(g.id,e)).limit(1);return t?.roleName}async getUserPermissions(e){let[t]=await this.db.select({roleId:g.roleId}).from(g).where(I(g.id,e)).limit(1);if(!t||!t.roleId)return[];return(await this.db.select({name:_e.name}).from($e).leftJoin(_e,I($e.permissionId,_e.id)).where(I($e.roleId,t.roleId))).map((i)=>i.name).filter((i)=>i)}async getUser(e){let[t]=await this.db.select({id:g.id,email:g.email,status:g.status,roleId:g.roleId,roleName:re.name,createdAt:g.createdAt,updatedAt:g.updatedAt}).from(g).leftJoin(re,I(g.roleId,re.id)).where(I(g.id,e)).limit(1);return t?{...t,role:t.roleName}:null}}O=n([Ts()],O);import{t as ae}from"najm-api";import{getCookie as Ss,Injectable as Es}from"najm-api";import Se from"jsonwebtoken";import{jwtDecode as Us}from"jwt-decode";import Rs from"timestring";class A{tokenRepository;accessSecretKey=process.env.JWT_ACCESS_SECRET;accessExpiresIn=process.env.ACCESS_EXPIRES_IN;refreshSecretKey=process.env.JWT_REFRESH_SECRET;refreshExpiresIn=process.env.REFRESH_EXPIRES_IN;constructor(e){this.tokenRepository=e}extractAccessToken(e){if(e&&e.startsWith("Bearer"))return e.split(" ")[1];throw new Error(ae("auth.errors.tokenMissing"))}verifyAccessToken(e){try{return Se.verify(e,this.accessSecretKey)}catch(t){throw new Error(ae("auth.errors.tokenVerificationFailed"))}}verifyRefreshToken(e){try{return Se.verify(e,this.refreshSecretKey).userId}catch(t){throw new Error(ae("auth.errors.tokenVerificationFailed"))}}async getUserIdByAccessToken(e){let t=this.extractAccessToken(e),i=this.verifyAccessToken(t).userId;if(!await this.tokenRepository.isUserExists(i))throw new Error(ae("users.errors.notFound"));return i}async storeRefreshToken(e,t){let a=Rs(this.refreshExpiresIn,"s"),i={userId:e,token:t,expiresAt:new Date(Date.now()+a*1000).toISOString()};await this.tokenRepository.storeRefreshToken(i)}getTokenExpire(e){return Us(e).exp}generateAccessToken(e){let t={expiresIn:this.accessExpiresIn};return Se.sign(e,this.accessSecretKey,t)}generateRefreshToken(e){let t={expiresIn:this.refreshExpiresIn};return Se.sign(e,this.refreshSecretKey,t)}async generateTokens(e){let t={userId:e},a=await this.generateAccessToken(t),i=await this.generateRefreshToken(t),l=this.getTokenExpire(a),c=this.getTokenExpire(i);return await this.storeRefreshToken(e,i),{accessToken:a,refreshToken:i,accessTokenExpiresAt:l,refreshTokenExpiresAt:c}}async refreshTokens(){let e=Ss("refreshToken"),t=this.verifyRefreshToken(e);if(!await this.tokenRepository.isUserExists(t))throw new Error(ae("users.errors.notFound"));let i=await this.tokenRepository.getRefreshToken(t);if(e!=i)throw new Error(ae("auth.errors.refreshTokenInvalid"));return await this.generateTokens(t)}async revokeToken(e){return await this.tokenRepository.revokeToken(e)}async getUserPermissions(e){if(!e)return;let t=await this.getUserIdByAccessToken(e);return await this.tokenRepository.getUserPermissions(t)}async getUserRole(e){if(!e)return;let t=await this.getUserIdByAccessToken(e);return await this.tokenRepository.getRoleNameById(t)}async getUser(e){if(!e)return;let t=await this.getUserIdByAccessToken(e),a=await this.tokenRepository.getUser(t);if(!a)return null;return a}async storeUserInCache(e,t){let a=t.get("user");if(a)return a;let i=await this.getUser(e);if(i)t.set("user",i);return i}}A=n([Es(),r("design:paramtypes",[typeof O==="undefined"?Object:O])],A);var f={ADMIN:"admin",PRINCIPAL:"principal",ACCOUNTING:"accounting",SECRETARY:"secretary",TEACHER:"teacher",STUDENT:"student",PARENT:"parent"},tt={ADMINISTRATORS:[f.ADMIN,f.PRINCIPAL],FINANCIAL:[f.ADMIN,f.ACCOUNTING],STAFF:[f.ADMIN,f.PRINCIPAL,f.ACCOUNTING,f.SECRETARY,f.TEACHER],END_USERS:[f.STUDENT,f.PARENT],ALL:[f.ADMIN,f.PRINCIPAL,f.ACCOUNTING,f.SECRETARY,f.TEACHER,f.STUDENT,f.PARENT]};class Ee{isInGroup(e,t){return t.includes(e?.toLowerCase())}isAdministrator(e){return this.isInGroup(e,tt.ADMINISTRATORS)}isStaff(e){return this.isInGroup(e,tt.STAFF)}hasAnyRole(e,t){return t.includes(e?.toLowerCase())}hasExactRole(e,t){return e?.toLowerCase()===t?.toLowerCase()}}Ee=n([st()],Ee);class q{roleChecker;tokenService;constructor(e,t){this.roleChecker=e;this.tokenService=t}async isAuth(e,t){return!!await this.tokenService.storeUserInCache(e,t)}async hasRoles(e,t,a){try{let i=await this.tokenService.storeUserInCache(e,t);if(!i?.role)return!1;let l=Array.isArray(a)?a:[a];return this.roleChecker.hasAnyRole(i.role,l)}catch{return!1}}}n([u(0,ze("authorization")),u(1,et()),r("design:type",Function),r("design:paramtypes",[Object,Object]),r("design:returntype",Object)],q.prototype,"isAuth",null),n([u(0,ze("authorization")),u(1,et()),u(2,Ns()),r("design:type",Function),r("design:paramtypes",[Object,Object,Object]),r("design:returntype",Object)],q.prototype,"hasRoles",null),q=n([st(),r("design:paramtypes",[typeof Ee==="undefined"?Object:Ee,typeof A==="undefined"?Object:A])],q);var ie=()=>V("admin"),Ra=()=>V("principal"),Na=()=>V("accounting"),Ka=()=>V("secretary"),Aa=()=>V("teacher"),Ba=()=>V("parent"),Pa=()=>V("student"),Ia=()=>V("admin","principal"),Va=()=>V("admin","accounting"),ba=()=>V("admin","principal","accounting","secretary","teacher"),_a=rt(q,"isAuth"),V=(...e)=>rt(q,"hasRoles")(...e);import{Controller as As,Get as at,Post as Bs,Put as Ps,Delete as Is,Params as Je,Body as it,t as pe}from"najm-api";import{Injectable as Ks}from"najm-api";class B{roleRepository;roleValidator;constructor(e,t){this.roleRepository=e;this.roleValidator=t}async getAll(){return await this.roleRepository.getAll()}async getById(e){return await this.roleValidator.checkRoleExists(e),await this.roleRepository.getById(e)}async getByName(e){return await this.roleRepository.getByName(e)}async create(e){return await this.roleValidator.validateCreateRole(e),await this.roleValidator.checkNameUnique(e.name),await this.roleRepository.create(e)}async update(e,t){return await this.roleValidator.checkRoleExists(e),await this.roleValidator.checkNameUnique(t.name,e),await this.roleRepository.update(e,t)}async delete(e){return await this.roleValidator.checkRoleExists(e),await this.roleRepository.delete(e)}async seedDefaultRoles(e){let t=[];for(let i of e)if(!await this.roleValidator.isRoleNameExists(i.name))t.push(i);return await Promise.all(t.map((i)=>this.roleRepository.create(i)))}async getRoleIdByName(e){return(await this.getByName(e))?.id}}B=n([Ks(),r("design:paramtypes",[typeof H==="undefined"?Object:H,typeof K==="undefined"?Object:K])],B);class le{roleService;constructor(e){this.roleService=e}async getRoles(){return{data:await this.roleService.getAll(),message:pe("roles.success.retrieved"),status:"success"}}async getRole(e){return{data:await this.roleService.getById(e),message:pe("roles.success.retrieved"),status:"success"}}async createRole(e){return{data:await this.roleService.create(e),message:pe("roles.success.created"),status:"success"}}async updateRole(e,t){return{data:await this.roleService.update(e,t),message:pe("roles.success.updated"),status:"success"}}async deleteRole(e){return{data:await this.roleService.delete(e),message:pe("roles.success.deleted"),status:"success"}}}n([at(),ie(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],le.prototype,"getRoles",null),n([at("/:id"),ie(),u(0,Je("id")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],le.prototype,"getRole",null),n([Bs(),ie(),u(0,it()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],le.prototype,"createRole",null),n([Ps("/:id"),ie(),u(0,Je("id")),u(1,it()),r("design:type",Function),r("design:paramtypes",[Object,Object]),r("design:returntype",Promise)],le.prototype,"updateRole",null),n([Is("/:id"),ie(),u(0,Je("id")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],le.prototype,"deleteRole",null),le=n([As("/roles"),r("design:paramtypes",[typeof B==="undefined"?Object:B])],le);import{nanoid as Js}from"nanoid";import{clean as Ms}from"@/shared";class W{roleValidator;roleService;userRepository;userValidator;encryptionService;constructor(e,t,a,i,l){this.roleValidator=e;this.roleService=t;this.userRepository=a;this.userValidator=i;this.encryptionService=l}sanitizeUser(e){if(!e)return e;let{password:t,...a}=e;return a}sanitizeUsers(e){return e.map((t)=>this.sanitizeUser(t))}async resolveUserRole(e,t){if(e)return await this.roleValidator.checkRoleExists(e),e;if(t){let i=await this.roleService.getByName(t);if(i)return i.id;throw new Error(`Role '${t}' not found`)}return(await this.roleService.getByName("Student")).id}async getAll(){let e=await this.userRepository.getAll();return this.sanitizeUsers(e)}async getById(e){await this.userValidator.checkUserExists(e);let t=await this.userRepository.getById(e);return this.sanitizeUser(t)}async getByEmail(e){let t=await this.userValidator.checkUserExistsByEmail(e);return this.sanitizeUser(t)}async create(e){let{id:t,email:a,image:i,emailVerified:l,password:c,roleId:d,role:E}=e,ue=t||Js(5),ss=c||"12345678";await this.userValidator.checkEmailUnique(e.email),await this.userValidator.checkUserIdIsUnique(t);let rs=await this.encryptionService.hashPassword(ss),as=await this.resolveUserRole(d,E),qe={id:ue,email:a,image:i,password:rs,roleId:as,emailVerified:l,status:"pending"};await this.userValidator.validateCreateUser(qe);let is=await this.userRepository.create(qe);return this.sanitizeUser(is)}async update(e,t){let{password:a,image:i}=t;await this.userValidator.checkUserExists(e),await this.userValidator.checkEmailUnique(t.email,e);let l=await this.userRepository.getById(e),c=await this.encryptionService.hashPassword(a),d={...t,image:i,...c&&{password:c}},E=Ms(d),ue=await this.userRepository.update(e,E);return this.sanitizeUser(ue)}async delete(e){await this.userValidator.checkUserExists(e);let t=await this.userRepository.delete(e);return this.sanitizeUser(t)}async deleteAll(){let e=await this.userRepository.deleteAll();return this.sanitizeUsers(e)}async getRoleName(e){return await this.userValidator.checkUserExists(e),await this.userRepository.getRoleNameById(e)}async getPassword(e){return await this.userValidator.checkUserExistsByEmail(e),await this.userRepository.getUserPassword(e)}async assignRole(e,t,a){await this.userValidator.checkUserExists(e);let i=await this.resolveUserRole(t,a),l=await this.userRepository.update(e,{roleId:i});return this.sanitizeUser(l)}async removeRole(e){await this.userValidator.checkUserExists(e);let t=await this.userRepository.update(e,{roleId:null});return this.sanitizeUser(t)}async seedAdminUser(){let t=await this.roleValidator.checkAdminRoleExists(),a=await this.userRepository.getByEmail("admin@admin.com");if(a)await this.delete(a.id);let i=await this.create({id:"USR00",name:"System Administrator",email:"admin@admin.com",password:"12345678",image:null,roleId:t.id,status:"active",emailVerified:!0});return this.sanitizeUser(i)}async updateLang(e){return bs(e),e}async getLang(){return _s()}}n([$s(),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],W.prototype,"create",null),W=n([Vs(),r("design:paramtypes",[typeof K==="undefined"?Object:K,typeof B==="undefined"?Object:B,typeof M==="undefined"?Object:M,typeof C==="undefined"?Object:C,typeof J==="undefined"?Object:J])],W);import{Controller as Cs,Get as me,Post as Me,Put as Hs,Delete as Ce,Params as X,Body as nt,t as R}from"najm-api";import{isAdmin as b,isAuth as ot}from"@/roles/RoleGuards";class ye{userService;constructor(e){this.userService=e}async getUsers(){return{data:await this.userService.getAll(),message:R("users.success.retrieved"),status:"success"}}async getLang(){return{data:{language:await this.userService.getLang()},message:R("users.success.retrieved"),status:"success"}}async updateLang(e){return{data:await this.userService.updateLang(e),message:R("users.success.updated"),status:"success"}}async getUser(e){return{data:await this.userService.getById(e),message:R("users.success.retrieved"),status:"success"}}async getByEmail(e){return{data:await this.userService.getByEmail(e),message:R("users.success.retrieved"),status:"success"}}async getRole(e){return{data:await this.userService.getRoleName(e),message:R("users.success.retrieved"),status:"success"}}async create(e){return{data:await this.userService.create(e),message:R("users.success.created"),status:"success"}}async update(e,t){return{data:await this.userService.update(e,t),message:R("users.success.updated"),status:"success"}}async delete(e){return{data:await this.userService.delete(e),message:R("users.success.deleted"),status:"success"}}async deleteAll(){return{data:await this.userService.deleteAll(),message:R("users.success.allDeleted"),status:"success"}}async assignRole(e,t){return await this.userService.assignRole(e,t),{message:R("users.success.updated"),status:"success"}}async removeRole(e){return await this.userService.removeRole(e),{message:R("users.success.updated"),status:"success"}}}n([me(),b(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],ye.prototype,"getUsers",null),n([me("/lang"),ot(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],ye.prototype,"getLang",null),n([Me("/lang/:language"),ot(),u(0,X("language")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"updateLang",null),n([me("/:id"),b(),u(0,X("id")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"getUser",null),n([me("/email/:email"),b(),u(0,X("email")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"getByEmail",null),n([me("/role/:userId"),b(),u(0,X("userId")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"getRole",null),n([Me(),b(),u(0,nt()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"create",null),n([Hs("/:id"),b(),u(0,X("id")),u(1,nt()),r("design:type",Function),r("design:paramtypes",[Object,Object]),r("design:returntype",Promise)],ye.prototype,"update",null),n([Ce("/:id"),b(),u(0,X("id")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"delete",null),n([Ce(),b(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],ye.prototype,"deleteAll",null),n([Me("/assign/:userId/:roleId"),b(),u(0,X("userId")),u(1,X("roleId")),r("design:type",Function),r("design:paramtypes",[Object,Object]),r("design:returntype",Promise)],ye.prototype,"assignRole",null),n([Ce("/remove/:userId"),b(),u(0,X("userId")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ye.prototype,"removeRole",null),ye=n([Cs("/users"),r("design:paramtypes",[typeof W==="undefined"?Object:W])],ye);class F{tokenService;userService;userValidator;cookieService;constructor(e,t,a,i){this.tokenService=e;this.userService=t;this.userValidator=a;this.cookieService=i}async registerUser(e){return await this.userService.create(e)}async loginUser(e){let{email:t,password:a}=e;if(!t||!a)throw new Error(ct("auth.errors.invalidCredentials"));let i=await this.userService.getPassword(t),{id:l}=await this.userService.getByEmail(t);await this.userValidator.checkPasswordValid(a,i);let c=await this.tokenService.generateTokens(l);return this.cookieService.setRefreshCookie(c.refreshToken),c}async refreshTokens(){let e=await this.tokenService.refreshTokens();return this.cookieService.setRefreshCookie(e.refreshToken),e}async logoutUser(e){return await this.userValidator.checkUserExists(e),await this.tokenService.revokeToken(e),this.cookieService.clearRefreshCookie(),{data:null,message:ct("auth.success.logout")}}async getUserProfile(e){let t=Xs();return{...e,language:t}}async forgotPassword(e){}}F=n([Ws(),r("design:paramtypes",[typeof A==="undefined"?Object:A,typeof W==="undefined"?Object:W,typeof C==="undefined"?Object:C,typeof G==="undefined"?Object:G])],F);import{isAuth as ut}from"@/roles/RoleGuards";class de{authService;constructor(e){this.authService=e}async registerUser(e){return{data:await this.authService.registerUser(e),message:ne("auth.success.register"),status:"success"}}async loginUser(e){return{data:await this.authService.loginUser(e),message:ne("auth.success.login"),status:"success"}}async refreshTokens(){return{data:await this.authService.refreshTokens(),message:ne("auth.success.tokenRefreshed"),status:"success"}}async logoutUser(e){return{data:await this.authService.logoutUser(e),message:ne("auth.success.logout"),status:"success"}}async userProfile(e){return{data:await this.authService.getUserProfile(e),message:ne("users.success.retrieved"),status:"success"}}async forgotPassword(e){return{data:await this.authService.forgotPassword(e.email),message:ne("auth.success.passwordReset"),status:"success"}}}n([We("/register"),u(0,Xe()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],de.prototype,"registerUser",null),n([We("/login"),u(0,Xe()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],de.prototype,"loginUser",null),n([He("/refresh"),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],de.prototype,"refreshTokens",null),n([He("/logout/:id"),u(0,Qs("id")),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],de.prototype,"logoutUser",null),n([He("/me"),ut(),u(0,Zs()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],de.prototype,"userProfile",null),n([We("/forgot-password"),ut(),u(0,Xe()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],de.prototype,"forgotPassword",null),de=n([Ds("/auth"),r("design:paramtypes",[typeof F==="undefined"?Object:F])],de);var Ue={userType:{values:["admin","teacher","student","parent"],translationKey:"enums.userType"},userStatus:{values:["active","inactive","pending"],translationKey:"enums.userStatus"},tokenStatus:{values:["active","revoked","expired"],translationKey:"enums.tokenStatus"},tokenType:{values:["access","refresh"],translationKey:"enums.tokenType"},fileStatus:{values:["active","deleted","archived"],translationKey:"enums.fileStatus"},gender:{values:["M","F"],translationKey:"common.gender"},studentStatus:{values:["active","inactive","graduated","transferred"],translationKey:"students.status"},teacherStatus:{values:["active","inactive","onLeave"],translationKey:"teachers.status"},employmentType:{values:["fullTime","partTime","contract","temporary"],translationKey:"teachers.employmentType"},relationshipType:{values:["father","mother","guardian","stepparent","grandparent","other"],translationKey:"parents.relationships"},semester:{values:["spring","summer","fall","winter"],translationKey:"academic.semester"},classStatus:{values:["active","completed","cancelled"],translationKey:"classes.status"},sectionStatus:{values:["active","inactive","archived"],translationKey:"sections.status"},language:{values:["en","fr","ar","es"],translationKey:"common.languages"},enrollmentStatus:{values:["enrolled","completed","dropped","failed"],translationKey:"enrollments.status"},assignmentStatus:{values:["active","completed","cancelled"],translationKey:"assignments.status"},calendarSystem:{values:["SEMESTER","TRIMESTER","QUARTER"],translationKey:"settings.calendarSystem"},assessmentType:{values:["quiz","assignment","project","participation","test","presentation"],translationKey:"assessments.type"},assessmentStatus:{values:["scheduled","active","completed","cancelled"],translationKey:"assessments.status"},submissionType:{values:["online","paper","presentation","practical","discussion"],translationKey:"assessments.submissionType"},examType:{values:["midterm","final","standardized"],translationKey:"exams.type"},examSecurity:{values:["low","medium","high"],translationKey:"exams.security"},examStatus:{values:["scheduled","active","completed","cancelled","rescheduled"],translationKey:"exams.status"},gradeStatus:{values:["graded","pending","draft","reviewed"],translationKey:"grades.status"},attendanceStatus:{values:["present","absent","late","excused"],translationKey:"attendance.status"},proficiencyLevel:{values:["beginner","intermediate","advanced","expert"],translationKey:"common.proficiencyLevel"},dayOfWeek:{values:["monday","tuesday","wednesday","thursday","friday","saturday","sunday"],translationKey:"common.days"},alertType:{values:["academic","attendance","behavioral","health","system","announcement","reminder","emergency"],translationKey:"alerts.type"},alertPriority:{values:["low","medium","high","critical"],translationKey:"alerts.priority"},alertStatus:{values:["active","acknowledged","resolved","dismissed"],translationKey:"alerts.status"},feeTypeStatus:{values:["active","inactive","archived"],translationKey:"fees.typeStatus"},feeCategory:{values:["tuition","registration","transport","cafeteria","books","sports","uniform","technology","fieldtrip","other"],translationKey:"feeTypes.category"},paymentType:{values:["recurring","oneTime"],translationKey:"payments.type"},schedule:{values:["monthly","quarterly","semester","annually","oneTime"],translationKey:"fees.schedule"},feeStatus:{values:["pending","partiallyPaid","paid","overdue"],translationKey:"fees.status"},feeInstallmentStatus:{values:["pending","partiallyPaid","paid","overdue"],translationKey:"fees.installmentStatus"},paymentMethod:{values:["cash","bankTransfer","check","creditCard","debitCard","online","mobilePayment"],translationKey:"payments.methods"},paymentStatus:{values:["completed","pending","failed","refunded"],translationKey:"payments.status"},eventType:{values:["academic","sports","cultural","holiday","exam","meeting","workshop","fieldtrip","ceremony","conference","other"],translationKey:"events.type"},eventStatus:{values:["scheduled","ongoing","completed","cancelled","postponed"],translationKey:"events.status"},eventVisibility:{values:["public","private","teachers","students","parents","staff"],translationKey:"events.visibility"},participantType:{values:["student","teacher","parent","staff"],translationKey:"events.participantType"},expenseCategory:{values:["salary","utilities","maintenance","supplies","equipment","transport","food","security","cleaning","insurance","rent","tax","marketing","training","technology","miscellaneous"],translationKey:"expenses.categories"},expenseStatus:{values:["pending","approved","paid","rejected","cancelled"],translationKey:"expenses.status"},trackerMode:{values:["tracking","gprs","sms","sleepTime","sleepShock","sleepDeep"],translationKey:"tracker.mode"},driverStatus:{values:["active","inactive","onLeave","suspended"],translationKey:"transport.driverStatus"},vehicleStatus:{values:["active","inactive","maintenance","retired"],translationKey:"transport.vehicleStatus"},vehicleType:{values:["sedan","minibus","fullbus","shuttle"],translationKey:"transport.vehicleType"},vehicleDocumentType:{values:["insurance","registration","inspection","emission","license"],translationKey:"transport.documentType"},busStatus:{values:["active","inactive","maintenance","retired"],translationKey:"transport.busStatus"},refuelStatus:{values:["pending","completed","cancelled"],translationKey:"transport.refuelStatus"},fuelType:{values:["gasoline","diesel","electric","hybrid","lpg","cng"],translationKey:"transport.fuelType"},maintenanceType:{values:["scheduled","repair","inspection","oilChange","filterChange","other"],translationKey:"transport.maintenanceType"},maintenanceStatus:{values:["scheduled","inProgress","completed","cancelled","overdue"],translationKey:"transport.maintenanceStatus"},maritalStatus:{values:["single","married","divorced","widowed","separated"],translationKey:"parents.maritalStatus"}},Ji=(e)=>Ue[e],Mi=(e)=>Ue[e]?.values||[];import{z as s}from"zod";import{z as js}from"zod";var o=(e)=>{let t=Ue[e]?.values;if(!t)throw new Error(`Enum ${e} not found`);return js.enum(t)},Xi=o("userType"),pt=o("userStatus"),Di=o("tokenStatus"),Qi=o("tokenType"),Zi=o("fileStatus"),he=o("gender"),lt=o("studentStatus"),mt=o("teacherStatus"),yt=o("employmentType"),dt=o("relationshipType"),ji=o("semester"),Li=o("classStatus"),ht=o("sectionStatus"),gt=o("language"),Yi=o("enrollmentStatus"),Gi=o("assignmentStatus"),ft=o("calendarSystem"),wt=o("assessmentType"),xt=o("assessmentStatus"),Oi=o("submissionType"),vt=o("examType"),qi=o("examSecurity"),kt=o("examStatus"),Tt=o("gradeStatus"),De=o("attendanceStatus"),Fi=o("proficiencyLevel"),zi=o("dayOfWeek"),St=o("alertType"),Et=o("alertPriority"),Ut=o("alertStatus"),Rt=o("feeTypeStatus"),Nt=o("paymentType"),Kt=o("schedule"),At=o("feeStatus"),Bt=o("feeInstallmentStatus"),ge=o("paymentMethod"),Pt=o("paymentStatus"),It=o("eventType"),Vt=o("eventStatus"),bt=o("eventVisibility"),_t=o("participantType"),$t=o("expenseCategory"),Jt=o("expenseStatus"),en=o("trackerMode"),Mt=o("driverStatus"),Ct=o("vehicleStatus"),Ht=o("vehicleType"),tn=o("vehicleDocumentType"),sn=o("busStatus"),Wt=o("refuelStatus"),Xt=o("fuelType"),rn=o("maintenanceType"),an=o("maintenanceStatus"),nn=o("maritalStatus");var h=s.preprocess((e)=>e??"",s.string().min(1,"ID is required")),m=s.string().min(1,"ID cannot be empty").nullish().optional(),oe=s.string().email("Invalid email format").or(s.literal("")),z=s.string().regex(/^[\+]?[1-9][\d]{0,15}$/,"Invalid phone number"),D=s.string().min(2,"Name must be at least 2 characters").max(100,"Name too long"),x=s.string().regex(/^(\d{4}-\d{2}-\d{2}|\d{2}\/\d{2}\/\d{4}|\d{2}-\d{2}-\d{2}|\d{2}-\d{2}-\d{4})$/,"Date must be in YYYY-MM-DD, MM/DD/YYYY, DD/MM/YYYY, DD-MM-YY, or DD-MM-YYYY format"),Q=s.string().regex(/^\d{4}-\d{2}-\d{2}$/,"Date must be in YYYY-MM-DD format").nullable().optional(),Re=s.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/,"Time must be in HH:MM format").optional().nullable(),Qe=s.string().min(8,"CIN must be at least 8 characters").max(20,"CIN too long"),Ne=s.string().max(500,"Address too long").optional(),Ze=s.string().min(9,"Academic year is required").regex(/^\d{4}-\d{4}$/,"Academic year must be in YYYY-YYYY format"),p=()=>{let e=(i)=>{return Object.assign(i,{positive:(c="Must be positive")=>e(i.refine((d)=>d>0,{message:c})),min:(c,d)=>e(i.refine((E)=>E>=c,{message:d||`Must be at least ${c}`})),max:(c,d)=>e(i.refine((E)=>E<=c,{message:d||`Cannot exceed ${c}`})),int:(c="Must be an integer")=>e(i.refine((d)=>Number.isInteger(d),{message:c}))})},t=(i)=>{if(i===null||i===void 0||Number.isNaN(i))return!1;if(typeof i==="number")return!0;if(typeof i==="string"){let l=i.trim();return l!==""&&!isNaN(Number(l))}return!1},a=s.any().refine(t,{message:"Must be a valid number"}).transform((i)=>typeof i==="string"?Number(i):i);return e(a)},pn=s.object({id:m,username:D.max(50).optional(),email:oe,password:s.string().min(8,"Password must be at least 8 characters"),roleId:m,roleName:D.max(50).optional(),lastLogin:Q,image:s.union([s.string(),s.instanceof(File),s.undefined()]).optional(),emailVerified:s.boolean().default(!1),status:pt,createdAt:Q}),ln=s.object({id:m,name:D.max(50),description:s.string().max(255,"Description too long").optional(),createdAt:Q}),Ls=s.object({id:m,classId:h,sectionId:h,studentCode:s.string(),name:D,email:oe,phone:z.nullish(),address:Ne,dateOfBirth:Q,gender:he,enrollmentDate:x,medicalConditions:s.string().max(1000,"Medical conditions description too long").nullish().optional(),previousSchool:s.string().max(500,"Previous school name too long").optional().nullable(),image:s.union([s.string(),s.instanceof(File),s.null()]).optional(),status:lt.default("active")}),Ys=s.object({id:m,name:D,email:oe.optional(),phone:z,gender:he.optional(),address:Ne,dateOfBirth:Q,cin:Qe,occupation:s.string().max(100,"Occupation too long").optional(),nationality:s.string().max(100,"Nationality too long").optional(),maritalStatus:s.string().max(50,"Marital status too long").optional(),relationshipType:dt,image:s.union([s.string(),s.instanceof(File),s.null()]).optional(),isEmergencyContact:s.boolean().optional().default(!1),financialResponsibility:s.boolean().optional().default(!1)}),mn=s.object({id:m,name:D,email:oe,cin:Qe,phone:z,address:Ne,gender:he.optional(),licenseNumber:s.string().min(5,"License number must be at least 5 characters").max(20,"License number too long"),licenseType:s.string().max(10,"License type too long"),licenseExpiry:x,hireDate:x,salary:p().positive("Salary must be positive").optional(),yearsOfExperience:p().int().min(0,"Years of experience must be non-negative").optional(),emergencyContact:D.optional(),emergencyPhone:z.optional(),image:s.union([s.string(),s.instanceof(File),s.null()]).optional(),status:Mt.default("active"),notes:s.string().max(1000,"Notes too long").optional().nullable()}),Gs=s.object({id:m,name:D,cin:Qe,email:oe,phone:z,address:Ne,gender:he.optional(),emergencyContact:D.optional(),emergencyPhone:z,status:mt.default("active"),image:s.union([s.string(),s.instanceof(File),s.null()]).optional()}),Os=s.object({specialization:s.string().max(100,"Specialization too long").optional(),yearsOfExperience:p().int().min(0,"Years of experience must be non-negative").optional(),salary:p().positive("Salary must be positive").optional(),hireDate:x,bankAccount:s.coerce.string().max(100,{message:"Bank account too long"}).optional(),employmentType:yt.optional(),workloadHours:p().int().min(0,"Workload hours must be non-negative").max(60,"Workload hours cannot exceed 60").optional(),academicDegrees:s.string().max(500,"Academic degrees description too long").optional()}),qs=s.object({classId:s.string().min(1,"Class is required"),sectionIds:s.array(s.string()).min(1,"At least one section is required"),subjectIds:s.array(s.string()).min(1,"At least one subject is required"),academicYear:s.string().optional()}),Fs=s.object({assignments:s.array(qs).min(1,"At least one parent is required")}),yn=s.object({...Gs.shape,...Os.shape,...Fs.shape}),dn=s.object({id:m,name:s.string().min(2,"Fee type name must be at least 2 characters").max(100,"Fee type name too long"),description:s.string().max(500,"Description too long").optional().nullable(),category:s.string(),amount:p().positive("Amount must be greater than 0").max(1e5,"Amount too large"),paymentType:Nt.default("recurring"),status:Rt.default("active").optional()}),zs=s.object({id:m,studentId:h,feeTypeId:h,academicYear:Ze.optional(),status:At.optional(),schedule:Kt,baseAmount:p().positive("Base amount must be positive").optional(),grossAmount:p().positive("Gross amount must be positive").optional(),netAmount:p().optional(),paidAmount:p().min(0,"Paid amount cannot be negative").optional(),discountAmount:p().min(0,"Discount cannot be negative").optional(),discountReason:s.string().max(500,"Discount reason too long").optional().nullable(),assignedBy:m.nullable(),notes:s.string().max(1000,"Notes too long").optional().nullable()}),Dt=zs.omit({studentId:!0}),hn=s.object({studentId:h,fees:s.array(Dt).min(1,"At least one fee is required")}),gn=s.object({feeId:m,number:p().int().min(1,"Installment must be at least 1"),dueDate:x,amount:p().positive("Amount must be greater than 0").max(1e5,"Amount too large"),paidAmount:p().min(0,"Paid amount cannot be negative").max(1e5,"Amount too large").optional(),status:Bt.optional()}),fn=s.object({studentId:m,amount:p().positive("Amount must be greater than 0").max(1e5,"Amount too large").optional(),paymentMethod:ge,paymentDate:x,checkNumber:s.preprocess((e)=>e===""?null:e,s.string().max(50,"Check number too long").optional().nullable()),checkDueDate:s.preprocess((e)=>e===""?null:e,Q.nullable()),transactionRef:s.preprocess((e)=>e===""?null:e,s.string().max(100,"Transaction reference too long").optional().nullable()),receiptNumber:s.preprocess((e)=>e===""?null:e,s.string().max(50,"Receipt number too long").optional().nullable()),status:Pt.default("completed"),processedBy:m,notes:s.preprocess((e)=>e===""?null:e,s.string().max(1000,"Notes too long").optional().nullable()),allocations:s.array(s.object({feeId:m,number:s.number().int().positive("Installment must be a positive number"),amount:p().positive("Amount must be greater than 0")})).optional().nullable()}),wn=s.object({paymentId:s.string().min(1,"Payment ID is required"),feeId:s.string().min(1,"Fee ID is required"),installmentId:s.string().optional(),amount:s.number().positive("Amount must be positive"),type:s.enum(["fee","installment"]).default("installment"),notes:s.string().optional()}),er=s.object({parents:s.array(Ys).optional().default([])}),tr=s.object({fees:s.array(Dt).min(1,"At least one fee is required")}),xn=s.object({...Ls.shape,...er.shape,...tr.shape}),vn=s.object({id:m,code:s.string().min(2,"Subject code must be at least 2 characters").max(10,"Subject code too long"),name:s.string().min(2,"Subject name must be at least 2 characters").max(100,"Subject name too long"),description:s.string().max(500,"Description too long").optional(),gradeLevel:p().int().min(1).max(12).optional()}),kn=s.object({id:m,classId:h,name:s.string().min(1,"Section name is required").max(10,"Section name too long"),maxStudents:p().int().min(1,"Max students must be at least 1").max(100,"Max students cannot exceed 100").default(30),roomNumber:p().max(1e4,"Room number too long").optional(),status:ht.default("active")}),Tn=s.object({id:m,name:s.string().min(1,"Class name is required").max(50,"Class name too long"),description:s.string().max(500,"Description too long").optional(),academicYear:Ze,level:s.string().min(1,"Class level is required")}),Sn=s.object({studentId:h,teacherId:h,subjectId:h,sectionId:h,date:x,status:De.default("present"),notes:s.string().max(500,"Notes too long").optional()}),sr=s.object({classId:h,sectionId:h,subjectId:h,teacherId:h,teacherAssignmentId:m,title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),description:s.string().max(1000,"Description too long").optional().nullable(),type:wt.default("quiz"),date:x,duration:p().int().min(1,"Duration must be at least 1 minute").max(480,"Duration cannot exceed 8 hours"),totalMarks:p().positive("Total marks must be greater than 0").max(1000,"Total marks cannot exceed 1000"),passingMarks:p().min(0,"Passing marks must be non-negative").max(1000,"Passing marks cannot exceed 1000"),instructions:s.string().max(2000,"Instructions too long").optional().nullable(),status:xt.default("scheduled"),assessmentId:m}),En=s.object({assessments:s.array(sr).min(1,"At least one assessment is required").max(50,"Cannot create more than 50 assessments at once")}),Un=s.object({assessmentId:h,teacherId:h,subjectId:h,sectionId:h,studentId:h,gradeId:h,assessmentTitle:s.string().min(3,"Assessment title must be at least 3 characters").max(200,"Assessment title too long").optional(),marksObtained:p().min(0,"Marks obtained must be non-negative").max(1000,"Marks obtained cannot exceed 1000"),feedback:s.string().max(1000,"Feedback too long").optional().nullable(),status:Tt.default("graded")}),Rn=s.object({classId:m,sectionId:h,subjectId:h,teacherId:h,examId:h,teacherAssignmentId:h,title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),description:s.string().max(1000,"Description too long").optional().nullable(),type:vt.default("midterm"),date:x,startTime:Re,endTime:Re,duration:p().int().min(30,"Exam duration must be at least 30 minutes").max(480,"Duration cannot exceed 8 hours"),totalMarks:p().positive("Total marks must be greater than 0").max(1000,"Total marks cannot exceed 1000"),passingMarks:p().min(0,"Passing marks must be non-negative").max(1000,"Passing marks cannot exceed 1000"),roomNumber:p().max(50,"Room number too long").optional().nullable(),allowedMaterials:s.string().max(500,"Allowed materials description too long").optional().nullable(),instructions:s.string().max(2000,"Instructions too long").optional().nullable(),status:kt.default("scheduled")}),Nn=s.object({title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),content:s.string().min(10,"Content must be at least 10 characters").max(5000,"Content too long"),authorId:m,targetAudience:s.enum(["all","students","teachers","parents","class"]),classId:m,isPublished:s.boolean().default(!1),publishDate:s.string().datetime("Invalid publish date").optional(),expiryDate:s.string().datetime("Invalid expiry date").optional()}),Kn=s.object({type:St,title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),message:s.string().min(10,"Message must be at least 10 characters").max(2000,"Message too long"),priority:Et.default("medium"),status:Ut.default("active"),studentId:m,teacherId:m,classId:m,subjectId:m,targetAudience:s.enum(["all","students","teachers","parents"]).optional(),authorId:m,isRead:s.boolean().default(!1)}),An=s.object({title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),description:s.string().max(5000,"Description too long").optional().nullable(),type:It,startDate:x,endDate:x,startTime:Re,endTime:Re,location:s.string().max(200,"Location too long").optional().nullable(),venue:s.string().max(200,"Venue too long").optional().nullable(),organizerId:m,classId:m.nullable(),sectionId:m.nullable(),visibility:bt.default("public"),status:Vt.default("scheduled"),capacity:p().int().positive("Capacity must be positive").optional().nullable(),registrationRequired:s.boolean().default(!1),registrationDeadline:Q.nullable(),attachments:s.any().optional().nullable(),notes:s.string().max(2000,"Notes too long").optional().nullable()}),Bn=s.object({eventId:m,participantId:m,participantType:_t,attendanceStatus:De.optional().nullable(),notes:s.string().max(500,"Notes too long").optional().nullable()}),Pn=s.object({id:m,category:$t,title:s.string().min(3,"Title must be at least 3 characters").max(200,"Title too long"),amount:p().positive("Amount must be greater than 0").max(1e7,"Amount too large"),expenseDate:x,paymentMethod:ge.optional().nullable(),paymentDate:Q.nullable(),vendor:s.string().max(200,"Vendor name too long").optional().nullable(),invoiceNumber:s.string().max(100,"Invoice number too long").optional().nullable(),receiptNumber:s.string().max(100,"Receipt number too long").optional().nullable(),checkNumber:s.string().max(50,"Check number too long").optional().nullable(),transactionRef:s.string().max(100,"Transaction reference too long").optional().nullable(),status:Jt.default("pending"),notes:s.string().max(1000,"Notes too long").optional().nullable()}),In=s.object({action:s.enum(["approve","reject"]),rejectionReason:s.string().min(10,"Rejection reason must be at least 10 characters").max(1000,"Rejection reason too long").optional().nullable()}),Vn=s.object({paymentMethod:ge,paymentDate:x,checkNumber:s.string().max(50,"Check number too long").optional().nullable(),transactionRef:s.string().max(100,"Transaction reference too long").optional().nullable(),notes:s.string().max(1000,"Notes too long").optional().nullable()}),bn=s.object({id:m,name:s.string().min(2,"Vehicle name must be at least 2 characters").max(100,"Vehicle name too long"),brand:s.string().min(2,"Brand must be at least 2 characters").max(100,"Brand too long"),model:s.string().min(2,"Model must be at least 2 characters").max(100,"Model too long"),year:p().int().min(1900,"Year must be after 1900").max(new Date().getFullYear()+1,"Year cannot be in future"),type:Ht.default("fullbus"),capacity:p().int().min(1,"Capacity must be at least 1").max(200,"Capacity cannot exceed 200"),licensePlate:s.string().min(2,"License plate must be at least 2 characters").max(50,"License plate too long"),driverId:m.nullable(),image:s.string().max(500,"Image path too long").optional().nullable().default("novehicle.png"),purchaseDate:Q.nullable(),purchasePrice:p().min(0,"Purchase price must be non-negative").max(1e7,"Purchase price too large").optional().nullable(),initialMileage:p().min(0,"Initial mileage must be non-negative").max(1e7,"Initial mileage too large").optional().nullable(),currentMileage:p().min(0,"Current mileage must be non-negative").max(1e7,"Current mileage too large").optional().nullable(),status:Ct.default("active"),notes:s.string().max(1000,"Notes too long").optional().nullable()}),_n=s.object({id:m,busId:m,refuelDate:x,quantity:p().positive("Quantity must be greater than 0").max(1e4,"Quantity too large"),unitPrice:p().positive("Unit price must be greater than 0").max(1e5,"Unit price too large"),totalCost:p().positive("Total cost must be greater than 0").max(1e7,"Total cost too large").optional(),fuelType:Xt.default("diesel"),odometer:p().min(0,"Odometer must be non-negative").max(1e7,"Odometer value too large").optional().nullable(),fuelStation:s.string().max(200,"Fuel station name too long").optional().nullable(),invoiceNumber:s.string().max(100,"Invoice number too long").optional().nullable(),paymentMethod:ge.optional().nullable(),paidBy:m.nullable(),status:Wt.default("completed"),notes:s.string().max(1000,"Notes too long").optional().nullable()}),$n=s.object({schoolName:s.string().min(2,"School name must be at least 2 characters").max(200,"School name too long"),schoolAddress:s.string().max(500,"School address too long").optional(),schoolPhone:z,schoolEmail:oe,schoolWebsite:s.string().url("Must be a valid URL").max(255,"School website URL too long").optional(),schoolLogo:s.string().url("Must be a valid image URL").max(255,"School logo URL too long").optional(),currentAcademicYear:Ze,gradingScale:s.any().optional(),attendanceRequirement:p().min(0,"Attendance requirement must be non-negative").max(100,"Attendance requirement cannot exceed 100").default(75),maxClassSize:p().int("Max class size must be an integer").min(1,"Max class size must be at least 1").max(200,"Max class size cannot exceed 200").default(34),minimumPassingGrade:p().min(0,"Minimum passing grade must be non-negative").max(100,"Minimum passing grade cannot exceed 100").default(60),defaultExamDuration:p().int("Default exam duration must be in minutes").min(15,"Exam duration must be at least 15 minutes").max(480,"Exam duration cannot exceed 480 minutes").default(120),calendarSystem:ft.default("SEMESTER"),startMonth:s.string().default("september"),endMonth:s.string().default("june"),academicAlerts:s.boolean().default(!0),attendanceAlerts:s.boolean().default(!0),eventAlerts:s.boolean().default(!0),homeworkAlerts:s.boolean().default(!0),feesReminder:s.boolean().default(!0),feesOverdueAlerts:s.boolean().default(!0),emailNotifications:s.boolean().default(!0),smsNotifications:s.boolean().default(!1),parentNotifications:s.boolean().default(!0),lowGradeAlerts:s.boolean().default(!0),allowLateSubmission:s.boolean().default(!0),examResultsAlerts:s.boolean().default(!0),disciplinaryAlerts:s.boolean().default(!0),achievementAlerts:s.boolean().default(!0),maintenanceNotifications:s.boolean().default(!0),twoFactorEnabled:s.boolean().default(!1),sessionTimeout:s.string().regex(/^\d{1,4}$/,"Session timeout must be a number between 1-9999 minutes").default("60"),passwordRequireSymbols:s.boolean().default(!0),loginNotifications:s.boolean().default(!0),parentAccessEnabled:s.boolean().default(!0),teacherAccessEnabled:s.boolean().default(!0),studentAccessEnabled:s.boolean().default(!0),timeZone:s.string().min(1,"Time zone is required").default("UTC"),language:gt.default("en"),theme:s.enum(["light","dark","system"]).default("system"),dateFormat:s.enum(["YYYY-MM-DD","MM/DD/YYYY","DD/MM/YYYY","DD-MM-YY","DD-MM-YYYY"]).default("MM/DD/YYYY"),timeFormat:s.enum(["12","24"]).default("12"),currency:s.string().length(3,"Currency must be a 3-letter ISO code").regex(/^[A-Z]{3}$/,"Currency must be uppercase ISO code").default("USD"),gradingPeriods:p().int("Grading periods must be an integer").min(1,"Grading periods must be at least 1").max(12,"Grading periods cannot exceed 12").default(4),schoolStartTime:s.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/,"Invalid start time format (HH:MM)").default("08:00"),schoolEndTime:s.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/,"Invalid end time format (HH:MM)").default("15:00"),lunchBreakDuration:p().int("Lunch break duration must be in minutes").min(15,"Lunch break must be at least 15 minutes").max(120,"Lunch break cannot exceed 120 minutes").default(30),maintenanceMode:s.boolean().default(!1),autoBackup:s.boolean().default(!0)}),Jn=s.object({id:m}),Mn=s.object({page:p().int().min(1).default(1),limit:p().int().min(1).max(100).default(10)}),Cn=s.object({dateFrom:x,dateTo:x});import{permissionsTable as w,rolePermissionsTable as S,rolesTable as fe}from"@/database/schema";import{eq as N,and as Qt}from"drizzle-orm";import{Repository as rr}from"najm-api";class _{async getAll(){return await this.db.select().from(w)}async getById(e){let[t]=await this.db.select().from(w).where(N(w.id,e));return t}async getByName(e){let[t]=await this.db.select().from(w).where(N(w.name,e));return t}async create(e){let[t]=await this.db.insert(w).values(e).returning();return t}async update(e,t){let[a]=await this.db.update(w).set(t).where(N(w.id,e)).returning();return a}async delete(e){let[t]=await this.db.delete(w).where(N(w.id,e)).returning();return t}async getPermissionsByRole(e){return await this.db.select({id:w.id,name:w.name,description:w.description,resource:w.resource,action:w.action}).from(S).leftJoin(w,N(S.permissionId,w.id)).where(N(S.roleId,e))}async getRolesByPermission(e){return await this.db.select({id:fe.id,name:fe.name,description:fe.description}).from(S).leftJoin(fe,N(S.roleId,fe.id)).where(N(S.permissionId,e))}async assignPermissionToRole(e,t){let[a]=await this.db.insert(S).values({roleId:e,permissionId:t}).returning();return a}async removePermissionFromRole(e,t){let[a]=await this.db.delete(S).where(Qt(N(S.roleId,e),N(S.permissionId,t))).returning();return a}async checkRoleHasPermission(e,t){let[a]=await this.db.select().from(S).where(Qt(N(S.roleId,e),N(S.permissionId,t)));return!!a}async deleteAll(){return await this.db.delete(S),await this.db.delete(w).returning()}}_=n([rr()],_);import{createGuard as ar,Injectable as ir,GuardParams as nr,Headers as or,Ctx as cr}from"najm-api";class ee{tokenService;constructor(e){this.tokenService=e}async getUserPermissions(e){let t=await this.tokenService.getUserPermissions(e);if(!t||!Array.isArray(t))return null;return t}checkPermissionMatch(e,t){if(e.includes(t))return!0;let[a,i]=t.split(":");if(a&&i){if(e.includes(`${a}:*`))return!0;if(e.includes(`*:${i}`))return!0}if(e.includes("*:*"))return!0;return!1}async hasPermission(e,t,a){await this.tokenService.storeUserInCache(e,t);let i=await this.getUserPermissions(e);if(!i)return!1;return this.checkPermissionMatch(i,a)}}n([u(0,or("authorization")),u(1,cr()),u(2,nr()),r("design:type",Function),r("design:paramtypes",[Object,Object,Object]),r("design:returntype",Object)],ee.prototype,"hasPermission",null),ee=n([ir(),r("design:paramtypes",[typeof A==="undefined"?Object:A])],ee);var ur=(...e)=>ar(ee,"hasPermission")(...e);import{Controller as dr,Get as Ae,Post as Zt,Put as hr,Delete as je,Params as Z,Body as jt,t as $}from"najm-api";import{Injectable as yr}from"najm-api";import{Injectable as pr,t as Ke}from"najm-api";import{parseSchema as lr}from"@/shared";import{z as we}from"zod";var mr=we.object({name:we.string().min(1,"Permission name is required"),description:we.string().optional(),resource:we.string().min(1,"Resource is required"),action:we.string().min(1,"Action is required")});class L{permissionRepository;roleValidator;constructor(e,t){this.permissionRepository=e;this.roleValidator=t}async validateCreatePermission(e){return lr(mr,e)}async isPermissionExists(e){return!!await this.permissionRepository.getById(e)}async isPermissionNameExists(e){return!!await this.permissionRepository.getByName(e)}async checkPermissionExists(e){let t=await this.isPermissionExists(e);if(!t)throw new Error(Ke("permissions.errors.notFound"));return t}async checkPermissionExistsByName(e){let t=await this.isPermissionNameExists(e);if(!t)throw new Error(Ke("permissions.errors.notFound"));return t}async checkPermissionNameUnique(e,t=null){if(!e)return;let a=await this.permissionRepository.getByName(e);if(a&&a.id!==t)throw new Error(Ke("permissions.errors.nameExists"))}async checkRoleExists(e){return await this.roleValidator.checkRoleExists(e)}async checkRoleExistsByName(e){return await this.roleValidator.checkRoleExistsByName(e)}async checkRoleHasPermission(e,t){if(await this.roleValidator.checkRoleExists(e),await this.checkPermissionExists(t),await this.permissionRepository.checkRoleHasPermission(e,t))throw new Error(Ke("permissions.errors.roleAlreadyHasPermission"))}}L=n([pr(),r("design:paramtypes",[typeof _==="undefined"?Object:_,typeof K==="undefined"?Object:K])],L);class Y{permissionRepository;permissionValidator;roleService;constructor(e,t,a){this.permissionRepository=e;this.permissionValidator=t;this.roleService=a}async getAll(){return await this.permissionRepository.getAll()}async getById(e){return await this.permissionValidator.checkPermissionExists(e),await this.permissionRepository.getById(e)}async getByName(e){return await this.permissionRepository.getByName(e)}async getByResource(e){return await this.permissionRepository.getAll().then((t)=>t.filter((a)=>a.resource===e))}async create(e){return await this.permissionValidator.validateCreatePermission(e),await this.permissionValidator.checkPermissionNameUnique(e.name),await this.permissionRepository.create(e)}async update(e,t){return await this.permissionValidator.checkPermissionExists(e),await this.permissionValidator.checkPermissionNameUnique(t.name,e),await this.permissionRepository.update(e,t)}async delete(e){return await this.permissionValidator.checkPermissionExists(e),await this.permissionRepository.delete(e)}async getPermissionsByRole(e){return await this.permissionRepository.getPermissionsByRole(e)}async getRolesByPermission(e){return await this.permissionValidator.checkPermissionExists(e),await this.permissionRepository.getRolesByPermission(e)}async assignPermissionToRole(e,t){return await this.permissionValidator.checkRoleHasPermission(e,t),await this.permissionRepository.assignPermissionToRole(e,t)}async removePermissionFromRole(e,t){return await this.permissionRepository.removePermissionFromRole(e,t)}async seedDefaultPermissions(e){let t=[];for(let a of e)try{let i=await this.create(a);t.push(i)}catch(i){continue}return t}async seedDefaultRolePermissions(e){let t=[];for(let{roleName:a,permissions:i}of e)try{await this.permissionValidator.checkRoleExistsByName(a);let l=await this.roleService.getByName(a);for(let c of i)try{await this.permissionValidator.checkPermissionExistsByName(c);let d=await this.getByName(c);await this.permissionValidator.checkRoleHasPermission(l.id,d.id),await this.assignPermissionToRole(l.id,d.id),t.push({role:a,permission:c})}catch(d){continue}}catch(l){continue}return t}async deleteAll(){return await this.permissionRepository.deleteAll()}}Y=n([yr(),r("design:paramtypes",[typeof _==="undefined"?Object:_,typeof L==="undefined"?Object:L,typeof B==="undefined"?Object:B])],Y);import{isAdmin as Lt}from"@/roles/RoleGuards";class ce{permissionService;constructor(e){this.permissionService=e}async getPermissions(){return{data:await this.permissionService.getAll(),message:$("permissions.success.retrieved"),status:"success"}}async getPermission(e){return{data:await this.permissionService.getById(e),message:$("permissions.success.retrieved"),status:"success"}}async create(e){return{data:await this.permissionService.create(e),message:$("permissions.success.created"),status:"success"}}async update(e,t){return{data:await this.permissionService.update(e,t),message:$("permissions.success.updated"),status:"success"}}async delete(e){return{data:await this.permissionService.delete(e),message:$("permissions.success.deleted"),status:"success"}}async getByRole(e){return{data:await this.permissionService.getPermissionsByRole(e),message:$("permissions.success.retrieved"),status:"success"}}async getRolesByPermission(e){return{data:await this.permissionService.getRolesByPermission(e),message:$("permissions.success.retrieved"),status:"success"}}async assignToRole(e,t){return{data:await this.permissionService.assignPermissionToRole(e,t),message:$("permissions.success.assigned"),status:"success"}}async removeFromRole(e,t){return{data:await this.permissionService.removePermissionFromRole(e,t),message:$("permissions.success.removed"),status:"success"}}async deleteAll(){return{data:await this.permissionService.deleteAll(),message:$("permissions.success.allDeleted"),status:"success"}}}n([Ae(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],ce.prototype,"getPermissions",null),n([Ae("/:id"),u(0,Z("id")),r("design:type",Function),r("design:paramtypes",[String]),r("design:returntype",Promise)],ce.prototype,"getPermission",null),n([Zt(),u(0,jt()),r("design:type",Function),r("design:paramtypes",[Object]),r("design:returntype",Promise)],ce.prototype,"create",null),n([hr("/:id"),u(0,Z("id")),u(1,jt()),r("design:type",Function),r("design:paramtypes",[String,Object]),r("design:returntype",Promise)],ce.prototype,"update",null),n([je("/:id"),u(0,Z("id")),r("design:type",Function),r("design:paramtypes",[String]),r("design:returntype",Promise)],ce.prototype,"delete",null),n([Ae("/role/:roleId"),u(0,Z("roleId")),r("design:type",Function),r("design:paramtypes",[String]),r("design:returntype",Promise)],ce.prototype,"getByRole",null),n([Ae("/roles/:permissionId"),u(0,Z("permissionId")),r("design:type",Function),r("design:paramtypes",[String]),r("design:returntype",Promise)],ce.prototype,"getRolesByPermission",null),n([Zt("/assign/:roleId/:permissionId"),u(0,Z("roleId")),u(1,Z("permissionId")),r("design:type",Function),r("design:paramtypes",[String,String]),r("design:returntype",Promise)],ce.prototype,"assignToRole",null),n([je("/remove/:roleId/:permissionId"),u(0,Z("roleId")),u(1,Z("permissionId")),r("design:type",Function),r("design:paramtypes",[String,String]),r("design:returntype",Promise)],ce.prototype,"removeFromRole",null),n([je(),Lt(),r("design:type",Function),r("design:paramtypes",[]),r("design:returntype",Promise)],ce.prototype,"deleteAll",null),ce=n([dr("/permissions"),Lt(),r("design:paramtypes",[typeof Y==="undefined"?Object:Y])],ce);import*as Yt from"fs/promises";import*as Le from"path";import gr from"lodash.isempty";import{sql as v}from"drizzle-orm";var fr=Le.join(process.cwd(),"avatars"),Ao=async(e,t)=>{try{return await e.parseAsync(t)}catch(a){let l=(a.issues||a.errors||[]).map((c)=>`${c.path.join(".")}: ${c.message}`).join("; ");throw new Error(l)}},Bo=(e)=>{let t={};for(let[a,i]of Object.entries(e))if(i!==null&&i!==void 0&&i!=="")t[a]=i;return t},Po=async(e)=>{try{let t=Le.join(fr,e),a=await Yt.readFile(t);return new File([a],e,{type:"image/png"})}catch(t){return null}},Gt=(e)=>{if(!e)return null;let t;if(e instanceof Date)t=e;else if(typeof e==="string")t=new Date(e);else return null;if(isNaN(t.getTime()))return null;return t.toISOString().split("T")[0]};function Io(e){if(!e)return null;let t=Gt(e);if(!t)return null;let a=new Date(t),i=new Date,l=i.getFullYear()-a.getFullYear(),c=i.getMonth()-a.getMonth();if(c<0||c===0&&i.getDate()<a.getDate())l--;return l}function Vo(e){if(!e)return null;let t=Gt(e);if(!t)return null;let a=new Date(t),i=new Date,l=i.getFullYear()-a.getFullYear(),c=i.getMonth()-a.getMonth();if(c<0||c===0&&i.getDate()<a.getDate())l--;return l}function bo(e,t){let a={};for(let i of t)if(e[i]!==void 0)a[i]=e[i];return a}var _o=gr;function $o(e,t){let i=Object.entries(e).flatMap(([c,d])=>[v.raw(`'${c}'`),d]),l=v`json_agg(json_build_object(${v.join(i,v`, `)})`;if(t)l=v`${l} ORDER BY ${t}`;return l=v`${l})`,v`COALESCE(${l}, '[]'::json)`}function Jo(e,t,a,i){let c=Object.entries(e).flatMap(([E,ue])=>[v.raw(`'${E}'`),ue]),d=v`
2
- SELECT json_agg(DISTINCT json_build_object(${v.join(c,v`, `)}))
3
- FROM ${t}
4
- WHERE ${a}
5
- `;if(i)d=v`
6
- SELECT json_agg(json_build_object(${v.join(c,v`, `)}) ORDER BY ${i})
7
- FROM ${t}
8
- WHERE ${a}
9
- `;return v`COALESCE((${d}), '[]'::json)`}var Mo=(e)=>typeof e==="string"&&e.trim().length>0&&(e.startsWith("/")||e.startsWith("http")||e.startsWith("storage/")),Co=(e)=>!!e&&typeof e!=="string"&&e instanceof File;import{drizzle as Sr}from"drizzle-orm/postgres-js";var Oe={};os(Oe,{usersTable:()=>Ye,tokensTable:()=>zt,timestamps:()=>ve,rolesTable:()=>Ie,rolePermissionsTable:()=>es,permissionsTable:()=>Ge,idField:()=>te});import{pgTable as xe,text as k,boolean as vr,timestamp as Pe}from"drizzle-orm/pg-core";import{nanoid as kr}from"nanoid";import{sql as Tr}from"drizzle-orm";import{getEnumConfig as wr}from"@/lib/ENUMS";import{pgEnum as xr}from"drizzle-orm/pg-core";var Be=(e)=>{let t=wr(e);if(!t)throw new Error(`Enum ${e} not found`);let a=t.name||e;return xr(a,t.values)},Ot=Be("userStatus"),qt=Be("tokenStatus"),Ft=Be("tokenType"),Do=Be("studentStatus");var ve={createdAt:Pe("created_at",{mode:"string"}).defaultNow(),updatedAt:Pe("updated_at",{mode:"string"}).defaultNow().$onUpdate(()=>Tr`CURRENT_TIMESTAMP`)},te=(e=5)=>k("id").primaryKey().notNull().$defaultFn(()=>kr(e)),Ie=xe("roles",{id:te(),name:k("name").notNull(),description:k("description")}),Ye=xe("users",{id:te(8),email:k("email").notNull().unique(),emailVerified:vr("email_verified").default(!1),password:k("password").notNull(),image:k("image").default("noavatar.png"),status:Ot("status").default("pending"),roleId:k("role_id").references(()=>Ie.id),lastLogin:Pe("last_login",{mode:"string"}),...ve}),zt=xe("tokens",{id:te(10),userId:k("user_id").references(()=>Ye.id,{onDelete:"cascade"}).unique().notNull(),token:k("token").notNull(),type:Ft("type").default("refresh"),status:qt("status").default("active"),expiresAt:Pe("expires_at",{mode:"string"}).notNull(),...ve}),Ge=xe("permissions",{id:te(),name:k("name").notNull().unique(),description:k("description"),resource:k("resource").notNull(),action:k("action").notNull(),...ve}),es=xe("role_permissions",{id:te(),roleId:k("role_id").references(()=>Ie.id).notNull(),permissionId:k("permission_id").references(()=>Ge.id).notNull(),...ve});var ts=process.env.DB_URL?Sr(process.env.DB_URL,{schema:Oe}):null;if(ts)console.warn("⚠️ najm-auth: Using standalone DB instance. This should only be used for development.");var Er={name:"najm-auth",version:"0.1.3",database:"auth",controllers:[ye,de,le,ce],services:[W,F,G,J,B,Y,A],repositories:[M,H,_,O],providers:[C,K,L,q,ee],onSetup:async()=>{console.log(" ✓ najm-auth plugin initialized"),console.log(" - Authentication endpoints ready"),console.log(" - Authorization system active"),console.log(" - Database connection established")},onTeardown:async()=>{console.log(" ✓ najm-auth plugin cleanup complete")}};export{Ht as vehicleTypeEnum,Ct as vehicleStatusEnum,bn as vehicleSchema,tn as vehicleDocumentTypeEnum,Ye as usersTable,Xi as userTypeEnum,pt as userStatusEnum,pn as userSchema,en as trackerModeEnum,zt as tokensTable,Qi as tokenTypeEnum,Di as tokenStatusEnum,mt as teacherStatusEnum,Os as teacherProfessionalSchema,Gs as teacherPersonalSchema,yn as teacherFullSchema,Oi as submissionTypeEnum,vn as subjectSchema,lt as studentStatusEnum,Ls as studentSchema,$n as settingsSchema,ji as semesterEnum,ht as sectionStatusEnum,kn as sectionSchema,Kt as scheduleEnum,Ie as rolesTable,ln as roleSchema,es as rolePermissionsTable,dt as relationshipTypeEnum,Wt as refuelStatusEnum,_n as refuelSchema,Fi as proficiencyLevelEnum,bo as pickProps,Ge as permissionsTable,Nt as paymentTypeEnum,Pt as paymentStatusEnum,ge as paymentMethodEnum,wn as paymentAllocationSchema,_t as participantTypeEnum,Ao as parseSchema,er as parentsSchema,Ys as parentSchema,Mn as paginationSchema,nn as maritalStatusEnum,rn as maintenanceTypeEnum,an as maintenanceStatusEnum,gt as languageEnum,Jo as jsonAggSubquery,$o as jsonAgg,Aa as isTeacher,Pa as isStudent,ba as isStaff,Ka as isSecretary,Ra as isPrincipal,Mo as isPath,Ba as isParent,Va as isFinancial,Co as isFile,_o as isEmpty,_a as isAuth,Ia as isAdministrator,ie as isAdmin,Na as isAccounting,Jn as idParamSchema,te as idField,Tt as gradeStatusEnum,Un as gradeSchema,Mi as getEnumValues,Ji as getEnumConfig,Po as getAvatarFile,he as genderEnum,xn as fullStudentSchema,Xt as fuelTypeEnum,Gt as formatDate,Zi as fileStatusEnum,tr as feesSchema,Rt as feeTypeStatusEnum,dn as feeTypeSchema,At as feeStatusEnum,zs as feeSchema,fn as feePaymentSchema,Bt as feeInstallmentStatusEnum,gn as feeInstallmentSchema,Jt as expenseStatusEnum,Pn as expenseSchema,Vn as expensePaymentSchema,$t as expenseCategoryEnum,In as expenseApprovalSchema,vt as examTypeEnum,kt as examStatusEnum,qi as examSecurityEnum,Rn as examSchema,bt as eventVisibilityEnum,It as eventTypeEnum,Vt as eventStatusEnum,An as eventSchema,Bn as eventParticipantSchema,Yi as enrollmentStatusEnum,yt as employmentTypeEnum,Mt as driverStatusEnum,mn as driverSchema,ts as db,zi as dayOfWeekEnum,Cn as dateRangeSchema,Bo as clean,Li as classStatusEnum,Tn as classSchema,ft as calendarSystemEnum,Vo as calculateYearsOfExperience,Io as calculateAge,sn as busStatusEnum,Dt as bulkFeeItemSchema,hn as bulkFeeFormSchema,En as bulkAssessmentSchema,fr as avatarsPath,De as attendanceStatusEnum,Sn as attendanceSchema,Fs as assignmentsSchema,Gi as assignmentStatusEnum,qs as assignmentSchema,wt as assessmentTypeEnum,xt as assessmentStatusEnum,sr as assessmentSchema,Nn as announcementSchema,St as alertTypeEnum,Ut as alertStatusEnum,Kn as alertSchema,Et as alertPriorityEnum,C as UserValidator,W as UserService,M as UserRepository,ye as UserController,A as TokenService,O as TokenRepository,K as RoleValidator,B as RoleService,H as RoleRepository,q as RoleGuards,le as RoleController,Ee as RoleChecker,V as Role,tt as ROLE_GROUPS,f as ROLES,L as PermissionValidator,Y as PermissionService,_ as PermissionRepository,ee as PermissionGuards,ce as PermissionController,ur as Permission,J as EncryptionService,Ue as ENUMS,G as CookieService,F as AuthService,Er as AuthPlugin,de as AuthController};
10
-
11
- //# debugId=46440D7F5E9F307264756E2164756E21
12
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src\\auth\\EncryptionService.ts", "src\\auth\\CookieService.ts", "src\\auth\\AuthController.ts", "src\\auth\\AuthService.ts", "src\\users\\UserRepository.ts", "src\\users\\UserValidator.ts", "src\\users\\UserService.ts", "src\\roles\\RoleRepository.ts", "src\\roles\\RoleValidator.ts", "src\\roles\\RoleGuards.ts", "src\\tokens\\TokenRepository.ts", "src\\tokens\\TokenService.ts", "src\\roles\\RoleController.ts", "src\\roles\\RoleService.ts", "src\\users\\UserController.ts", "src\\lib\\ENUMS.ts", "src\\lib\\validations.ts", "src\\lib\\ZodEnum.ts", "src\\permissions\\PermissionRepository.ts", "src\\permissions\\PermissionGuards.ts", "src\\permissions\\PermissionController.ts", "src\\permissions\\PermissionService.ts", "src\\permissions\\PermissionValidator.ts", "src\\shared\\index.ts", "src\\database\\db.ts", "src\\database\\schema\\index.ts", "src\\database\\schema\\PgEnum.ts", "src\\plugin.ts"],
  "sourcesContent": [
    "\nimport { Injectable } from 'najm-api';\nimport bcrypt from 'bcrypt'\n\n@Injectable()\nexport class EncryptionService {\n  constructor() { }\n\n  async hashPassword(password) {\n    if (!password) return null;\n    if (typeof password !== 'string' || password.trim().length === 0) {\n      return null;\n    }\n    return bcrypt.hash(password, 10);\n  }\n\n  async comparePassword(password, hashedPassword) {\n    return bcrypt.compare(password, hashedPassword);\n  }\n\n}\n",
    "\nimport { setCookie, Injectable, deleteCookie } from 'najm-api';\nimport timestring from 'timestring';\n\n@Injectable()\nexport class CookieService {\n\n  setRefreshCookie(refreshToken) {\n    const maxAge = timestring(process.env.REFRESH_EXPIRES_IN, 's');  /// '1y'\n    setCookie('refreshToken', refreshToken, {\n      httpOnly: false,\n      sameSite: 'Lax',\n      maxAge,\n      path: '/api/auth/refresh',\n    })\n  }\n\n  clearRefreshCookie(): void {\n    deleteCookie('refreshToken', {\n      httpOnly: false,\n      sameSite: 'Lax',\n      path: '/api/auth/refresh',\n      maxAge: 0\n    });\n  }\n\n\n}\n",
    "import { Controller, Get, Post, Params, Body, User, t } from 'najm-api';\nimport { AuthService } from './AuthService';\nimport { isAuth } from '@/roles/RoleGuards';\n\n\n@Controller('/auth')\nexport class AuthController {\n  constructor(private authService: AuthService) { }\n\n  @Post('/register')\n  async registerUser(@Body() body) {\n    const data = await this.authService.registerUser(body);\n    return {\n      data,\n      message: t('auth.success.register'),\n      status: 'success'\n    };\n  }\n\n  @Post('/login')\n  async loginUser(@Body() body) {\n    const data = await this.authService.loginUser(body);\n    return {\n      data,\n      message: t('auth.success.login'),\n      status: 'success'\n    };\n  }\n\n  @Get('/refresh')\n  async refreshTokens() {\n    const data = await this.authService.refreshTokens();\n    return {\n      data,\n      message: t('auth.success.tokenRefreshed'),\n      status: 'success'\n    };\n  }\n\n  @Get('/logout/:id')\n  async logoutUser(@Params('id') id) {\n    const data = await this.authService.logoutUser(id);\n    return {\n      data,\n      message: t('auth.success.logout'),\n      status: 'success'\n    };\n  }\n\n\n  @Get('/me')\n  @isAuth()\n  async userProfile(@User() user) {\n    const data = await this.authService.getUserProfile(user);\n    return {\n      data,\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post('/forgot-password')\n  @isAuth()\n  async forgotPassword(@Body() body) {\n    const data = await this.authService.forgotPassword(body.email);\n    return {\n      data,\n      message: t('auth.success.passwordReset'),\n      status: 'success'\n    };\n  }\n}",
    "import { t } from 'najm-api';\nimport { Injectable, getCurrentLanguage } from 'najm-api';\nimport { UserService, UserValidator } from '../users';\nimport { TokenService } from '../tokens';\nimport { CookieService } from './CookieService';\n\n@Injectable()\nexport class AuthService {\n\n  constructor(\n    private tokenService: TokenService,\n    private userService: UserService,\n    private userValidator: UserValidator,\n    private cookieService: CookieService,\n  ) { }\n\n  async registerUser(body) {\n    return await this.userService.create(body)\n  }\n\n  async loginUser(body) {\n    const { email, password } = body;\n\n    if (!email || !password) {\n      throw new Error(t('auth.errors.invalidCredentials'))\n    }\n\n    const existingPassword = await this.userService.getPassword(email);\n    const { id } = await this.userService.getByEmail(email);\n    await this.userValidator.checkPasswordValid(password, existingPassword);\n\n    const data = await this.tokenService.generateTokens(id);\n    this.cookieService.setRefreshCookie(data.refreshToken)\n    return data;\n  }\n\n  async refreshTokens() {\n    const data = await this.tokenService.refreshTokens();\n    this.cookieService.setRefreshCookie(data.refreshToken);\n    return data;\n  }\n\n  async logoutUser(userId) {\n    await this.userValidator.checkUserExists(userId);\n    await this.tokenService.revokeToken(userId);\n    this.cookieService.clearRefreshCookie();\n    return { data: null, message: t('auth.success.logout') };\n  }\n\n  async getUserProfile(userData) {\n    const lang = getCurrentLanguage();\n    return {\n      ...userData,\n      language: lang,\n    };\n  }\n\n  async forgotPassword(email) {\n\n  }\n}\n\n\n",
    "\nimport { rolesTable, usersTable, permissionsTable, rolePermissionsTable } from '@/database/schema';\nimport { eq, ne } from 'drizzle-orm';\nimport { Repository,DB } from 'najm-api';\n\n@Repository()\nexport class UserRepository {\n  declare db: DB;\n\n  private getUser() {\n    return {\n      id: usersTable.id,\n      email: usersTable.email,\n      emailVerified: usersTable.emailVerified,\n      image: usersTable.image,\n      status: usersTable.status,\n      roleId: usersTable.roleId,\n      role: rolesTable.name,\n      createdAt: usersTable.createdAt,\n      updatedAt: usersTable.updatedAt,\n    };\n  }\n\n  async getAll() {\n    const allUsers = await this.db\n      .select(this.getUser())\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id));\n\n    // Add permissions to each user\n    return Promise.all(allUsers.map(async (user) => ({\n      ...user,\n      permissions: await this.getUserPermissions(user.id)\n    })));\n  }\n\n  async getById(id) {\n    const [user] = await this.db\n      .select(this.getUser())\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id))\n      .where(eq(usersTable.id, id))\n      .limit(1);\n\n    if (!user) return user;\n\n    return {\n      ...user,\n      permissions: await this.getUserPermissions(user.id)\n    };\n  }\n\n  async getByEmail(email) {\n    const [existingUser] = await this.db\n      .select(this.getUser())\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id))\n      .where(eq(usersTable.email, email));\n    return existingUser;\n  }\n\n\n  async create(data) {\n    const [newUser] = await this.db.insert(usersTable).values(data).returning();\n    return newUser;\n  }\n\n  async update(id, data) {\n    const [updatedUser] = await this.db.update(usersTable).set(data).where(eq(usersTable.id, id)).returning();\n    return updatedUser;\n  }\n\n  async delete(id) {\n    const [deletedUser] = await this.db.delete(usersTable).where(eq(usersTable.id, id)).returning();\n    return deletedUser;\n  }\n\n  async deleteAll() {\n    const adminRole = await this.db\n      .select({ id: rolesTable.id })\n      .from(rolesTable)\n      .where(eq(rolesTable.name, 'admin'))\n      .limit(1);\n\n    if (adminRole.length === 0) {\n      const deletedUsers = await this.db.delete(usersTable).returning();\n      return deletedUsers;\n    }\n    const deletedUsers = await this.db\n      .delete(usersTable)\n      .where(ne(usersTable.roleId, adminRole[0].id))\n      .returning();\n\n    return deletedUsers;\n  }\n\n  async getRoleNameById(userId) {\n    const [role] = await this.db.select({\n      roleName: rolesTable.name\n    })\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id))\n      .where(eq(usersTable.id, userId))\n\n    return role.roleName\n  }\n\n  async getUserPassword(email) {\n    const [user] = await this.db\n      .select({\n        id: usersTable.id,\n        email: usersTable.email,\n        password: usersTable.password\n      })\n      .from(usersTable)\n      .where(eq(usersTable.email, email))\n      .limit(1);\n    return user.password;\n  }\n\n  async getUserPermissions(userId) {\n    const [user] = await this.db\n      .select({ roleId: usersTable.roleId })\n      .from(usersTable)\n      .where(eq(usersTable.id, userId))\n      .limit(1);\n\n    if (!user || !user.roleId) return [];\n\n    const userPermissions = await this.db\n      .select({\n        name: permissionsTable.name,\n      })\n      .from(rolePermissionsTable)\n      .leftJoin(permissionsTable, eq(rolePermissionsTable.permissionId, permissionsTable.id))\n      .where(eq(rolePermissionsTable.roleId, user.roleId));\n\n    return userPermissions.map(p => p.name).filter(name => name);\n  }\n\n}",
    "import { Injectable, t } from 'najm-api';\nimport { UserRepository } from './UserRepository';\nimport { EncryptionService } from '../auth';\nimport { parseSchema } from '@/shared';\nimport { userSchema } from '@/lib/validations';\n\n@Injectable()\nexport class UserValidator {\n   constructor(\n      private userRepository: UserRepository,\n      private encryptionService: EncryptionService,\n   ) { }\n\n   async validateCreateUser(data) {\n      return parseSchema(userSchema, data);\n   }\n\n   async isEmailExists(email) {\n      const existingUser = await this.userRepository.getByEmail(email);\n      return !!existingUser\n   }\n\n   async isPasswordValid(password, hashedPassword) {\n      const isPasswordValid = await this.encryptionService.comparePassword(password, hashedPassword);\n      return !!isPasswordValid\n   }\n\n   async isUserExist(id) {\n      const existingUser = await this.userRepository.getById(id);\n      return !!existingUser\n   }\n\n   async checkUserIdIsUnique(id) {\n      if (!id) return;\n      const existingUser = await this.userRepository.getById(id);\n      if (existingUser) {\n         throw new Error(t('users.errors.idExists'));\n      }\n   }\n\n   async isCorrectPass(password) {\n      return password &&\n         typeof password === 'string' &&\n         password.trim().length > 0;\n   }\n\n   async hasRole(userId, roles) {\n      const roleName = await this.userRepository.getRoleNameById(userId);\n\n      if (!roleName) {\n         throw Error(t('auth.errors.accessDenied'))\n      }\n\n      const hasRole = roles.some(\n         item => roleName.toLowerCase() === item.toLowerCase()\n      )\n\n      if (!hasRole) {\n         throw Error(t('auth.errors.accessDenied'))\n      }\n\n      return true\n\n   }\n\n   //======================= throw errors\n\n   async checkUserExistsByEmail(email) {\n      const user = await this.userRepository.getByEmail(email);\n      if (!user) {\n         throw new Error(t('auth.errors.invalidCredentials'))\n      }\n      return user\n   }\n\n   async checkUserExists(id) {\n      const userExists = await this.isUserExist(id);\n      if (!userExists) {\n         throw new Error(t('users.errors.notFound'));\n      }\n      return userExists;\n   }\n\n\n   async checkEmailUnique(email, excludeId = null) {\n      if (!email) return;\n      const existingUser = await this.userRepository.getByEmail(email);\n      if (existingUser && existingUser.id !== excludeId) {\n         throw new Error(t('auth.errors.emailExists'));\n      }\n   }\n\n   async checkEmailExists(email) {\n      const user = await this.userRepository.getByEmail(email);\n      if (!user) {\n         throw new Error(t('users.errors.notFound'));\n      }\n      return user;\n   }\n\n   async checkPasswordValid(password, hashedPassword) {\n      const isPasswordValid = await this.isPasswordValid(password, hashedPassword);\n      if (!isPasswordValid) {\n         throw new Error(t('auth.errors.invalidCredentials'))\n      }\n   }\n\n}",
    "import { Injectable, setLanguage, getCurrentLanguage, Transactional } from 'najm-api';\nimport { UserRepository } from './UserRepository';\nimport { UserValidator } from './UserValidator';\nimport { RoleService, RoleValidator } from '../roles';\nimport { EncryptionService } from '../auth';\nimport { nanoid } from 'nanoid';\nimport { clean } from '@/shared';\n\n\n@Injectable()\nexport class UserService {\n  constructor(\n    private roleValidator: RoleValidator,\n    private roleService: RoleService,\n    private userRepository: UserRepository,\n    private userValidator: UserValidator,\n    private encryptionService: EncryptionService,\n  ) { }\n\n  private sanitizeUser(user) {\n    if (!user) return user;\n    const { password, ...sanitizedUser } = user;\n    return sanitizedUser;\n  }\n\n  private sanitizeUsers(users) {\n    return users.map(user => this.sanitizeUser(user));\n  }\n\n  private async resolveUserRole(roleId, roleName) {\n\n    if (roleId) {\n      await this.roleValidator.checkRoleExists(roleId);\n      return roleId;\n    }\n\n    if (roleName) {\n      const roleByName = await this.roleService.getByName(roleName);\n      if (roleByName) {\n        return roleByName.id;\n      }\n      throw new Error(`Role '${roleName}' not found`);\n    }\n\n    const defaultRole = await this.roleService.getByName('Student');\n    return defaultRole.id;\n  }\n\n  async getAll() {\n    const users = await this.userRepository.getAll();\n    return this.sanitizeUsers(users);\n  }\n\n  async getById(id) {\n    await this.userValidator.checkUserExists(id);\n    const user = await this.userRepository.getById(id);\n    return this.sanitizeUser(user);\n  }\n\n  async getByEmail(email) {\n    const user = await this.userValidator.checkUserExistsByEmail(email);\n    return this.sanitizeUser(user);\n  }\n\n  @Transactional()\n  async create(data) {\n    const { id, email, image, emailVerified, password, roleId, role } = data;\n    let userId = id || nanoid(5);\n    let pass = password || '12345678';\n\n    await this.userValidator.checkEmailUnique(data.email);\n    await this.userValidator.checkUserIdIsUnique(id);\n\n    const hashedPassword = await this.encryptionService.hashPassword(pass);\n    const resolvedRoleId = await this.resolveUserRole(roleId, role);\n\n    const userDetails = {\n      id: userId,\n      email,\n      image,\n      password: hashedPassword,\n      roleId: resolvedRoleId,\n      emailVerified,\n      status: 'pending'\n    };\n\n    await this.userValidator.validateCreateUser(userDetails);\n    const newUser = await this.userRepository.create(userDetails);\n\n    return this.sanitizeUser(newUser);\n\n\n  }\n\n  async update(id, data) {\n\n    const { password, image } = data;\n\n    await this.userValidator.checkUserExists(id);\n    await this.userValidator.checkEmailUnique(data.email, id);\n\n    const currentUser = await this.userRepository.getById(id);\n    const hashedPassword = await this.encryptionService.hashPassword(password);\n\n    const updateData = {\n      ...data,\n      image,\n      ...(hashedPassword && { password: hashedPassword })\n    };\n\n    const cleanedUpdateData = clean(updateData);\n    const updatedUser = await this.userRepository.update(id, cleanedUpdateData);\n    return this.sanitizeUser(updatedUser);\n  }\n\n  async delete(id) {\n    await this.userValidator.checkUserExists(id);\n    const user = await this.userRepository.delete(id);\n    return this.sanitizeUser(user);\n  }\n\n  async deleteAll() {\n    const deletedUsers = await this.userRepository.deleteAll();\n    return this.sanitizeUsers(deletedUsers);\n  }\n\n  async getRoleName(id) {\n    await this.userValidator.checkUserExists(id);\n    return await this.userRepository.getRoleNameById(id);\n  }\n\n  async getPassword(email) {\n    await this.userValidator.checkUserExistsByEmail(email);\n    return await this.userRepository.getUserPassword(email);\n  }\n\n  async assignRole(id, roleId, roleName?) {\n    await this.userValidator.checkUserExists(id);\n    const resolvedRoleId = await this.resolveUserRole(roleId, roleName);\n    const updatedUser = await this.userRepository.update(id, { roleId: resolvedRoleId });\n    return this.sanitizeUser(updatedUser);\n  }\n\n  async removeRole(id) {\n    await this.userValidator.checkUserExists(id);\n    const updatedUser = await this.userRepository.update(id, { roleId: null });\n    return this.sanitizeUser(updatedUser);\n  }\n\n  async seedAdminUser() {\n    const email = 'admin@admin.com';\n    const adminRole = await this.roleValidator.checkAdminRoleExists();\n\n    const existingUser = await this.userRepository.getByEmail(email);\n\n    if (existingUser) {\n      await this.delete(existingUser.id);\n    }\n\n    const newAdminUser = await this.create({\n      id: 'USR00',\n      name: 'System Administrator',\n      email,\n      password: '12345678',\n      image:null,\n      roleId: adminRole.id,\n      status: 'active',\n      emailVerified: true\n    });\n\n    return this.sanitizeUser(newAdminUser);\n  }\n\n  async updateLang(language) {\n    setLanguage(language)\n    return language\n  }\n\n  async getLang() {\n    return getCurrentLanguage();\n  }\n\n}",
    "\n\nimport { rolesTable } from '@/database/schema';\nimport { eq } from 'drizzle-orm';\nimport { Repository, DB } from 'najm-api';\n\n@Repository()\nexport class RoleRepository {\n\n  declare db: DB;\n\n  async getAll() {\n    return await this.db.select().from(rolesTable);\n  }\n\n  async getById(id:string) {\n    const [existingRole] = await this.db.select().from(rolesTable).where(eq(rolesTable.id, id));\n    return existingRole;\n  }\n\n  async getByName(name:string) {\n    const [existingRole] = await this.db.select().from(rolesTable).where(eq(rolesTable.name, name));\n    return existingRole\n  }\n\n  async create(data) {\n    const [newRole] = await this.db.insert(rolesTable).values(data).returning();\n    return newRole\n  }\n\n  async update(id, data) {\n    const [updatedRole] = await this.db.update(rolesTable)\n      .set(data)\n      .where(eq(rolesTable.id, id))\n      .returning();\n\n      return updatedRole\n  }\n\n  async delete(id) {\n    const [deletedRole] = await this.db.delete(rolesTable)\n      .where(eq(rolesTable.id, id))\n      .returning();\n      return deletedRole\n  }\n}\n",
    "import { Injectable, t } from 'najm-api';\nimport { RoleRepository } from './RoleRepository';\nimport { parseSchema } from '@/shared';\nimport { roleSchema } from '@/lib/validations';\n\n\n@Injectable()\nexport class RoleValidator {\n    constructor(\n        private roleRepository: RoleRepository,\n    ) { }\n\n    async validateCreateRole(data) {\n        return parseSchema(roleSchema, data);\n    }\n\n    async isRoleNameExists(roleName: string) {\n        const existingRole = await this.roleRepository.getByName(roleName);\n        return !!existingRole;\n    }\n\n    async isRoleIdExists(id: string) {\n        const existingRole = await this.roleRepository.getById(id);\n        return !!existingRole;\n    }\n\n    async checkNameUnique(roleName: string, excludeId = null) {\n        if (!roleName) return;\n        const existingRole = await this.roleRepository.getByName(roleName);\n\n        if (existingRole && existingRole.id !== excludeId) {\n            throw new Error(t('roles.errors.exists'));\n        }\n    }\n\n    async checkRoleExists(id: string) {\n        const roleIdExists = await this.isRoleIdExists(id);\n        if (!roleIdExists) {\n            throw new Error(t('roles.errors.notFound'));\n        }\n    }\n\n    async checkRoleExistsByName(roleName: string) {\n        const roleNameExists = await this.isRoleNameExists(roleName);\n        if (!roleNameExists) {\n            throw new Error(t('roles.errors.notFound'));\n        }\n    }\n\n    async checkAdminRoleExists() {\n        const adminRole = await this.roleRepository.getByName('admin');\n        if (!adminRole) {\n            throw new Error(t('users.errors.adminRoleNotFound'));\n        }\n        return adminRole;\n    }\n}",
    "import { Injectable, Headers, createGuard, GuardParams, Ctx } from \"najm-api\";\nimport { TokenService } from \"../tokens\";\n\n// ============ CONSTANTS ============ //\n\nexport const ROLES = {\n  ADMIN: 'admin',\n  PRINCIPAL: 'principal',\n  ACCOUNTING: 'accounting',\n  SECRETARY: 'secretary',\n  TEACHER: 'teacher',\n  STUDENT: 'student',\n  PARENT: 'parent',\n} as const;\n\nexport const ROLE_GROUPS = {\n  ADMINISTRATORS: [ROLES.ADMIN, ROLES.PRINCIPAL],\n  FINANCIAL: [ROLES.ADMIN, ROLES.ACCOUNTING],\n  STAFF: [ROLES.ADMIN, ROLES.PRINCIPAL, ROLES.ACCOUNTING, ROLES.SECRETARY, ROLES.TEACHER],\n  END_USERS: [ROLES.STUDENT, ROLES.PARENT],\n  ALL: [ROLES.ADMIN, ROLES.PRINCIPAL, ROLES.ACCOUNTING, ROLES.SECRETARY, ROLES.TEACHER, ROLES.STUDENT, ROLES.PARENT],\n} as const;\n\nexport type Role = typeof ROLES[keyof typeof ROLES];\nexport type RoleGroup = typeof ROLE_GROUPS[keyof typeof ROLE_GROUPS];\n\n// ============ ROLE CHECKER (Injectable for Services) ============ //\n\n@Injectable()\nexport class RoleChecker {\n  isInGroup(userRole, group: readonly string[]): boolean {\n    return group.includes(userRole?.toLowerCase());\n  }\n\n  isAdministrator(userRole): boolean {\n    return this.isInGroup(userRole, ROLE_GROUPS.ADMINISTRATORS);\n  }\n\n  isStaff(userRole): boolean {\n    return this.isInGroup(userRole, ROLE_GROUPS.STAFF);\n  }\n\n  hasAnyRole(userRole, roles: readonly string[]): boolean {\n    return roles.includes(userRole?.toLowerCase());\n  }\n\n  hasExactRole(userRole, requiredRole): boolean {\n    return userRole?.toLowerCase() === requiredRole?.toLowerCase();\n  }\n\n}\n\n// ============ ROLE GUARDS (Injectable for Decorators) ============ //\n\n@Injectable()\nexport class RoleGuards {\n  constructor(\n    private roleChecker: RoleChecker,\n    private tokenService: TokenService\n  ) { }\n\n  async isAuth(@Headers('authorization') auth, @Ctx() ctx) {\n    const user = await this.tokenService.storeUserInCache(auth, ctx);\n    return !!user;\n  }\n\n  async hasRoles(@Headers('authorization') auth, @Ctx() ctx, @GuardParams() roles) {\n    try {\n      const user = await this.tokenService.storeUserInCache(auth, ctx);\n      if (!user?.role) return false;\n      const roleArray = Array.isArray(roles) ? roles : [roles];\n      return this.roleChecker.hasAnyRole(user.role, roleArray);\n    } catch {\n      return false;\n    }\n  }\n}\n\n// ============ EXPORTED GUARD DECORATORS ============ //\n\nexport const isAdmin = () => Role('admin');\nexport const isPrincipal = () => Role('principal');\nexport const isAccounting = () => Role('accounting');\nexport const isSecretary = () => Role('secretary');\nexport const isTeacher = () => Role('teacher');\nexport const isParent = () => Role('parent');\nexport const isStudent = () => Role('student');\nexport const isAdministrator = () => Role('admin', 'principal');\nexport const isFinancial = () => Role('admin', 'accounting');\nexport const isStaff = () => Role('admin', 'principal', 'accounting', 'secretary', 'teacher');\nexport const isAuth = createGuard(RoleGuards, 'isAuth');\n\nexport const Role = (...roles) => createGuard(RoleGuards, 'hasRoles')(...roles);\n",
    "\nimport { tokensTable, usersTable, rolesTable, permissionsTable, rolePermissionsTable } from '@/database/schema';\nimport { eq } from 'drizzle-orm';\nimport { Repository, DB } from 'najm-api';\n\n@Repository()\nexport class TokenRepository {\n  declare db: DB;\n\n  async storeRefreshToken(tokenData: { userId: string; token: string; expiresAt: string }) {\n    return await this.db\n      .insert(tokensTable)\n      .values(tokenData)\n      .onConflictDoUpdate({\n        target: tokensTable.userId,\n        set: {\n          token: tokenData.token,\n          expiresAt: tokenData.expiresAt,\n        }\n      }).returning();\n  }\n\n  async getRefreshToken(userId: string) {\n    const [token] = await this.db.select().from(tokensTable).where(eq(tokensTable.userId, userId));\n    return token?.token;\n  }\n\n  async revokeToken(userId: string) {\n    const [deletedToken] = await this.db.delete(tokensTable).where(eq(tokensTable.userId, userId)).returning();\n    return deletedToken;\n  }\n\n  async isUserExists(userId: string) {\n    const [user] = await this.db\n      .select({ id: usersTable.id })\n      .from(usersTable)\n      .where(eq(usersTable.id, userId))\n      .limit(1);\n    return !!user;\n  }\n\n  async getRoleNameById(userId: string) {\n    const [role] = await this.db.select({\n      roleName: rolesTable.name\n    })\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id))\n      .where(eq(usersTable.id, userId))\n      .limit(1);\n\n    return role?.roleName;\n  }\n\n  async getUserPermissions(userId: string) {\n    const [user] = await this.db\n      .select({ roleId: usersTable.roleId })\n      .from(usersTable)\n      .where(eq(usersTable.id, userId))\n      .limit(1);\n\n    if (!user || !user.roleId) return [];\n\n    const userPermissions = await this.db\n      .select({\n        name: permissionsTable.name,\n      })\n      .from(rolePermissionsTable)\n      .leftJoin(permissionsTable, eq(rolePermissionsTable.permissionId, permissionsTable.id))\n      .where(eq(rolePermissionsTable.roleId, user.roleId));\n\n    return userPermissions.map(p => p.name).filter(name => name);\n  }\n\n  async getUser(userId: string) {\n    const [user] = await this.db\n      .select({\n        id: usersTable.id,\n        email: usersTable.email,\n        status: usersTable.status,\n        roleId: usersTable.roleId,\n        roleName: rolesTable.name,\n        createdAt: usersTable.createdAt,\n        updatedAt: usersTable.updatedAt,\n      })\n      .from(usersTable)\n      .leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id))\n      .where(eq(usersTable.id, userId))\n      .limit(1);\n\n    return user ? { ...user, role: user.roleName } : null;\n  }\n}",
    "import { t } from 'najm-api';\nimport { getCookie, Injectable, Ctx, Headers } from 'najm-api';\nimport jwt from 'jsonwebtoken'\nimport { jwtDecode } from 'jwt-decode';\nimport { TokenRepository } from './TokenRepository';\nimport timestring from 'timestring';\n\ninterface JwtPayload {\n  userId: string;\n  exp?: number;\n  iat?: number;\n}\n\n@Injectable()\nexport class TokenService {\n\n  private accessSecretKey = process.env.JWT_ACCESS_SECRET;\n  private accessExpiresIn = process.env.ACCESS_EXPIRES_IN;\n  private refreshSecretKey = process.env.JWT_REFRESH_SECRET;\n  private refreshExpiresIn = process.env.REFRESH_EXPIRES_IN;\n\n  constructor(\n    private tokenRepository: TokenRepository\n  ) { }\n\n  //==== Validate tokens\n\n  extractAccessToken(authorization) {\n    if (authorization && authorization.startsWith('Bearer')) {\n      return authorization.split(' ')[1]\n    }\n    throw new Error(t('auth.errors.tokenMissing'))\n  }\n\n  verifyAccessToken(token: string): JwtPayload {\n    try {\n      return jwt.verify(token, this.accessSecretKey) as JwtPayload;\n    } catch (error) {\n      throw new Error(t('auth.errors.tokenVerificationFailed'))\n    }\n  }\n\n  verifyRefreshToken(token: string): string {\n    try {\n      const decoded = jwt.verify(token, this.refreshSecretKey) as JwtPayload;\n      return decoded.userId;\n    } catch (error) {\n      throw new Error(t('auth.errors.tokenVerificationFailed'))\n    }\n  }\n\n  async getUserIdByAccessToken(header: string): Promise<string> {\n    const token = this.extractAccessToken(header);\n    const decodedToken = this.verifyAccessToken(token);\n    const userId = decodedToken.userId;\n    const userExists = await this.tokenRepository.isUserExists(userId);\n\n    if (!userExists) {\n      throw new Error(t('users.errors.notFound'))\n    }\n\n    return userId;\n  }\n\n  //=== Generate tokens\n\n  async storeRefreshToken(userId, refreshToken) {\n    const expireInSecond = timestring(this.refreshExpiresIn, 's');\n    const tokenData = {\n      userId,\n      token: refreshToken,\n      expiresAt: new Date(Date.now() + expireInSecond * 1000).toISOString()\n    }\n    await this.tokenRepository.storeRefreshToken(tokenData);\n  }\n\n  getTokenExpire(token): number | undefined {\n    return jwtDecode(token).exp\n  }\n\n  generateAccessToken(data: { userId: string }): string {\n    const options = { expiresIn: this.accessExpiresIn } as any;\n    return jwt.sign(data, this.accessSecretKey, options);\n  }\n\n  generateRefreshToken(data: { userId: string }): string {\n    const options = { expiresIn: this.refreshExpiresIn } as any;\n    return jwt.sign(data, this.refreshSecretKey, options);\n  }\n\n  async generateTokens(userId) {\n    const tokenData = { userId }\n    const accessToken = await this.generateAccessToken(tokenData);\n    const refreshToken = await this.generateRefreshToken(tokenData);\n    const accessTokenExpiresAt = this.getTokenExpire(accessToken);\n    const refreshTokenExpiresAt = this.getTokenExpire(refreshToken);\n    await this.storeRefreshToken(userId, refreshToken);\n\n    return {\n      accessToken,\n      refreshToken,\n      accessTokenExpiresAt,\n      refreshTokenExpiresAt,\n    }\n  }\n\n  async refreshTokens() {\n    const newRefreshToken = getCookie('refreshToken');\n    const userId = this.verifyRefreshToken(newRefreshToken);\n    const userExists = await this.tokenRepository.isUserExists(userId);\n\n    if (!userExists) {\n      throw new Error(t('users.errors.notFound'));\n    }\n\n    const storedRefreshToken = await this.tokenRepository.getRefreshToken(userId)\n\n    if (newRefreshToken != storedRefreshToken) {\n      throw new Error(t('auth.errors.refreshTokenInvalid'))\n    }\n    return await this.generateTokens(userId)\n  }\n\n  async revokeToken(userId) {\n    return await this.tokenRepository.revokeToken(userId)\n  }\n\n  async getUserPermissions(auth) {\n    if (!auth) return;\n    const userId = await this.getUserIdByAccessToken(auth);\n    const permissions = await this.tokenRepository.getUserPermissions(userId);\n    return permissions;\n  }\n\n  async getUserRole(auth) {\n    if (!auth) return;\n    const userId = await this.getUserIdByAccessToken(auth);\n    const roleName = await this.tokenRepository.getRoleNameById(userId);\n    return roleName;\n  }\n\n  async getUser(auth) {\n    if (!auth) return;\n    const userId = await this.getUserIdByAccessToken(auth);\n    const user = await this.tokenRepository.getUser(userId);\n    if (!user) return null;\n    return user;\n  }\n\n  async storeUserInCache(auth, ctx) {\n    const cachedUser = ctx.get('user');\n    if (cachedUser) return cachedUser;\n    const user = await this.getUser(auth);\n    if (user) {\n      ctx.set('user', user);\n    }\n    return user;\n  }\n\n}",
    "import { Controller, Get, Post, Put, Delete, Params, Body,t, User } from 'najm-api';\nimport { RoleService } from './RoleService';\nimport { isAdmin } from './RoleGuards';\n\n@Controller('/roles')\nexport class RoleController {\n  constructor(\n    private roleService: RoleService,\n  ) { }\n\n  @Get()\n  @isAdmin()\n  async getRoles() {\n    const roles = await this.roleService.getAll();\n    return {\n      data: roles,\n      message: t('roles.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Get('/:id')\n  @isAdmin()\n  async getRole(@Params('id') id) {\n    const role = await this.roleService.getById(id);\n    return {\n      data: role,\n      message: t('roles.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post()\n  @isAdmin()\n  async createRole(@Body() body) {\n    const newRole = await this.roleService.create(body);\n    return {\n      data: newRole,\n      message: t('roles.success.created'),\n      status: 'success'\n    };\n  }\n\n  @Put('/:id')\n  @isAdmin()\n  async updateRole(@Params('id') id, @Body() body) {\n    const updatedRole = await this.roleService.update(id, body);\n    return {\n      data: updatedRole,\n      message: t('roles.success.updated'),\n      status: 'success'\n    };\n  }\n\n  @Delete('/:id')\n  @isAdmin()\n  async deleteRole(@Params('id') id) {\n    const result = await this.roleService.delete(id);\n    return {\n      data: result,\n      message: t('roles.success.deleted'),\n      status: 'success'\n    };\n  }\n}",
    "\nimport { Injectable } from 'najm-api';\nimport { RoleRepository } from './RoleRepository';\nimport { RoleValidator } from './RoleValidator';\n\n@Injectable()\nexport class RoleService {\n  constructor(\n    private roleRepository: RoleRepository,\n    private roleValidator: RoleValidator\n  ) { }\n\n  async getAll() {\n    return await this.roleRepository.getAll();\n  }\n\n  async getById(id) {\n    await this.roleValidator.checkRoleExists(id);\n    return await this.roleRepository.getById(id);\n  }\n\n  async getByName(name) {\n    return await this.roleRepository.getByName(name);\n  }\n\n  async create(data) {\n    await this.roleValidator.validateCreateRole(data);\n    await this.roleValidator.checkNameUnique(data.name);\n    return await this.roleRepository.create(data);\n  }\n\n  async update(id, data) {\n    await this.roleValidator.checkRoleExists(id);\n    await this.roleValidator.checkNameUnique(data.name, id);\n    return await this.roleRepository.update(id, data);\n  }\n\n  async delete(id) {\n    await this.roleValidator.checkRoleExists(id);\n    return await this.roleRepository.delete(id);\n  }\n\n  async seedDefaultRoles(defaultRoles) {\n\n    const rolesToCreate = [];\n\n    for (const role of defaultRoles) {\n      const exists = await this.roleValidator.isRoleNameExists(role.name);\n      if (!exists) {\n        rolesToCreate.push(role);\n      }\n    }\n\n    const createdRoles = await Promise.all(\n      rolesToCreate.map(role => this.roleRepository.create(role))\n    );\n\n    return createdRoles;\n  }\n\n  async getRoleIdByName(name) {\n    const teacherRole = await this.getByName(name);\n    return teacherRole?.id;\n  }\n\n\n}",
    "import { Controller, Get, Post, Put, Delete, Params, Body, t } from 'najm-api';\nimport { UserService } from './UserService';\n\nimport { isAdmin, isAuth } from '@/roles/RoleGuards';\n\n@Controller('/users')\nexport class UserController {\n  constructor(private userService: UserService) { }\n\n  @Get()\n  @isAdmin() \n  async getUsers() {\n    const users = await this.userService.getAll();\n    return {\n      data: users,\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Get('/lang')\n  @isAuth() \n  async getLang() {\n    const language = await this.userService.getLang();\n    return {\n      data: { language },\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post('/lang/:language')\n  @isAuth() \n  async updateLang(@Params('language') language) {\n    const data = await this.userService.updateLang(language);\n    return {\n      data,\n      message: t('users.success.updated'),\n      status: 'success'\n    };\n  }\n\n  @Get('/:id')\n  @isAdmin() \n  async getUser(@Params('id') id) {\n    const user = await this.userService.getById(id);\n    return {\n      data: user,\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Get('/email/:email')\n  @isAdmin()\n  async getByEmail(@Params('email') email) {\n    const user = await this.userService.getByEmail(email);\n    return {\n      data: user,\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n\n  @Get('/role/:userId')\n  @isAdmin() \n  async getRole(@Params('userId') userId) {\n    const role = await this.userService.getRoleName(userId);\n\n    return {\n      data: role,\n      message: t('users.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post()\n  @isAdmin() \n  async create(@Body() body) {\n    const newUser = await this.userService.create(body);\n    return {\n      data: newUser,\n      message: t('users.success.created'),\n      status: 'success'\n    };\n  }\n\n  @Put('/:id')\n  @isAdmin() \n  async update(@Params('id') id, @Body() body) {\n    const updatedUser = await this.userService.update(id, body);\n    return {\n      data: updatedUser,\n      message: t('users.success.updated'),\n      status: 'success'\n    };\n  }\n\n  @Delete('/:id')\n  @isAdmin() \n  async delete(@Params('id') id) {\n    const result = await this.userService.delete(id);\n    return {\n      data: result,\n      message: t('users.success.deleted'),\n      status: 'success'\n    };\n  }\n\n  @Delete()\n  @isAdmin() \n  async deleteAll() {\n    const result = await this.userService.deleteAll();\n    return {\n      data: result,\n      message: t('users.success.allDeleted'),\n      status: 'success'\n    };\n  }\n\n  @Post('/assign/:userId/:roleId')\n  @isAdmin() \n  async assignRole(@Params('userId') userId, @Params('roleId') roleId) {\n    await this.userService.assignRole(userId, roleId);\n\n    return {\n      message: t('users.success.updated'),\n      status: 'success'\n    };\n  }\n\n  @Delete('/remove/:userId')\n  @isAdmin() \n  async removeRole(@Params('userId') userId) {\n    await this.userService.removeRole(userId);\n\n    return {\n      message: t('users.success.updated'),\n      status: 'success'\n    };\n  }\n\n}",
    "export const ENUMS = {\r\n  // Core System\r\n  userType: {\r\n    values: ['admin', 'teacher', 'student', 'parent'],\r\n    translationKey: 'enums.userType'\r\n  },\r\n  userStatus: {\r\n    values: ['active', 'inactive', 'pending'],\r\n    translationKey: 'enums.userStatus'\r\n  },\r\n  tokenStatus: {\r\n    values: ['active', 'revoked', 'expired'],\r\n    translationKey: 'enums.tokenStatus'\r\n  },\r\n  tokenType: {\r\n    values: ['access', 'refresh'],\r\n    translationKey: 'enums.tokenType'\r\n  },\r\n  fileStatus: {\r\n    values: ['active', 'deleted', 'archived'],\r\n    translationKey: 'enums.fileStatus'\r\n  },\r\n\r\n  // Educational System\r\n  gender: {\r\n    values: ['M', 'F'],\r\n    translationKey: 'common.gender'\r\n  },\r\n  studentStatus: {\r\n    values: ['active', 'inactive', 'graduated', 'transferred'],\r\n    translationKey: 'students.status'\r\n  },\r\n  teacherStatus: {\r\n    values: ['active', 'inactive', 'onLeave'],\r\n    translationKey: 'teachers.status',\r\n  },\r\n  employmentType: {\r\n    values: ['fullTime', 'partTime', 'contract', 'temporary'],\r\n    translationKey: 'teachers.employmentType',\r\n  },\r\n  relationshipType: {\r\n    values: ['father', 'mother', 'guardian', 'stepparent', 'grandparent', 'other'],\r\n    translationKey: 'parents.relationships'\r\n  },\r\n  semester: {\r\n    values: ['spring', 'summer', 'fall', 'winter'],\r\n    translationKey: 'academic.semester'\r\n  },\r\n  classStatus: {\r\n    values: ['active', 'completed', 'cancelled'],\r\n    translationKey: 'classes.status'\r\n  },\r\n  sectionStatus: {\r\n    values: ['active', 'inactive', 'archived'],\r\n    translationKey: 'sections.status'\r\n  },\r\n  language: {\r\n    values: ['en', 'fr', 'ar', 'es'],\r\n    translationKey: 'common.languages'\r\n  },\r\n  enrollmentStatus: {\r\n    values: ['enrolled', 'completed', 'dropped', 'failed'],\r\n    translationKey: 'enrollments.status'\r\n  },\r\n  assignmentStatus: {\r\n    values: ['active', 'completed', 'cancelled'],\r\n    translationKey: 'assignments.status'\r\n  },\r\n\r\n  calendarSystem: {\r\n    values: ['SEMESTER', 'TRIMESTER', 'QUARTER'],\r\n    translationKey: 'settings.calendarSystem'\r\n  },\r\n\r\n  // Assessment & Exams\r\n  assessmentType: {\r\n    values: ['quiz', 'assignment', 'project', 'participation', 'test', 'presentation'],\r\n    translationKey: 'assessments.type'\r\n  },\r\n  assessmentStatus: {\r\n    values: ['scheduled', 'active', 'completed', 'cancelled'],\r\n    translationKey: 'assessments.status'\r\n  },\r\n  submissionType: {\r\n    values: ['online', 'paper', 'presentation', 'practical', 'discussion'],\r\n    translationKey: 'assessments.submissionType'\r\n  },\r\n  examType: {\r\n    values: ['midterm', 'final', 'standardized'],\r\n    translationKey: 'exams.type'\r\n  },\r\n  examSecurity: {\r\n    values: ['low', 'medium', 'high'],\r\n    translationKey: 'exams.security'\r\n  },\r\n  examStatus: {\r\n    values: ['scheduled', 'active', 'completed', 'cancelled', 'rescheduled'],\r\n    translationKey: 'exams.status'\r\n  },\r\n  gradeStatus: {\r\n    values: ['graded', 'pending', 'draft', 'reviewed'],\r\n    translationKey: 'grades.status'\r\n  },\r\n  attendanceStatus: {\r\n    values: ['present', 'absent', 'late', 'excused'],\r\n    translationKey: 'attendance.status'\r\n  },\r\n  proficiencyLevel: {\r\n    values: ['beginner', 'intermediate', 'advanced', 'expert'],\r\n    translationKey: 'common.proficiencyLevel'\r\n  },\r\n  dayOfWeek: {\r\n    values: ['monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday', 'sunday'],\r\n    translationKey: 'common.days'\r\n  },\r\n\r\n  // Alerts\r\n  alertType: {\r\n    values: ['academic', 'attendance', 'behavioral', 'health', 'system', 'announcement', 'reminder', 'emergency'],\r\n    translationKey: 'alerts.type'\r\n  },\r\n  alertPriority: {\r\n    values: ['low', 'medium', 'high', 'critical'],\r\n    translationKey: 'alerts.priority'\r\n  },\r\n  alertStatus: {\r\n    values: ['active', 'acknowledged', 'resolved', 'dismissed'],\r\n    translationKey: 'alerts.status'\r\n  },\r\n\r\n  // Fees & Payments\r\n  feeTypeStatus: {\r\n    values: ['active', 'inactive', 'archived'],\r\n    translationKey: 'fees.typeStatus'\r\n  },\r\n  feeCategory: {\r\n    values: ['tuition', 'registration', 'transport', 'cafeteria', 'books', 'sports', 'uniform', 'technology', 'fieldtrip', 'other'],\r\n    translationKey: 'feeTypes.category'\r\n  },\r\n  paymentType: {\r\n    values: ['recurring', 'oneTime'],\r\n    translationKey: 'payments.type',\r\n  },\r\n  schedule: {\r\n    values: ['monthly', 'quarterly', 'semester', 'annually', 'oneTime'],\r\n    translationKey: 'fees.schedule',\r\n  },\r\n  feeStatus: {\r\n    values: ['pending', 'partiallyPaid', 'paid', 'overdue'],\r\n    translationKey: 'fees.status',\r\n  },\r\n  feeInstallmentStatus: {\r\n    values: ['pending', 'partiallyPaid', 'paid', 'overdue'],\r\n    translationKey: 'fees.installmentStatus',\r\n  },\r\n  paymentMethod: {\r\n    values: ['cash', 'bankTransfer', 'check', 'creditCard', 'debitCard', 'online', 'mobilePayment'],\r\n    translationKey: 'payments.methods',\r\n  },\r\n  \r\n  paymentStatus: {\r\n    values: ['completed', 'pending', 'failed', 'refunded'],\r\n    translationKey: 'payments.status'\r\n  },\r\n\r\n  // Events\r\n  eventType: {\r\n    values: ['academic', 'sports', 'cultural', 'holiday', 'exam', 'meeting', 'workshop', 'fieldtrip', 'ceremony', 'conference', 'other'],\r\n    translationKey: 'events.type'\r\n  },\r\n  eventStatus: {\r\n    values: ['scheduled', 'ongoing', 'completed', 'cancelled', 'postponed'],\r\n    translationKey: 'events.status'\r\n  },\r\n  eventVisibility: {\r\n    values: ['public', 'private', 'teachers', 'students', 'parents', 'staff'],\r\n    translationKey: 'events.visibility'\r\n  },\r\n  participantType: {\r\n    values: ['student', 'teacher', 'parent', 'staff'],\r\n    translationKey: 'events.participantType'\r\n  },\r\n\r\n  // Expenses\r\n  expenseCategory: {\r\n    values: ['salary', 'utilities', 'maintenance', 'supplies', 'equipment', 'transport', 'food', 'security', 'cleaning', 'insurance', 'rent', 'tax', 'marketing', 'training', 'technology', 'miscellaneous'],\r\n    translationKey: 'expenses.categories'\r\n  },\r\n  expenseStatus: {\r\n    values: ['pending', 'approved', 'paid', 'rejected', 'cancelled'],\r\n    translationKey: 'expenses.status'\r\n  },\r\n\r\n  // Tracker\r\n  trackerMode: {\r\n    values: ['tracking', 'gprs', 'sms', 'sleepTime', 'sleepShock', 'sleepDeep'],\r\n    translationKey: 'tracker.mode',\r\n  },\r\n\r\n  // Transport\r\n  driverStatus: {\r\n    values: ['active', 'inactive', 'onLeave', 'suspended'],\r\n    translationKey: 'transport.driverStatus',\r\n  },\r\n  vehicleStatus: {\r\n    values: ['active', 'inactive', 'maintenance', 'retired'],\r\n    translationKey: 'transport.vehicleStatus'\r\n  },\r\n  vehicleType: {\r\n    values: ['sedan', 'minibus', 'fullbus', 'shuttle'],\r\n    translationKey: 'transport.vehicleType'\r\n  },\r\n  vehicleDocumentType: {\r\n    values: ['insurance', 'registration', 'inspection', 'emission', 'license'],\r\n    translationKey: 'transport.documentType'\r\n  },\r\n  busStatus: {\r\n    values: ['active', 'inactive', 'maintenance', 'retired'],\r\n    translationKey: 'transport.busStatus'\r\n  },\r\n  refuelStatus: {\r\n    values: ['pending', 'completed', 'cancelled'],\r\n    translationKey: 'transport.refuelStatus'\r\n  },\r\n  fuelType: {\r\n    values: ['gasoline', 'diesel', 'electric', 'hybrid', 'lpg', 'cng'],\r\n    translationKey: 'transport.fuelType'\r\n  },\r\n  maintenanceType: {\r\n    values: ['scheduled', 'repair', 'inspection', 'oilChange', 'filterChange', 'other'],\r\n    translationKey: 'transport.maintenanceType',\r\n  },\r\n  maintenanceStatus: {\r\n    values: ['scheduled', 'inProgress', 'completed', 'cancelled', 'overdue'],\r\n    translationKey: 'transport.maintenanceStatus',\r\n  },\r\n\r\n  // Personal\r\n  maritalStatus: {\r\n    values: ['single', 'married', 'divorced', 'widowed', 'separated'],\r\n    translationKey: 'parents.maritalStatus'\r\n  }\r\n}\r\n\r\n// Helper functions\r\nexport const getEnumConfig = (enumKey) => ENUMS[enumKey]\r\nexport const getEnumValues = (enumKey) => ENUMS[enumKey]?.values || []",
    "import { z } from 'zod';\nimport { alertPriorityEnum, alertStatusEnum, alertTypeEnum, assessmentStatusEnum, assessmentTypeEnum, attendanceStatusEnum, calendarSystemEnum, driverStatusEnum, employmentTypeEnum, eventStatusEnum, eventTypeEnum, eventVisibilityEnum, examStatusEnum, examTypeEnum, expenseCategoryEnum, expenseStatusEnum, feeStatusEnum, feeTypeStatusEnum, fuelTypeEnum, genderEnum, gradeStatusEnum, languageEnum, participantTypeEnum, paymentMethodEnum, feeInstallmentStatusEnum, paymentStatusEnum, paymentTypeEnum, refuelStatusEnum, relationshipTypeEnum, scheduleEnum, sectionStatusEnum, studentStatusEnum, teacherStatusEnum, userStatusEnum, vehicleStatusEnum, vehicleTypeEnum } from './ZodEnum';\n\nconst requiredId = z.preprocess((val) => val ?? \"\", z.string().min(1, \"ID is required\"));\nconst optionalId = z.string().min(1, \"ID cannot be empty\").nullish().optional();\nconst emailField = z.string().email('Invalid email format').or(z.literal(\"\"));\nconst phoneField = z.string().regex(/^[\\+]?[1-9][\\d]{0,15}$/, 'Invalid phone number');\nconst nameField = z.string().min(2, 'Name must be at least 2 characters').max(100, 'Name too long');\nconst dateField = z.string().regex(/^(\\d{4}-\\d{2}-\\d{2}|\\d{2}\\/\\d{2}\\/\\d{4}|\\d{2}-\\d{2}-\\d{2}|\\d{2}-\\d{2}-\\d{4})$/, 'Date must be in YYYY-MM-DD, MM/DD/YYYY, DD/MM/YYYY, DD-MM-YY, or DD-MM-YYYY format');\nconst optionalDateField = z.string().regex(/^\\d{4}-\\d{2}-\\d{2}$/, 'Date must be in YYYY-MM-DD format').nullable().optional();\nconst timeField = z.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, 'Time must be in HH:MM format').optional().nullable();\nconst cinField = z.string().min(8, 'CIN must be at least 8 characters').max(20, 'CIN too long');\nconst addressField = z.string().max(500, 'Address too long').optional();\nconst academicYearField = z.string().min(9, 'Academic year is required').regex(/^\\d{4}-\\d{4}$/, 'Academic year must be in YYYY-YYYY format');\n\n\nconst num = () => {\n  const createChainable = (currentSchema: z.ZodTypeAny): any => {\n    const methods = {\n      positive: (msg = 'Must be positive') =>\n        createChainable(currentSchema.refine((val: number) => val > 0, { message: msg })),\n\n      min: (value: number, msg?: string) =>\n        createChainable(currentSchema.refine(\n          (val: number) => val >= value,\n          { message: msg || `Must be at least ${value}` }\n        )),\n\n      max: (value: number, msg?: string) =>\n        createChainable(currentSchema.refine(\n          (val: number) => val <= value,\n          { message: msg || `Cannot exceed ${value}` }\n        )),\n\n      int: (msg = 'Must be an integer') =>\n        createChainable(currentSchema.refine(\n          (val: number) => Number.isInteger(val),\n          { message: msg }\n        ))\n    };\n\n    return Object.assign(currentSchema, methods);\n  };\n\n  const isValidNumber = (val: any) => {\n    if (val === null || val === undefined || Number.isNaN(val)) return false;\n    if (typeof val === 'number') return true;\n    if (typeof val === 'string') {\n      const trimmed = val.trim();\n      return trimmed !== '' && !isNaN(Number(trimmed));\n    }\n    return false;\n  };\n\n  const baseSchema = z.any()\n    .refine(isValidNumber, { message: 'Must be a valid number' })\n    .transform((val) => typeof val === 'string' ? Number(val) : val);\n\n  return createChainable(baseSchema);\n};\n//=====================================================================//\n// USER-ROLE VALIDATION SCHEMA\n//=====================================================================//\n\nexport const userSchema = z.object({\n  id: optionalId,\n  username: nameField.max(50).optional(),\n  email: emailField,\n  password: z.string().min(8, \"Password must be at least 8 characters\"),\n  roleId: optionalId,\n  roleName: nameField.max(50).optional(),\n  lastLogin: optionalDateField,\n  image: z.union([z.string(), z.instanceof(File), z.undefined()]).optional(),\n  emailVerified: z.boolean().default(false),\n  status: userStatusEnum,\n  createdAt: optionalDateField,\n});\n\nexport const roleSchema = z.object({\n  id: optionalId,\n  name: nameField.max(50),\n  description: z.string().max(255, \"Description too long\").optional(),\n  createdAt: optionalDateField,\n});\n\n//=====================================================================//\n// PEOPLE VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const studentSchema = z.object({\n  id: optionalId,\n  classId: requiredId,\n  sectionId: requiredId,\n  studentCode: z.string(),\n  name: nameField,\n  email: emailField,\n  phone: phoneField.nullish(),\n  address: addressField,\n  dateOfBirth: optionalDateField,\n  gender: genderEnum,\n  enrollmentDate: dateField,\n  medicalConditions: z.string().max(1000, 'Medical conditions description too long').nullish().optional(),\n  previousSchool: z.string().max(500, 'Previous school name too long').optional().nullable(),\n  image: z.union([z.string(), z.instanceof(File), z.null()]).optional(),\n  status: studentStatusEnum.default('active'),\n});\n\nexport const parentSchema = z.object({\n  id: optionalId,\n  name: nameField,\n  email: emailField.optional(),\n  phone: phoneField,\n  gender: genderEnum.optional(),\n  address: addressField,\n  dateOfBirth: optionalDateField,\n  cin: cinField,\n  occupation: z.string().max(100, 'Occupation too long').optional(),\n  nationality: z.string().max(100, 'Nationality too long').optional(),\n  maritalStatus: z.string().max(50, 'Marital status too long').optional(),\n  relationshipType: relationshipTypeEnum,\n  image: z.union([z.string(), z.instanceof(File), z.null()]).optional(),\n  isEmergencyContact: z.boolean().optional().default(false),\n  financialResponsibility: z.boolean().optional().default(false),\n});\n\nexport const driverSchema = z.object({\n  id: optionalId,\n  name: nameField,\n  email: emailField,\n  cin: cinField,\n  phone: phoneField,\n  address: addressField,\n  gender: genderEnum.optional(),\n  licenseNumber: z.string().min(5, 'License number must be at least 5 characters').max(20, 'License number too long'),\n  licenseType: z.string().max(10, 'License type too long'),\n  licenseExpiry: dateField,\n  hireDate: dateField,\n  salary: num().positive('Salary must be positive').optional(),\n  yearsOfExperience: num().int().min(0, 'Years of experience must be non-negative').optional(),\n  emergencyContact: nameField.optional(),\n  emergencyPhone: phoneField.optional(),\n  image: z.union([z.string(), z.instanceof(File), z.null()]).optional(),\n  status: driverStatusEnum.default('active'),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n});\n\nexport const teacherPersonalSchema = z.object({\n  id: optionalId,\n  name: nameField,\n  cin: cinField,\n  email: emailField,\n  phone: phoneField,\n  address: addressField,\n  gender: genderEnum.optional(),\n  emergencyContact: nameField.optional(),\n  emergencyPhone: phoneField,\n  status: teacherStatusEnum.default('active'),\n  image: z.union([z.string(), z.instanceof(File), z.null()]).optional(),\n});\n\nexport const teacherProfessionalSchema = z.object({\n  specialization: z.string().max(100, 'Specialization too long').optional(),\n  yearsOfExperience: num().int().min(0, 'Years of experience must be non-negative').optional(),\n  salary: num().positive('Salary must be positive').optional(),\n  hireDate: dateField,\nbankAccount: z.coerce\n  .string()\n  .max(100, { message: 'Bank account too long' })\n  .optional(),\n  employmentType: employmentTypeEnum.optional(),\n    workloadHours: num().int().min(0, 'Workload hours must be non-negative').max(60, 'Workload hours cannot exceed 60').optional(),\n    academicDegrees: z.string().max(500, 'Academic degrees description too long').optional(),\n});\n\nexport const assignmentSchema = z.object({\n  classId: z.string().min(1, 'Class is required'),\n  sectionIds: z.array(z.string()).min(1, 'At least one section is required'),\n  subjectIds: z.array(z.string()).min(1, 'At least one subject is required'),\n  academicYear: z.string().optional()\n});\n\nexport const assignmentsSchema = z.object({\n  assignments: z.array(assignmentSchema).min(1, 'At least one parent is required')\n})\n\nexport const teacherFullSchema = z.object({\n  ...teacherPersonalSchema.shape,\n  ...teacherProfessionalSchema.shape,\n  ...assignmentsSchema.shape,\n});\n\n//=====================================================================//\n// FEES VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const feeTypeSchema = z.object({\n  id: optionalId,\n  name: z.string().min(2, 'Fee type name must be at least 2 characters').max(100, 'Fee type name too long'),\n  description: z.string().max(500, 'Description too long').optional().nullable(),\n  category: z.string(),\n  amount: num().positive('Amount must be greater than 0').max(1_000_00, 'Amount too large'),\n  paymentType: paymentTypeEnum.default('recurring'),\n  status: feeTypeStatusEnum.default('active').optional(),\n});\n\nexport const feeSchema = z.object({\n  id: optionalId,\n  studentId: requiredId,\n  feeTypeId: requiredId,\n  academicYear: academicYearField.optional(),\n  status: feeStatusEnum.optional(),\n  schedule: scheduleEnum,\n  baseAmount: num().positive('Base amount must be positive').optional(),\n  grossAmount: num().positive('Gross amount must be positive').optional(),\n  netAmount: num().optional(),\n  paidAmount: num().min(0, 'Paid amount cannot be negative').optional(),\n  discountAmount: num().min(0, 'Discount cannot be negative').optional(),\n  discountReason: z.string().max(500, 'Discount reason too long').optional().nullable(),\n  assignedBy: optionalId.nullable(),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n})\n\nexport const bulkFeeItemSchema = feeSchema.omit({ studentId: true })\n\nexport const bulkFeeFormSchema = z.object({\n  studentId: requiredId,\n  fees: z.array(bulkFeeItemSchema).min(1, 'At least one fee is required')\n});\n\nexport const feeInstallmentSchema = z.object({\n  feeId: optionalId,\n  number: num().int().min(1, 'Installment must be at least 1'),\n  dueDate: dateField,\n  amount: num().positive('Amount must be greater than 0').max(1_000_00, 'Amount too large'),\n  paidAmount: num().min(0, 'Paid amount cannot be negative').max(1_000_00, 'Amount too large').optional(),\n  status: feeInstallmentStatusEnum.optional(),\n});\n\nexport const feePaymentSchema = z.object({\n  studentId: optionalId,\n  amount: num().positive('Amount must be greater than 0').max(1_000_00, 'Amount too large').optional(),\n  paymentMethod: paymentMethodEnum,\n  paymentDate: dateField,\n  checkNumber: z.preprocess((val) => val === \"\" ? null : val, z.string().max(50, 'Check number too long').optional().nullable()),\n  checkDueDate: z.preprocess((val) => val === \"\" ? null : val, optionalDateField.nullable()),\n  transactionRef: z.preprocess((val) => val === \"\" ? null : val, z.string().max(100, 'Transaction reference too long').optional().nullable()),\n  receiptNumber: z.preprocess((val) => val === \"\" ? null : val, z.string().max(50, 'Receipt number too long').optional().nullable()),\n  status: paymentStatusEnum.default('completed'),\n  processedBy: optionalId,\n  notes: z.preprocess((val) => val === \"\" ? null : val, z.string().max(1000, 'Notes too long').optional().nullable()),\n  allocations: z.array(z.object({\n    feeId: optionalId,\n    number: z.number().int().positive('Installment must be a positive number'),\n    amount: num().positive('Amount must be greater than 0')\n  })).optional().nullable(),\n})\n\nexport const paymentAllocationSchema = z.object({\n  paymentId: z.string().min(1, 'Payment ID is required'),\n  feeId: z.string().min(1, 'Fee ID is required'),\n  installmentId: z.string().optional(),\n  amount: z.number().positive('Amount must be positive'),\n  type: z.enum(['fee', 'installment']).default('installment'),\n  notes: z.string().optional(),\n});\n\nexport const parentsSchema = z.object({\n  parents: z.array(parentSchema).optional().default([])\n})\n\nexport const feesSchema = z.object({\n  fees: z.array(bulkFeeItemSchema).min(1, 'At least one fee is required')\n})\n\nexport const fullStudentSchema = z.object({\n  ...studentSchema.shape,\n  ...parentsSchema.shape,\n  ...feesSchema.shape\n});\n\n//=====================================================================//\n// SCHOOL STRUCTURE VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const subjectSchema = z.object({\n  id: optionalId,\n  code: z.string().min(2, 'Subject code must be at least 2 characters').max(10, 'Subject code too long'),\n  name: z.string().min(2, 'Subject name must be at least 2 characters').max(100, 'Subject name too long'),\n  description: z.string().max(500, 'Description too long').optional(),\n  gradeLevel: num().int().min(1).max(12).optional(),\n});\n\nexport const sectionSchema = z.object({\n  id: optionalId,\n  classId: requiredId,\n  name: z.string().min(1, 'Section name is required').max(10, 'Section name too long'),\n  maxStudents: num().int().min(1, 'Max students must be at least 1').max(100, 'Max students cannot exceed 100').default(30),\n  roomNumber: num().max(10000, 'Room number too long').optional(),\n  status: sectionStatusEnum.default('active'),\n});\n\nexport const classSchema = z.object({\n  id: optionalId,\n  name: z.string().min(1, 'Class name is required').max(50, 'Class name too long'),\n  description: z.string().max(500, 'Description too long').optional(),\n  academicYear: academicYearField,\n  level: z.string().min(1, 'Class level is required'),\n});\n\nexport const attendanceSchema = z.object({\n  studentId: requiredId,\n  teacherId: requiredId,\n  subjectId: requiredId,\n  sectionId: requiredId,\n  date: dateField,\n  status: attendanceStatusEnum.default('present'),\n  notes: z.string().max(500, 'Notes too long').optional(),\n})\n\nexport const assessmentSchema = z.object({\n  classId: requiredId,\n  sectionId: requiredId,\n  subjectId: requiredId,\n  teacherId: requiredId,\n  teacherAssignmentId: optionalId,\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  description: z.string().max(1000, 'Description too long').optional().nullable(),\n  type: assessmentTypeEnum.default('quiz'),\n  date: dateField,\n  duration: num().int().min(1, 'Duration must be at least 1 minute').max(480, 'Duration cannot exceed 8 hours'),\n  totalMarks: num().positive('Total marks must be greater than 0').max(1000, 'Total marks cannot exceed 1000'),\n  passingMarks: num().min(0, 'Passing marks must be non-negative').max(1000, 'Passing marks cannot exceed 1000'),\n  instructions: z.string().max(2000, 'Instructions too long').optional().nullable(),\n  status: assessmentStatusEnum.default('scheduled'),\n  assessmentId: optionalId,\n})\n\nexport const bulkAssessmentSchema = z.object({\n  assessments: z.array(assessmentSchema)\n    .min(1, 'At least one assessment is required')\n    .max(50, 'Cannot create more than 50 assessments at once'),\n})\n\nexport const gradeSchema = z.object({\n  assessmentId: requiredId,\n  teacherId: requiredId,\n  subjectId: requiredId,\n  sectionId: requiredId,\n  studentId: requiredId,\n  gradeId: requiredId,\n  assessmentTitle: z.string().min(3, 'Assessment title must be at least 3 characters').max(200, 'Assessment title too long').optional(),\n  marksObtained: num().min(0, 'Marks obtained must be non-negative').max(1000, 'Marks obtained cannot exceed 1000'),\n  feedback: z.string().max(1000, 'Feedback too long').optional().nullable(),\n  status: gradeStatusEnum.default('graded'),\n})\n\nexport const examSchema = z.object({\n  classId: optionalId,\n  sectionId: requiredId,\n  subjectId: requiredId,\n  teacherId: requiredId,\n  examId: requiredId,\n  teacherAssignmentId: requiredId,\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  description: z.string().max(1000, 'Description too long').optional().nullable(),\n  type: examTypeEnum.default('midterm'),\n  date: dateField,\n  startTime: timeField,\n  endTime: timeField,\n  duration: num().int().min(30, 'Exam duration must be at least 30 minutes').max(480, 'Duration cannot exceed 8 hours'),\n  totalMarks: num().positive('Total marks must be greater than 0').max(1000, 'Total marks cannot exceed 1000'),\n  passingMarks: num().min(0, 'Passing marks must be non-negative').max(1000, 'Passing marks cannot exceed 1000'),\n  roomNumber: num().max(50, 'Room number too long').optional().nullable(),\n  allowedMaterials: z.string().max(500, 'Allowed materials description too long').optional().nullable(),\n  instructions: z.string().max(2000, 'Instructions too long').optional().nullable(),\n  status: examStatusEnum.default('scheduled'),\n})\n\n//=====================================================================//\n// ANNOUNCEMENT VALIDATION SCHEMA\n//=====================================================================//\n\nexport const announcementSchema = z.object({\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  content: z.string().min(10, 'Content must be at least 10 characters').max(5000, 'Content too long'),\n  authorId: optionalId,\n  targetAudience: z.enum(['all', 'students', 'teachers', 'parents', 'class']),\n  classId: optionalId,\n  isPublished: z.boolean().default(false),\n  publishDate: z.string().datetime('Invalid publish date').optional(),\n  expiryDate: z.string().datetime('Invalid expiry date').optional(),\n})\n\n//=====================================================================//\n// ALERT VALIDATION SCHEMA\n//=====================================================================//\n\nexport const alertSchema = z.object({\n  type: alertTypeEnum,\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  message: z.string().min(10, 'Message must be at least 10 characters').max(2000, 'Message too long'),\n  priority: alertPriorityEnum.default('medium'),\n  status: alertStatusEnum.default('active'),\n  studentId: optionalId,\n  teacherId: optionalId,\n  classId: optionalId,\n  subjectId: optionalId,\n  targetAudience: z.enum(['all', 'students', 'teachers', 'parents']).optional(),\n  authorId: optionalId,\n  isRead: z.boolean().default(false),\n});\n\n//=====================================================================//\n// EVENTS VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const eventSchema = z.object({\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  description: z.string().max(5000, 'Description too long').optional().nullable(),\n  type: eventTypeEnum,\n  startDate: dateField,\n  endDate: dateField,\n  startTime: timeField,\n  endTime: timeField,\n  location: z.string().max(200, 'Location too long').optional().nullable(),\n  venue: z.string().max(200, 'Venue too long').optional().nullable(),\n  organizerId: optionalId,\n  classId: optionalId.nullable(),\n  sectionId: optionalId.nullable(),\n  visibility: eventVisibilityEnum.default('public'),\n  status: eventStatusEnum.default('scheduled'),\n  capacity: num().int().positive('Capacity must be positive').optional().nullable(),\n  registrationRequired: z.boolean().default(false),\n  registrationDeadline: optionalDateField.nullable(),\n  attachments: z.any().optional().nullable(),\n  notes: z.string().max(2000, 'Notes too long').optional().nullable(),\n})\n\nexport const eventParticipantSchema = z.object({\n  eventId: optionalId,\n  participantId: optionalId,\n  participantType: participantTypeEnum,\n  attendanceStatus: attendanceStatusEnum.optional().nullable(),\n  notes: z.string().max(500, 'Notes too long').optional().nullable(),\n});\n\n//=====================================================================//\n// EXPENSES VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const expenseSchema = z.object({\n  id: optionalId,\n  category: expenseCategoryEnum,\n  title: z.string().min(3, 'Title must be at least 3 characters').max(200, 'Title too long'),\n  amount: num().positive('Amount must be greater than 0').max(10000000, 'Amount too large'),\n  expenseDate: dateField,\n  paymentMethod: paymentMethodEnum.optional().nullable(),\n  paymentDate: optionalDateField.nullable(),\n  vendor: z.string().max(200, 'Vendor name too long').optional().nullable(),\n  invoiceNumber: z.string().max(100, 'Invoice number too long').optional().nullable(),\n  receiptNumber: z.string().max(100, 'Receipt number too long').optional().nullable(),\n  checkNumber: z.string().max(50, 'Check number too long').optional().nullable(),\n  transactionRef: z.string().max(100, 'Transaction reference too long').optional().nullable(),\n  status: expenseStatusEnum.default('pending'),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n});\n\nexport const expenseApprovalSchema = z.object({\n  action: z.enum(['approve', 'reject']),\n  rejectionReason: z.string().min(10, 'Rejection reason must be at least 10 characters').max(1000, 'Rejection reason too long').optional().nullable(),\n})\n\nexport const expensePaymentSchema = z.object({\n  paymentMethod: paymentMethodEnum,\n  paymentDate: dateField,\n  checkNumber: z.string().max(50, 'Check number too long').optional().nullable(),\n  transactionRef: z.string().max(100, 'Transaction reference too long').optional().nullable(),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n})\n\n//=====================================================================//\n// TRANSPORT VALIDATION SCHEMAS\n//=====================================================================//\n\nexport const vehicleSchema = z.object({\n  id: optionalId,\n  name: z.string().min(2, 'Vehicle name must be at least 2 characters').max(100, 'Vehicle name too long'),\n  brand: z.string().min(2, 'Brand must be at least 2 characters').max(100, 'Brand too long'),\n  model: z.string().min(2, 'Model must be at least 2 characters').max(100, 'Model too long'),\n  year: num().int().min(1900, 'Year must be after 1900').max(new Date().getFullYear() + 1, 'Year cannot be in future'),\n  type: vehicleTypeEnum.default('fullbus'),\n  capacity: num().int().min(1, 'Capacity must be at least 1').max(200, 'Capacity cannot exceed 200'),\n  licensePlate: z.string().min(2, 'License plate must be at least 2 characters').max(50, 'License plate too long'),\n  driverId: optionalId.nullable(),\n  image: z.string().max(500, 'Image path too long').optional().nullable().default('novehicle.png'),\n  purchaseDate: optionalDateField.nullable(),\n  purchasePrice: num().min(0, 'Purchase price must be non-negative').max(10000000, 'Purchase price too large').optional().nullable(),\n  initialMileage: num().min(0, 'Initial mileage must be non-negative').max(10000000, 'Initial mileage too large').optional().nullable(),\n  currentMileage: num().min(0, 'Current mileage must be non-negative').max(10000000, 'Current mileage too large').optional().nullable(),\n  status: vehicleStatusEnum.default('active'),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n});\n\nexport const refuelSchema = z.object({\n  id: optionalId,\n  busId: optionalId,\n  refuelDate: dateField,\n  quantity: num().positive('Quantity must be greater than 0').max(10000, 'Quantity too large'),\n  unitPrice: num().positive('Unit price must be greater than 0').max(100000, 'Unit price too large'),\n  totalCost: num().positive('Total cost must be greater than 0').max(10000000, 'Total cost too large').optional(),\n  fuelType: fuelTypeEnum.default('diesel'),\n  odometer: num().min(0, 'Odometer must be non-negative').max(10000000, 'Odometer value too large').optional().nullable(),\n  fuelStation: z.string().max(200, 'Fuel station name too long').optional().nullable(),\n  invoiceNumber: z.string().max(100, 'Invoice number too long').optional().nullable(),\n  paymentMethod: paymentMethodEnum.optional().nullable(),\n  paidBy: optionalId.nullable(),\n  status: refuelStatusEnum.default('completed'),\n  notes: z.string().max(1000, 'Notes too long').optional().nullable(),\n});\n\n//=====================================================================//\n// SETTINGS VALIDATION SCHEMA\n//=====================================================================//\n\nexport const settingsSchema = z.object({\n  // School Information\n  schoolName: z.string().min(2, 'School name must be at least 2 characters').max(200, 'School name too long'),\n  schoolAddress: z.string().max(500, 'School address too long').optional(),\n  schoolPhone: phoneField,\n  schoolEmail: emailField,\n  schoolWebsite: z.string().url('Must be a valid URL').max(255, 'School website URL too long').optional(), // New\n  schoolLogo: z.string().url('Must be a valid image URL').max(255, 'School logo URL too long').optional(), // New\n  currentAcademicYear: academicYearField,\n\n  // Academic Settings\n  gradingScale: z.any().optional(),\n  attendanceRequirement: num().min(0, 'Attendance requirement must be non-negative').max(100, 'Attendance requirement cannot exceed 100').default(75.00),\n  maxClassSize: num().int('Max class size must be an integer').min(1, 'Max class size must be at least 1').max(200, 'Max class size cannot exceed 200').default(34),\n  minimumPassingGrade: num().min(0, 'Minimum passing grade must be non-negative').max(100, 'Minimum passing grade cannot exceed 100').default(60.00),\n  defaultExamDuration: num().int('Default exam duration must be in minutes').min(15, 'Exam duration must be at least 15 minutes').max(480, 'Exam duration cannot exceed 480 minutes').default(120),\n  calendarSystem: calendarSystemEnum.default('SEMESTER'),\n  startMonth: z.string().default('september'),\n  endMonth: z.string().default('june'),\n\n  // Notification Settings\n  academicAlerts: z.boolean().default(true),\n  attendanceAlerts: z.boolean().default(true),\n  eventAlerts: z.boolean().default(true),\n  homeworkAlerts: z.boolean().default(true),\n  feesReminder: z.boolean().default(true),\n  feesOverdueAlerts: z.boolean().default(true),\n  emailNotifications: z.boolean().default(true),\n  smsNotifications: z.boolean().default(false),\n  parentNotifications: z.boolean().default(true),\n  lowGradeAlerts: z.boolean().default(true),\n  allowLateSubmission: z.boolean().default(true),\n  examResultsAlerts: z.boolean().default(true),\n  disciplinaryAlerts: z.boolean().default(true),\n  achievementAlerts: z.boolean().default(true),\n  maintenanceNotifications: z.boolean().default(true),\n\n  // Security Settings\n  twoFactorEnabled: z.boolean().default(false),\n  sessionTimeout: z.string().regex(/^\\d{1,4}$/, 'Session timeout must be a number between 1-9999 minutes').default('60'),\n  passwordRequireSymbols: z.boolean().default(true),\n  loginNotifications: z.boolean().default(true),\n  parentAccessEnabled: z.boolean().default(true),\n  teacherAccessEnabled: z.boolean().default(true),\n  studentAccessEnabled: z.boolean().default(true),\n\n  // System Preferences\n  timeZone: z.string().min(1, 'Time zone is required').default('UTC'),\n  language: languageEnum.default('en'),\n  theme: z.enum(['light', 'dark', 'system']).default('system'),\n  dateFormat: z.enum(['YYYY-MM-DD', 'MM/DD/YYYY', 'DD/MM/YYYY', 'DD-MM-YY', 'DD-MM-YYYY']).default('MM/DD/YYYY'), // Adjusted default to match table\n  timeFormat: z.enum(['12', '24']).default('12'),\n  currency: z.string().length(3, 'Currency must be a 3-letter ISO code').regex(/^[A-Z]{3}$/, 'Currency must be uppercase ISO code').default('USD'),\n\n  // Academic Calendar Settings\n  gradingPeriods: num().int('Grading periods must be an integer').min(1, 'Grading periods must be at least 1').max(12, 'Grading periods cannot exceed 12').default(4),\n  schoolStartTime: z.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, 'Invalid start time format (HH:MM)').default('08:00'),\n  schoolEndTime: z.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, 'Invalid end time format (HH:MM)').default('15:00'),\n  lunchBreakDuration: num().int('Lunch break duration must be in minutes').min(15, 'Lunch break must be at least 15 minutes').max(120, 'Lunch break cannot exceed 120 minutes').default(30),\n\n  // Maintenance & Backup Settings\n  maintenanceMode: z.boolean().default(false),\n  autoBackup: z.boolean().default(true),\n\n});\n\n//=====================================================================//\n// UTILITY SCHEMAS\n//=====================================================================//\n\nexport const idParamSchema = z.object({\n  id: optionalId,\n});\n\nexport const paginationSchema = z.object({\n  page: num().int().min(1).default(1),\n  limit: num().int().min(1).max(100).default(10),\n});\n\nexport const dateRangeSchema = z.object({\n  dateFrom: dateField,\n  dateTo: dateField,\n})\n\n",
    "// lib/validations/enums.js\r\nimport { z } from 'zod'\r\nimport { ENUMS } from './ENUMS'\r\n\r\nconst createZodEnum = (enumKey) => {\r\n  const values = ENUMS[enumKey]?.values\r\n  if (!values) throw new Error(`Enum ${enumKey} not found`)\r\n  return z.enum(values)\r\n}\r\n\r\n// Core System\r\nexport const userTypeEnum = createZodEnum('userType')\r\nexport const userStatusEnum = createZodEnum('userStatus')\r\nexport const tokenStatusEnum = createZodEnum('tokenStatus')\r\nexport const tokenTypeEnum = createZodEnum('tokenType')\r\nexport const fileStatusEnum = createZodEnum('fileStatus')\r\n\r\n// Educational System\r\nexport const genderEnum = createZodEnum('gender')\r\nexport const studentStatusEnum = createZodEnum('studentStatus')\r\nexport const teacherStatusEnum = createZodEnum('teacherStatus')\r\nexport const employmentTypeEnum = createZodEnum('employmentType')\r\nexport const relationshipTypeEnum = createZodEnum('relationshipType')\r\nexport const semesterEnum = createZodEnum('semester')\r\nexport const classStatusEnum = createZodEnum('classStatus')\r\nexport const sectionStatusEnum = createZodEnum('sectionStatus')\r\nexport const languageEnum = createZodEnum('language')\r\nexport const enrollmentStatusEnum = createZodEnum('enrollmentStatus')\r\nexport const assignmentStatusEnum = createZodEnum('assignmentStatus')\r\nexport const calendarSystemEnum = createZodEnum('calendarSystem')\r\n\r\n// Assessment & Exams\r\nexport const assessmentTypeEnum = createZodEnum('assessmentType')\r\nexport const assessmentStatusEnum = createZodEnum('assessmentStatus')\r\nexport const submissionTypeEnum = createZodEnum('submissionType')\r\nexport const examTypeEnum = createZodEnum('examType')\r\nexport const examSecurityEnum = createZodEnum('examSecurity')\r\nexport const examStatusEnum = createZodEnum('examStatus')\r\nexport const gradeStatusEnum = createZodEnum('gradeStatus')\r\nexport const attendanceStatusEnum = createZodEnum('attendanceStatus')\r\nexport const proficiencyLevelEnum = createZodEnum('proficiencyLevel')\r\nexport const dayOfWeekEnum = createZodEnum('dayOfWeek')\r\n\r\n// Alerts\r\nexport const alertTypeEnum = createZodEnum('alertType')\r\nexport const alertPriorityEnum = createZodEnum('alertPriority')\r\nexport const alertStatusEnum = createZodEnum('alertStatus')\r\n\r\n// Fees & Payments\r\nexport const feeTypeStatusEnum = createZodEnum('feeTypeStatus')\r\nexport const paymentTypeEnum = createZodEnum('paymentType')\r\nexport const scheduleEnum = createZodEnum('schedule')\r\nexport const feeStatusEnum = createZodEnum('feeStatus')\r\nexport const feeInstallmentStatusEnum = createZodEnum('feeInstallmentStatus')\r\nexport const paymentMethodEnum = createZodEnum('paymentMethod')\r\nexport const paymentStatusEnum = createZodEnum('paymentStatus')\r\n\r\n// Events\r\nexport const eventTypeEnum = createZodEnum('eventType')\r\nexport const eventStatusEnum = createZodEnum('eventStatus')\r\nexport const eventVisibilityEnum = createZodEnum('eventVisibility')\r\nexport const participantTypeEnum = createZodEnum('participantType')\r\n\r\n// Expenses\r\nexport const expenseCategoryEnum = createZodEnum('expenseCategory')\r\nexport const expenseStatusEnum = createZodEnum('expenseStatus')\r\n\r\n// Tracker\r\nexport const trackerModeEnum = createZodEnum('trackerMode')\r\n\r\n// Transport\r\nexport const driverStatusEnum = createZodEnum('driverStatus')\r\nexport const vehicleStatusEnum = createZodEnum('vehicleStatus')\r\nexport const vehicleTypeEnum = createZodEnum('vehicleType')\r\nexport const vehicleDocumentTypeEnum = createZodEnum('vehicleDocumentType')\r\nexport const busStatusEnum = createZodEnum('busStatus')\r\nexport const refuelStatusEnum = createZodEnum('refuelStatus')\r\nexport const fuelTypeEnum = createZodEnum('fuelType')\r\nexport const maintenanceTypeEnum = createZodEnum('maintenanceType')\r\nexport const maintenanceStatusEnum = createZodEnum('maintenanceStatus')\r\n\r\n// Personal\r\nexport const maritalStatusEnum = createZodEnum('maritalStatus')",
    "\n\nimport { permissionsTable, rolePermissionsTable, rolesTable } from '@/database/schema';\nimport { eq, and } from 'drizzle-orm';\nimport { Repository, DB } from 'najm-api';\n\n@Repository()\nexport class PermissionRepository {\n  declare db: DB;\n\n  async getAll() {\n    return await this.db.select().from(permissionsTable);\n  }\n\n  async getById(id: string) {\n    const [existingPermission] = await this.db\n      .select()\n      .from(permissionsTable)\n      .where(eq(permissionsTable.id, id));\n    return existingPermission;\n  }\n\n  async getByName(name: string) {\n    const [existingPermission] = await this.db\n      .select()\n      .from(permissionsTable)\n      .where(eq(permissionsTable.name, name));\n    return existingPermission;\n  }\n\n  async create(data) {\n    const [newPermission] = await this.db\n      .insert(permissionsTable)\n      .values(data)\n      .returning();\n    return newPermission;\n  }\n\n  async update(id: string, data) {\n    const [updatedPermission] = await this.db\n      .update(permissionsTable)\n      .set(data)\n      .where(eq(permissionsTable.id, id))\n      .returning();\n    return updatedPermission;\n  }\n\n  async delete(id: string) {\n    const [deletedPermission] = await this.db\n      .delete(permissionsTable)\n      .where(eq(permissionsTable.id, id))\n      .returning();\n    return deletedPermission;\n  }\n\n  async getPermissionsByRole(roleId: string) {\n    return await this.db\n      .select({\n        id: permissionsTable.id,\n        name: permissionsTable.name,\n        description: permissionsTable.description,\n        resource: permissionsTable.resource,\n        action: permissionsTable.action,\n      })\n      .from(rolePermissionsTable)\n      .leftJoin(permissionsTable, eq(rolePermissionsTable.permissionId, permissionsTable.id))\n      .where(eq(rolePermissionsTable.roleId, roleId));\n  }\n\n  async getRolesByPermission(permissionId: string) {\n    return await this.db\n      .select({\n        id: rolesTable.id,\n        name: rolesTable.name,\n        description: rolesTable.description,\n      })\n      .from(rolePermissionsTable)\n      .leftJoin(rolesTable, eq(rolePermissionsTable.roleId, rolesTable.id))\n      .where(eq(rolePermissionsTable.permissionId, permissionId));\n  }\n\n  async assignPermissionToRole(roleId: string, permissionId: string) {\n    const [newRolePermission] = await this.db\n      .insert(rolePermissionsTable)\n      .values({ roleId, permissionId })\n      .returning();\n    return newRolePermission;\n  }\n\n  async removePermissionFromRole(roleId: string, permissionId: string) {\n    const [deletedRolePermission] = await this.db\n      .delete(rolePermissionsTable)\n      .where(and(eq(rolePermissionsTable.roleId, roleId), eq(rolePermissionsTable.permissionId, permissionId)))\n      .returning();\n    return deletedRolePermission;\n  }\n\n  async checkRoleHasPermission(roleId: string, permissionId: string) {\n    const [rolePermission] = await this.db\n      .select()\n      .from(rolePermissionsTable)\n      .where(and(eq(rolePermissionsTable.roleId, roleId), eq(rolePermissionsTable.permissionId, permissionId)));\n    return !!rolePermission;\n  }\n\n  async deleteAll() {\n    // First delete all role-permission relationships\n    await this.db.delete(rolePermissionsTable);\n\n    // Then delete all permissions\n    const deletedPermissions = await this.db\n      .delete(permissionsTable)\n      .returning();\n\n    return deletedPermissions;\n  }\n}",
    "import { createGuard, Injectable, GuardParams, Headers, Ctx } from \"najm-api\";\nimport { TokenService } from \"../tokens\";\n\n\n@Injectable()\nexport class PermissionGuards {\n  constructor(private tokenService: TokenService) { }\n\n  private async getUserPermissions(auth) {\n    const permissions = await this.tokenService.getUserPermissions(auth);\n    if (!permissions || !Array.isArray(permissions)) return null;\n    return permissions;\n  }\n\n  private checkPermissionMatch(permissions: string[], requiredPermission: string): boolean {\n    if (permissions.includes(requiredPermission)) {\n      return true;\n    }\n\n    const [requiredAction, requiredResource] = requiredPermission.split(':');\n    if (requiredAction && requiredResource) {\n      if (permissions.includes(`${requiredAction}:*`)) {\n        return true;\n      }\n      if (permissions.includes(`*:${requiredResource}`)) {\n        return true;\n      }\n    }\n    if (permissions.includes('*:*')) {\n      return true;\n    }\n    return false;\n  }\n\n\n  async hasPermission(@Headers('authorization') auth, @Ctx() ctx, @GuardParams() requiredPermission) {\n    \n    await this.tokenService.storeUserInCache(auth, ctx);\n    const permissions = await this.getUserPermissions(auth);\n    if (!permissions) return false;\n    const check = this.checkPermissionMatch(permissions, requiredPermission);\n    return check\n  }\n}\n\nexport const Permission = (...permissions) => createGuard(PermissionGuards, 'hasPermission')(...permissions);\n\n",
    "import { Controller, Get, Post, Put, Delete, Params, Body, t } from 'najm-api';\nimport { PermissionService } from './PermissionService';\nimport { isAdmin } from '@/roles/RoleGuards';\n\n@Controller('/permissions')\n@isAdmin()\nexport class PermissionController {\n  constructor(private permissionService: PermissionService) { }\n\n  @Get()\n  async getPermissions() {\n    const permissions = await this.permissionService.getAll();\n    return {\n      data: permissions,\n      message: t('permissions.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Get('/:id')\n  async getPermission(@Params('id') id: string) {\n    const permission = await this.permissionService.getById(id);\n    return {\n      data: permission,\n      message: t('permissions.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post()\n  async create(@Body() body) {\n    const newPermission = await this.permissionService.create(body);\n    return {\n      data: newPermission,\n      message: t('permissions.success.created'),\n      status: 'success'\n    };\n  }\n\n  @Put('/:id')\n  async update(@Params('id') id: string, @Body() body) {\n    const updatedPermission = await this.permissionService.update(id, body);\n    return {\n      data: updatedPermission,\n      message: t('permissions.success.updated'),\n      status: 'success'\n    };\n  }\n\n  @Delete('/:id')\n  async delete(@Params('id') id: string) {\n    const result = await this.permissionService.delete(id);\n    return {\n      data: result,\n      message: t('permissions.success.deleted'),\n      status: 'success'\n    };\n  }\n\n  @Get('/role/:roleId')\n  async getByRole(@Params('roleId') roleId: string) {\n    const permissions = await this.permissionService.getPermissionsByRole(roleId);\n    return {\n      data: permissions,\n      message: t('permissions.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Get('/roles/:permissionId')\n  async getRolesByPermission(@Params('permissionId') permissionId: string) {\n    const roles = await this.permissionService.getRolesByPermission(permissionId);\n    return {\n      data: roles,\n      message: t('permissions.success.retrieved'),\n      status: 'success'\n    };\n  }\n\n  @Post('/assign/:roleId/:permissionId')\n  async assignToRole(\n    @Params('roleId') roleId: string,\n    @Params('permissionId') permissionId: string\n  ) {\n    const result = await this.permissionService.assignPermissionToRole(roleId, permissionId);\n    return {\n      data: result,\n      message: t('permissions.success.assigned'),\n      status: 'success'\n    };\n  }\n\n  @Delete('/remove/:roleId/:permissionId')\n  async removeFromRole(\n    @Params('roleId') roleId: string,\n    @Params('permissionId') permissionId: string\n  ) {\n    const result = await this.permissionService.removePermissionFromRole(roleId, permissionId);\n    return {\n      data: result,\n      message: t('permissions.success.removed'),\n      status: 'success'\n    };\n  }\n\n  @Delete()\n  @isAdmin()\n  async deleteAll() {\n    const result = await this.permissionService.deleteAll();\n    return {\n      data: result,\n      message: t('permissions.success.allDeleted'),\n      status: 'success'\n    };\n  }\n}",
    "import { Injectable } from 'najm-api';\nimport { PermissionRepository } from './PermissionRepository';\nimport { PermissionValidator } from './PermissionValidator';\nimport { RoleService } from '../roles/RoleService';\n\n@Injectable()\nexport class PermissionService {\n  constructor(\n    private permissionRepository: PermissionRepository,\n    private permissionValidator: PermissionValidator,\n    private roleService: RoleService\n  ) { }\n\n  async getAll() {\n    return await this.permissionRepository.getAll();\n  }\n\n  async getById(id: string) {\n    await this.permissionValidator.checkPermissionExists(id);\n    return await this.permissionRepository.getById(id);\n  }\n\n  async getByName(name: string) {\n    return await this.permissionRepository.getByName(name);\n  }\n\n  async getByResource(resource: string) {\n    return await this.permissionRepository.getAll().then(permissions =>\n      permissions.filter(p => p.resource === resource)\n    );\n  }\n\n  async create(data) {\n    await this.permissionValidator.validateCreatePermission(data);\n    await this.permissionValidator.checkPermissionNameUnique(data.name);\n    return await this.permissionRepository.create(data);\n  }\n\n  async update(id: string, data) {\n    await this.permissionValidator.checkPermissionExists(id);\n    await this.permissionValidator.checkPermissionNameUnique(data.name, id);\n    return await this.permissionRepository.update(id, data);\n  }\n\n  async delete(id: string) {\n    await this.permissionValidator.checkPermissionExists(id);\n    return await this.permissionRepository.delete(id);\n  }\n\n  async getPermissionsByRole(roleId: string) {\n    return await this.permissionRepository.getPermissionsByRole(roleId);\n  }\n\n  async getRolesByPermission(permissionId: string) {\n    await this.permissionValidator.checkPermissionExists(permissionId);\n    return await this.permissionRepository.getRolesByPermission(permissionId);\n  }\n\n  async assignPermissionToRole(roleId: string, permissionId: string) {\n    await this.permissionValidator.checkRoleHasPermission(roleId, permissionId);\n    return await this.permissionRepository.assignPermissionToRole(roleId, permissionId);\n  }\n\n  async removePermissionFromRole(roleId: string, permissionId: string) {\n    return await this.permissionRepository.removePermissionFromRole(roleId, permissionId);\n  }\n\n  async seedDefaultPermissions(defaultPermissions) {\n    const createdPermissions = [];\n\n    for (const permission of defaultPermissions) {\n      try {\n        const permissionEntity = await this.create(permission);\n        createdPermissions.push(permissionEntity);\n      } catch (error) {\n        continue;\n      }\n    }\n    return createdPermissions;\n  }\n\n  async seedDefaultRolePermissions(defaultRolePermissions) {\n    const results = [];\n\n    for (const { roleName, permissions } of defaultRolePermissions) {\n      try {\n        await this.permissionValidator.checkRoleExistsByName(roleName);\n        const role = await this.roleService.getByName(roleName);\n\n        for (const permissionName of permissions) {\n          try {\n            await this.permissionValidator.checkPermissionExistsByName(permissionName);\n            const permission = await this.getByName(permissionName);\n\n            await this.permissionValidator.checkRoleHasPermission(role.id, permission.id);\n\n            await this.assignPermissionToRole(role.id, permission.id);\n            results.push({ role: roleName, permission: permissionName });\n          } catch (error) {\n            continue;\n          }\n        }\n      } catch (error) {\n        continue;\n      }\n    }\n\n    return results;\n  }\n\n  async deleteAll() {\n    return await this.permissionRepository.deleteAll();\n  }\n\n}",
    "import { Injectable, t } from 'najm-api';\nimport { PermissionRepository } from './PermissionRepository';\nimport { RoleValidator } from '../roles';\nimport { parseSchema } from '@/shared';\nimport { z } from 'zod';\n\nconst permissionSchema = z.object({\n  name: z.string().min(1, 'Permission name is required'),\n  description: z.string().optional(),\n  resource: z.string().min(1, 'Resource is required'),\n  action: z.string().min(1, 'Action is required'),\n});\n\n@Injectable()\nexport class PermissionValidator {\n  constructor(\n    private permissionRepository: PermissionRepository,\n    private roleValidator: RoleValidator\n  ) {}\n\n  async validateCreatePermission(data) {\n    return parseSchema(permissionSchema, data);\n  }\n\n  async isPermissionExists(id: string) {\n    const existingPermission = await this.permissionRepository.getById(id);\n    return !!existingPermission;\n  }\n\n  async isPermissionNameExists(name: string) {\n    const existingPermission = await this.permissionRepository.getByName(name);\n    return !!existingPermission;\n  }\n\n  //======================= throw errors\n\n  async checkPermissionExists(id: string) {\n    const permissionExists = await this.isPermissionExists(id);\n    if (!permissionExists) {\n      throw new Error(t('permissions.errors.notFound'));\n    }\n    return permissionExists;\n  }\n\n  async checkPermissionExistsByName(name: string) {\n    const permissionExists = await this.isPermissionNameExists(name);\n    if (!permissionExists) {\n      throw new Error(t('permissions.errors.notFound'));\n    }\n    return permissionExists;\n  }\n\n  async checkPermissionNameUnique(name: string, excludeId = null) {\n    if (!name) return;\n    const existingPermission = await this.permissionRepository.getByName(name);\n    if (existingPermission && existingPermission.id !== excludeId) {\n      throw new Error(t('permissions.errors.nameExists'));\n    }\n  }\n\n  async checkRoleExists(id: string) {\n    return await this.roleValidator.checkRoleExists(id);\n  }\n\n  async checkRoleExistsByName(name: string) {\n    return await this.roleValidator.checkRoleExistsByName(name);\n  }\n\n  async checkRoleHasPermission(roleId: string, permissionId: string) {\n    await this.roleValidator.checkRoleExists(roleId);\n    await this.checkPermissionExists(permissionId);\n\n    const hasPermission = await this.permissionRepository.checkRoleHasPermission(roleId, permissionId);\n    if (hasPermission) {\n      throw new Error(t('permissions.errors.roleAlreadyHasPermission'));\n    }\n  }\n\n}",
    "\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport _isEmpty from 'lodash.isempty';\nimport { SQL, sql } from 'drizzle-orm';\nimport { AnyPgColumn } from 'drizzle-orm/pg-core';\n\nexport const avatarsPath = path.join(process.cwd(), 'avatars');\n\nexport const parseSchema = async (schema, data) => {\n  try {\n    return await schema.parseAsync(data);\n  } catch (error) {\n    const errors = error.issues || error.errors || [];\n    const errorMessage = errors\n      .map(err => `${err.path.join('.')}: ${err.message}`)\n      .join('; ');\n    throw new Error(errorMessage);\n  }\n};\n\nexport const clean = (obj: any): any => {\n  const cleaned = {};\n\n  for (const [key, value] of Object.entries(obj)) {\n    if (value !== null && value !== undefined && value !== '') {\n      cleaned[key] = value;\n    }\n  }\n\n  return cleaned;\n}\n\nexport const getAvatarFile = async (fileName) => {\n  try {\n    const filePath = path.join(avatarsPath, fileName);\n    const buffer: any = await fs.readFile(filePath);\n    const file = new File([buffer], fileName, {\n      type: 'image/png'\n    });\n    return file;\n  }\n  catch (error) {\n    return null;\n  }\n}\n\nexport const formatDate = (dateValue) => {\n  if (!dateValue) return null;\n\n  let date: Date;\n\n  if (dateValue instanceof Date) {\n    date = dateValue;\n  } else if (typeof dateValue === 'string') {\n    date = new Date(dateValue);\n  } else {\n    return null;\n  }\n\n  if (isNaN(date.getTime())) return null;\n\n  return date.toISOString().split('T')[0];\n}\n\nexport function calculateAge(dateOfBirth) {\n  if (!dateOfBirth) return null;\n\n  const formattedDate = formatDate(dateOfBirth);\n  if (!formattedDate) return null;\n\n  const birth = new Date(formattedDate);\n  const today = new Date();\n  let age = today.getFullYear() - birth.getFullYear();\n  const monthDiff = today.getMonth() - birth.getMonth();\n\n  if (monthDiff < 0 || (monthDiff === 0 && today.getDate() < birth.getDate())) {\n    age--;\n  }\n\n  return age;\n}\n\nexport function calculateYearsOfExperience(hireDate) {\n  if (!hireDate) return null;\n\n  const formattedDate = formatDate(hireDate);\n  if (!formattedDate) return null;\n\n  const hire = new Date(formattedDate);\n  const today = new Date();\n  let years = today.getFullYear() - hire.getFullYear();\n  const monthDiff = today.getMonth() - hire.getMonth();\n\n  if (monthDiff < 0 || (monthDiff === 0 && today.getDate() < hire.getDate())) {\n    years--;\n  }\n\n  return years;\n}\n\nexport function pickProps<T>(source: T, keys): Partial<T> {\n  const result = {};\n\n  for (const key of keys) {\n    if (source[key] !== undefined) {\n      result[key] = source[key];\n    }\n  }\n\n  return result;\n}\n\nexport const isEmpty = _isEmpty;\n\nexport function jsonAgg(\n  fields: Record<string, AnyPgColumn>,\n  orderBy?: AnyPgColumn | SQL\n) {\n  const fieldEntries = Object.entries(fields);\n  const jsonBuildArgs = fieldEntries.flatMap(([key, column]) => [\n    sql.raw(`'${key}'`),\n    column\n  ]);\n\n  let aggregation = sql`json_agg(json_build_object(${sql.join(jsonBuildArgs, sql`, `)})`;\n\n  if (orderBy) {\n    aggregation = sql`${aggregation} ORDER BY ${orderBy}`;\n  }\n\n  aggregation = sql`${aggregation})`;\n\n  // Wrap with COALESCE to return empty array instead of null\n  return sql`COALESCE(${aggregation}, '[]'::json)`;\n}\n\nexport function jsonAggSubquery(\n  fields: Record<string, AnyPgColumn>,\n  from: any,\n  where: SQL,\n  orderBy?: AnyPgColumn | SQL\n) {\n  const fieldEntries = Object.entries(fields);\n  const jsonBuildArgs = fieldEntries.flatMap(([key, column]) => [\n    sql.raw(`'${key}'`),\n    column\n  ]);\n\n  let aggregation = sql`\n    SELECT json_agg(DISTINCT json_build_object(${sql.join(jsonBuildArgs, sql`, `)}))\n    FROM ${from}\n    WHERE ${where}\n  `;\n\n  if (orderBy) {\n    aggregation = sql`\n      SELECT json_agg(json_build_object(${sql.join(jsonBuildArgs, sql`, `)}) ORDER BY ${orderBy})\n      FROM ${from}\n      WHERE ${where}\n    `;\n  }\n\n  return sql`COALESCE((${aggregation}), '[]'::json)`;\n}\n\nexport const isPath = (img) =>\n  typeof img === 'string' && img.trim().length > 0 && (img.startsWith('/') || img.startsWith('http') || img.startsWith('storage/'));\n\n\nexport const isFile = (img) =>\n  !!img && typeof img !== 'string' && img instanceof File;\n\n",
    "import { drizzle, PostgresJsDatabase } from 'drizzle-orm/postgres-js';\nimport * as schema from './schema';\nimport { DB as GenericDB } from 'najm-api';\n\nexport type DB = PostgresJsDatabase<typeof schema>;\n\n/**\n * ⚠️  IMPORTANT: Development-Only Database Instance\n *\n * This DB instance is ONLY for standalone development and testing.\n * When used as a plugin in production, the database will be injected\n * automatically by the najm-api plugin system.\n *\n * DO NOT use this in production - it creates a duplicate connection!\n *\n * In production:\n * - Use najm-auth as a plugin: `import { AuthPlugin } from 'najm-auth/plugin'`\n * - The main app registers the database in DatabaseRegistry\n * - Repositories get the DB automatically injected via dependency injection\n *\n * @example Production Usage\n * ```typescript\n * import { Server, DatabaseRegistry } from 'najm-api';\n * import { AuthPlugin } from 'najm-auth/plugin';\n *\n * const app = await Server({\n *   databases: {\n *     auth: myDbInstance  // Single source of truth\n *   },\n *   plugins: [AuthPlugin]\n * });\n * ```\n */\nexport const db: DB | GenericDB = process.env.DB_URL\n  ? drizzle(process.env.DB_URL, { schema })\n  : null as any;  // Will be injected by plugin system in production\n\nif (process.env.NODE_ENV === 'development' && db) {\n  console.warn('⚠️  najm-auth: Using standalone DB instance. This should only be used for development.');\n}",
    "import { pgTable, text, boolean, timestamp } from 'drizzle-orm/pg-core';\r\nimport { nanoid } from 'nanoid';\r\n\r\nimport { sql } from 'drizzle-orm';\r\nimport { tokenStatusEnum, tokenTypeEnum, userStatusEnum } from './PgEnum';\r\n\r\nexport const timestamps = {\r\n  createdAt: timestamp('created_at', { mode: 'string' }).defaultNow(),\r\n  updatedAt: timestamp('updated_at', { mode: 'string' }).defaultNow().$onUpdate(() => sql`CURRENT_TIMESTAMP`),\r\n};\r\n\r\nexport const idField = (length = 5) =>\r\n  text('id').primaryKey().notNull().$defaultFn(() => nanoid(length));\r\n\r\nexport const rolesTable = pgTable('roles', {\r\n  id: idField(),\r\n  name: text('name').notNull(),\r\n  description: text('description'),\r\n});\r\n\r\nexport const usersTable = pgTable('users', {\r\n  id: idField(8),\r\n  email: text('email').notNull().unique(),\r\n  emailVerified: boolean('email_verified').default(false),\r\n  password: text('password').notNull(),\r\n  image: text('image').default('noavatar.png'),\r\n  status: userStatusEnum('status').default('pending'),\r\n  roleId: text('role_id').references(() => rolesTable.id),\r\n  lastLogin: timestamp('last_login', { mode: 'string' }),\r\n  ...timestamps,\r\n});\r\n\r\nexport const tokensTable = pgTable('tokens', {\r\n  id: idField(10),\r\n  userId: text('user_id').references(() => usersTable.id, { onDelete: 'cascade' }).unique().notNull(),\r\n  token: text('token').notNull(),\r\n  type: tokenTypeEnum('type').default('refresh'),\r\n  status: tokenStatusEnum('status').default('active'),\r\n  expiresAt: timestamp('expires_at', { mode: 'string' }).notNull(),\r\n  ...timestamps,\r\n});\r\n\r\n\r\nexport const permissionsTable = pgTable('permissions', {\r\n  id: idField(),\r\n  name: text('name').notNull().unique(),\r\n  description: text('description'),\r\n  resource: text('resource').notNull(),\r\n  action: text('action').notNull(),\r\n  ...timestamps,\r\n});\r\n\r\n\r\nexport const rolePermissionsTable = pgTable('role_permissions', {\r\n  id: idField(),\r\n  roleId: text('role_id').references(() => rolesTable.id).notNull(),\r\n  permissionId: text('permission_id').references(() => permissionsTable.id).notNull(),\r\n  ...timestamps,\r\n});\r\n\r\nexport type User = typeof usersTable.$inferSelect;\r\nexport type NewUser = typeof usersTable.$inferInsert;\r\nexport type Token = typeof tokensTable.$inferSelect;\r\nexport type NewToken = typeof tokensTable.$inferInsert;\r\nexport type Role = typeof rolesTable.$inferSelect;\r\nexport type NewRole = typeof rolesTable.$inferInsert;\r\nexport type Permission = typeof permissionsTable.$inferSelect;\r\nexport type NewPermission = typeof permissionsTable.$inferInsert;\r\nexport type RolePermission = typeof rolePermissionsTable.$inferSelect;\r\nexport type NewRolePermission = typeof rolePermissionsTable.$inferInsert;\r\n",
    "// --- Drizzle pgEnum Generation ---\nimport { getEnumConfig } from \"@/lib/ENUMS\";\nimport { pgEnum } from \"drizzle-orm/pg-core\";\n\nconst createPgEnum = (enumKey) => {\n  const config = getEnumConfig(enumKey)\n  if (!config) throw new Error(`Enum ${enumKey} not found`)\n  const enumName = config.name || enumKey\n  return pgEnum(enumName, config.values)\n}\n\nexport const userStatusEnum = createPgEnum('userStatus');\nexport const tokenStatusEnum = createPgEnum('tokenStatus');\nexport const tokenTypeEnum = createPgEnum('tokenType');\nexport const studentStatusEnum = createPgEnum('studentStatus');\n",
    "import { PluginConfig } from 'najm-api';\r\n\r\n// Import all controllers\r\nimport { UserController } from './users/UserController';\r\nimport { AuthController } from './auth/AuthController';\r\nimport { RoleController } from './roles/RoleController';\r\nimport { PermissionController } from './permissions/PermissionController';\r\n\r\n// Import all services\r\nimport { UserService } from './users/UserService';\r\nimport { AuthService } from './auth/AuthService';\r\nimport { CookieService } from './auth/CookieService';\r\nimport { EncryptionService } from './auth/EncryptionService';\r\nimport { RoleService } from './roles/RoleService';\r\nimport { PermissionService } from './permissions/PermissionService';\r\nimport { TokenService } from './tokens/TokenService';\r\n\r\n// Import all repositories\r\nimport { UserRepository } from './users/UserRepository';\r\nimport { RoleRepository } from './roles/RoleRepository';\r\nimport { PermissionRepository } from './permissions/PermissionRepository';\r\nimport { TokenRepository } from './tokens/TokenRepository';\r\n\r\n// Import all validators\r\nimport { UserValidator } from './users/UserValidator';\r\nimport { RoleValidator } from './roles/RoleValidator';\r\nimport { PermissionValidator } from './permissions/PermissionValidator';\r\n\r\n// Import all guards\r\nimport { RoleGuards } from './roles/RoleGuards';\r\nimport { PermissionGuards } from './permissions/PermissionGuards';\r\n\r\n/**\r\n * najm-auth Plugin Configuration\r\n *\r\n * This plugin provides comprehensive authentication and authorization features:\r\n * - User management with secure password hashing\r\n * - Role-based access control (RBAC)\r\n * - Permission management with guards\r\n * - JWT token authentication\r\n * - Session management with cookies\r\n * - Encryption utilities\r\n *\r\n * @example\r\n * ```typescript\r\n * import { Server } from 'najm-api';\r\n * import { AuthPlugin } from 'najm-auth/plugin';\r\n * import { db } from './database';\r\n *\r\n * const app = await Server({\r\n *   databases: {\r\n *     default: db,\r\n *     auth: db  // Can use same DB or separate\r\n *   },\r\n *   plugins: [AuthPlugin]\r\n * });\r\n * ```\r\n */\r\nexport const AuthPlugin: PluginConfig = {\r\n  name: 'najm-auth',\r\n  version: '0.1.3',\r\n\r\n  /**\r\n   * Database configuration\r\n   * By default uses 'auth' database, but can be changed to 'default'\r\n   * The main application should register this database in DatabaseRegistry\r\n   */\r\n  database: 'auth',\r\n\r\n  /**\r\n   * HTTP Controllers\r\n   * Provides REST API endpoints for authentication and authorization\r\n   */\r\n  controllers: [\r\n    UserController,       // /users - User CRUD operations\r\n    AuthController,       // /auth - Login, logout, register, verify\r\n    RoleController,       // /roles - Role management\r\n    PermissionController, // /permissions - Permission management\r\n  ],\r\n\r\n  /**\r\n   * Business Logic Services\r\n   * Core services for authentication and authorization\r\n   */\r\n  services: [\r\n    UserService,        // User business logic\r\n    AuthService,        // Authentication logic (login, register, verify)\r\n    CookieService,      // Cookie management for sessions\r\n    EncryptionService,  // Password hashing and encryption\r\n    RoleService,        // Role management logic\r\n    PermissionService,  // Permission management logic\r\n    TokenService,       // JWT token generation and validation\r\n  ],\r\n\r\n  /**\r\n   * Data Access Repositories\r\n   * Database access layer for entities\r\n   */\r\n  repositories: [\r\n    UserRepository,       // Users table access\r\n    RoleRepository,       // Roles table access\r\n    PermissionRepository, // Permissions table access\r\n    TokenRepository,      // Tokens table access\r\n  ],\r\n\r\n  /**\r\n   * Additional Providers\r\n   * Validators and Guards for request validation and route protection\r\n   */\r\n  providers: [\r\n    UserValidator,       // User data validation\r\n    RoleValidator,       // Role data validation\r\n    PermissionValidator, // Permission data validation\r\n    RoleGuards,          // Role-based route guards\r\n    PermissionGuards,    // Permission-based route guards\r\n  ],\r\n\r\n  /**\r\n   * Setup hook - called after plugin registration\r\n   */\r\n  onSetup: async () => {\r\n    console.log('  ✓ najm-auth plugin initialized');\r\n    console.log('    - Authentication endpoints ready');\r\n    console.log('    - Authorization system active');\r\n    console.log('    - Database connection established');\r\n  },\r\n\r\n  /**\r\n   * Teardown hook - called before plugin removal\r\n   */\r\n  onTeardown: async () => {\r\n    console.log('  ✓ najm-auth plugin cleanup complete');\r\n  },\r\n};"
  ],
  "mappings": "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",
  "debugId": "46440D7F5E9F307264756E2164756E21",
  "names": []
}
1
+ var __defProp = Object.defineProperty;
2
+ var __export = (target, all) => {
3
+ for (var name in all)
4
+ __defProp(target, name, {
5
+ get: all[name],
6
+ enumerable: true,
7
+ configurable: true,
8
+ set: (newValue) => all[name] = () => newValue
9
+ });
10
+ };
11
+ var __legacyDecorateClassTS = function(decorators, target, key, desc) {
12
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
13
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function")
14
+ r = Reflect.decorate(decorators, target, key, desc);
15
+ else
16
+ for (var i = decorators.length - 1;i >= 0; i--)
17
+ if (d = decorators[i])
18
+ r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
19
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
20
+ };
21
+ var __legacyDecorateParamTS = (index, decorator) => (target, key) => decorator(target, key, index);
22
+ var __legacyMetadataTS = (k, v) => {
23
+ if (typeof Reflect === "object" && typeof Reflect.metadata === "function")
24
+ return Reflect.metadata(k, v);
25
+ };
26
+
27
+ // src/auth/EncryptionService.ts
28
+ import { Injectable } from "najm-api";
29
+ import bcrypt from "bcrypt";
30
+ class EncryptionService {
31
+ constructor() {}
32
+ async hashPassword(password) {
33
+ if (!password)
34
+ return null;
35
+ if (typeof password !== "string" || password.trim().length === 0) {
36
+ return null;
37
+ }
38
+ return bcrypt.hash(password, 10);
39
+ }
40
+ async comparePassword(password, hashedPassword) {
41
+ return bcrypt.compare(password, hashedPassword);
42
+ }
43
+ }
44
+ EncryptionService = __legacyDecorateClassTS([
45
+ Injectable(),
46
+ __legacyMetadataTS("design:paramtypes", [])
47
+ ], EncryptionService);
48
+ // src/auth/CookieService.ts
49
+ import { setCookie, Injectable as Injectable2, deleteCookie } from "najm-api";
50
+ import timestring from "timestring";
51
+ class CookieService {
52
+ setRefreshCookie(refreshToken) {
53
+ const maxAge = timestring(process.env.REFRESH_EXPIRES_IN, "s");
54
+ setCookie("refreshToken", refreshToken, {
55
+ httpOnly: false,
56
+ sameSite: "Lax",
57
+ maxAge,
58
+ path: "/api/auth/refresh"
59
+ });
60
+ }
61
+ clearRefreshCookie() {
62
+ deleteCookie("refreshToken", {
63
+ httpOnly: false,
64
+ sameSite: "Lax",
65
+ path: "/api/auth/refresh",
66
+ maxAge: 0
67
+ });
68
+ }
69
+ }
70
+ CookieService = __legacyDecorateClassTS([
71
+ Injectable2()
72
+ ], CookieService);
73
+ // src/auth/AuthController.ts
74
+ import { Controller as Controller3, Get as Get3, Post as Post3, Params as Params3, Body as Body3, User as User2, t as t7 } from "najm-api";
75
+
76
+ // src/auth/AuthService.ts
77
+ import { t as t6 } from "najm-api";
78
+ import { Injectable as Injectable9, getCurrentLanguage as getCurrentLanguage2 } from "najm-api";
79
+
80
+ // src/users/UserRepository.ts
81
+ import { rolesTable, usersTable, permissionsTable, rolePermissionsTable } from "@/database/schema";
82
+ import { eq, ne } from "drizzle-orm";
83
+ import { Repository } from "najm-api";
84
+ class UserRepository {
85
+ getUser() {
86
+ return {
87
+ id: usersTable.id,
88
+ email: usersTable.email,
89
+ emailVerified: usersTable.emailVerified,
90
+ image: usersTable.image,
91
+ status: usersTable.status,
92
+ roleId: usersTable.roleId,
93
+ role: rolesTable.name,
94
+ createdAt: usersTable.createdAt,
95
+ updatedAt: usersTable.updatedAt
96
+ };
97
+ }
98
+ async getAll() {
99
+ const allUsers = await this.db.select(this.getUser()).from(usersTable).leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id));
100
+ return Promise.all(allUsers.map(async (user) => ({
101
+ ...user,
102
+ permissions: await this.getUserPermissions(user.id)
103
+ })));
104
+ }
105
+ async getById(id) {
106
+ const [user] = await this.db.select(this.getUser()).from(usersTable).leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id)).where(eq(usersTable.id, id)).limit(1);
107
+ if (!user)
108
+ return user;
109
+ return {
110
+ ...user,
111
+ permissions: await this.getUserPermissions(user.id)
112
+ };
113
+ }
114
+ async getByEmail(email) {
115
+ const [existingUser] = await this.db.select(this.getUser()).from(usersTable).leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id)).where(eq(usersTable.email, email));
116
+ return existingUser;
117
+ }
118
+ async create(data) {
119
+ const [newUser] = await this.db.insert(usersTable).values(data).returning();
120
+ return newUser;
121
+ }
122
+ async update(id, data) {
123
+ const [updatedUser] = await this.db.update(usersTable).set(data).where(eq(usersTable.id, id)).returning();
124
+ return updatedUser;
125
+ }
126
+ async delete(id) {
127
+ const [deletedUser] = await this.db.delete(usersTable).where(eq(usersTable.id, id)).returning();
128
+ return deletedUser;
129
+ }
130
+ async deleteAll() {
131
+ const adminRole = await this.db.select({ id: rolesTable.id }).from(rolesTable).where(eq(rolesTable.name, "admin")).limit(1);
132
+ if (adminRole.length === 0) {
133
+ const deletedUsers2 = await this.db.delete(usersTable).returning();
134
+ return deletedUsers2;
135
+ }
136
+ const deletedUsers = await this.db.delete(usersTable).where(ne(usersTable.roleId, adminRole[0].id)).returning();
137
+ return deletedUsers;
138
+ }
139
+ async getRoleNameById(userId) {
140
+ const [role] = await this.db.select({
141
+ roleName: rolesTable.name
142
+ }).from(usersTable).leftJoin(rolesTable, eq(usersTable.roleId, rolesTable.id)).where(eq(usersTable.id, userId));
143
+ return role.roleName;
144
+ }
145
+ async getUserPassword(email) {
146
+ const [user] = await this.db.select({
147
+ id: usersTable.id,
148
+ email: usersTable.email,
149
+ password: usersTable.password
150
+ }).from(usersTable).where(eq(usersTable.email, email)).limit(1);
151
+ return user.password;
152
+ }
153
+ async getUserPermissions(userId) {
154
+ const [user] = await this.db.select({ roleId: usersTable.roleId }).from(usersTable).where(eq(usersTable.id, userId)).limit(1);
155
+ if (!user || !user.roleId)
156
+ return [];
157
+ const userPermissions = await this.db.select({
158
+ name: permissionsTable.name
159
+ }).from(rolePermissionsTable).leftJoin(permissionsTable, eq(rolePermissionsTable.permissionId, permissionsTable.id)).where(eq(rolePermissionsTable.roleId, user.roleId));
160
+ return userPermissions.map((p) => p.name).filter((name) => name);
161
+ }
162
+ }
163
+ UserRepository = __legacyDecorateClassTS([
164
+ Repository()
165
+ ], UserRepository);
166
+ // src/users/UserValidator.ts
167
+ import { Injectable as Injectable3, t } from "najm-api";
168
+ import { parseSchema } from "@/shared";
169
+ import { userSchema } from "@/lib/validations";
170
+ class UserValidator {
171
+ userRepository;
172
+ encryptionService;
173
+ constructor(userRepository, encryptionService) {
174
+ this.userRepository = userRepository;
175
+ this.encryptionService = encryptionService;
176
+ }
177
+ async validateCreateUser(data) {
178
+ return parseSchema(userSchema, data);
179
+ }
180
+ async isEmailExists(email) {
181
+ const existingUser = await this.userRepository.getByEmail(email);
182
+ return !!existingUser;
183
+ }
184
+ async isPasswordValid(password, hashedPassword) {
185
+ const isPasswordValid = await this.encryptionService.comparePassword(password, hashedPassword);
186
+ return !!isPasswordValid;
187
+ }
188
+ async isUserExist(id) {
189
+ const existingUser = await this.userRepository.getById(id);
190
+ return !!existingUser;
191
+ }
192
+ async checkUserIdIsUnique(id) {
193
+ if (!id)
194
+ return;
195
+ const existingUser = await this.userRepository.getById(id);
196
+ if (existingUser) {
197
+ throw new Error(t("users.errors.idExists"));
198
+ }
199
+ }
200
+ async isCorrectPass(password) {
201
+ return password && typeof password === "string" && password.trim().length > 0;
202
+ }
203
+ async hasRole(userId, roles) {
204
+ const roleName = await this.userRepository.getRoleNameById(userId);
205
+ if (!roleName) {
206
+ throw Error(t("auth.errors.accessDenied"));
207
+ }
208
+ const hasRole = roles.some((item) => roleName.toLowerCase() === item.toLowerCase());
209
+ if (!hasRole) {
210
+ throw Error(t("auth.errors.accessDenied"));
211
+ }
212
+ return true;
213
+ }
214
+ async checkUserExistsByEmail(email) {
215
+ const user = await this.userRepository.getByEmail(email);
216
+ if (!user) {
217
+ throw new Error(t("auth.errors.invalidCredentials"));
218
+ }
219
+ return user;
220
+ }
221
+ async checkUserExists(id) {
222
+ const userExists = await this.isUserExist(id);
223
+ if (!userExists) {
224
+ throw new Error(t("users.errors.notFound"));
225
+ }
226
+ return userExists;
227
+ }
228
+ async checkEmailUnique(email, excludeId = null) {
229
+ if (!email)
230
+ return;
231
+ const existingUser = await this.userRepository.getByEmail(email);
232
+ if (existingUser && existingUser.id !== excludeId) {
233
+ throw new Error(t("auth.errors.emailExists"));
234
+ }
235
+ }
236
+ async checkEmailExists(email) {
237
+ const user = await this.userRepository.getByEmail(email);
238
+ if (!user) {
239
+ throw new Error(t("users.errors.notFound"));
240
+ }
241
+ return user;
242
+ }
243
+ async checkPasswordValid(password, hashedPassword) {
244
+ const isPasswordValid = await this.isPasswordValid(password, hashedPassword);
245
+ if (!isPasswordValid) {
246
+ throw new Error(t("auth.errors.invalidCredentials"));
247
+ }
248
+ }
249
+ }
250
+ UserValidator = __legacyDecorateClassTS([
251
+ Injectable3(),
252
+ __legacyMetadataTS("design:paramtypes", [
253
+ typeof UserRepository === "undefined" ? Object : UserRepository,
254
+ typeof EncryptionService === "undefined" ? Object : EncryptionService
255
+ ])
256
+ ], UserValidator);
257
+ // src/users/UserService.ts
258
+ import { Injectable as Injectable8, setLanguage, getCurrentLanguage, Transactional } from "najm-api";
259
+
260
+ // src/roles/RoleRepository.ts
261
+ import { rolesTable as rolesTable2 } from "@/database/schema";
262
+ import { eq as eq2 } from "drizzle-orm";
263
+ import { Repository as Repository2 } from "najm-api";
264
+ class RoleRepository {
265
+ async getAll() {
266
+ return await this.db.select().from(rolesTable2);
267
+ }
268
+ async getById(id) {
269
+ const [existingRole] = await this.db.select().from(rolesTable2).where(eq2(rolesTable2.id, id));
270
+ return existingRole;
271
+ }
272
+ async getByName(name) {
273
+ const [existingRole] = await this.db.select().from(rolesTable2).where(eq2(rolesTable2.name, name));
274
+ return existingRole;
275
+ }
276
+ async create(data) {
277
+ const [newRole] = await this.db.insert(rolesTable2).values(data).returning();
278
+ return newRole;
279
+ }
280
+ async update(id, data) {
281
+ const [updatedRole] = await this.db.update(rolesTable2).set(data).where(eq2(rolesTable2.id, id)).returning();
282
+ return updatedRole;
283
+ }
284
+ async delete(id) {
285
+ const [deletedRole] = await this.db.delete(rolesTable2).where(eq2(rolesTable2.id, id)).returning();
286
+ return deletedRole;
287
+ }
288
+ }
289
+ RoleRepository = __legacyDecorateClassTS([
290
+ Repository2()
291
+ ], RoleRepository);
292
+ // src/roles/RoleValidator.ts
293
+ import { Injectable as Injectable4, t as t2 } from "najm-api";
294
+ import { parseSchema as parseSchema2 } from "@/shared";
295
+ import { roleSchema } from "@/lib/validations";
296
+ class RoleValidator {
297
+ roleRepository;
298
+ constructor(roleRepository) {
299
+ this.roleRepository = roleRepository;
300
+ }
301
+ async validateCreateRole(data) {
302
+ return parseSchema2(roleSchema, data);
303
+ }
304
+ async isRoleNameExists(roleName) {
305
+ const existingRole = await this.roleRepository.getByName(roleName);
306
+ return !!existingRole;
307
+ }
308
+ async isRoleIdExists(id) {
309
+ const existingRole = await this.roleRepository.getById(id);
310
+ return !!existingRole;
311
+ }
312
+ async checkNameUnique(roleName, excludeId = null) {
313
+ if (!roleName)
314
+ return;
315
+ const existingRole = await this.roleRepository.getByName(roleName);
316
+ if (existingRole && existingRole.id !== excludeId) {
317
+ throw new Error(t2("roles.errors.exists"));
318
+ }
319
+ }
320
+ async checkRoleExists(id) {
321
+ const roleIdExists = await this.isRoleIdExists(id);
322
+ if (!roleIdExists) {
323
+ throw new Error(t2("roles.errors.notFound"));
324
+ }
325
+ }
326
+ async checkRoleExistsByName(roleName) {
327
+ const roleNameExists = await this.isRoleNameExists(roleName);
328
+ if (!roleNameExists) {
329
+ throw new Error(t2("roles.errors.notFound"));
330
+ }
331
+ }
332
+ async checkAdminRoleExists() {
333
+ const adminRole = await this.roleRepository.getByName("admin");
334
+ if (!adminRole) {
335
+ throw new Error(t2("users.errors.adminRoleNotFound"));
336
+ }
337
+ return adminRole;
338
+ }
339
+ }
340
+ RoleValidator = __legacyDecorateClassTS([
341
+ Injectable4(),
342
+ __legacyMetadataTS("design:paramtypes", [
343
+ typeof RoleRepository === "undefined" ? Object : RoleRepository
344
+ ])
345
+ ], RoleValidator);
346
+ // src/roles/RoleGuards.ts
347
+ import { Injectable as Injectable6, Headers as Headers2, createGuard, GuardParams, Ctx as Ctx2 } from "najm-api";
348
+
349
+ // src/tokens/TokenRepository.ts
350
+ import { tokensTable, usersTable as usersTable2, rolesTable as rolesTable3, permissionsTable as permissionsTable2, rolePermissionsTable as rolePermissionsTable2 } from "@/database/schema";
351
+ import { eq as eq3 } from "drizzle-orm";
352
+ import { Repository as Repository3 } from "najm-api";
353
+ class TokenRepository {
354
+ async storeRefreshToken(tokenData) {
355
+ return await this.db.insert(tokensTable).values(tokenData).onConflictDoUpdate({
356
+ target: tokensTable.userId,
357
+ set: {
358
+ token: tokenData.token,
359
+ expiresAt: tokenData.expiresAt
360
+ }
361
+ }).returning();
362
+ }
363
+ async getRefreshToken(userId) {
364
+ const [token] = await this.db.select().from(tokensTable).where(eq3(tokensTable.userId, userId));
365
+ return token?.token;
366
+ }
367
+ async revokeToken(userId) {
368
+ const [deletedToken] = await this.db.delete(tokensTable).where(eq3(tokensTable.userId, userId)).returning();
369
+ return deletedToken;
370
+ }
371
+ async isUserExists(userId) {
372
+ const [user] = await this.db.select({ id: usersTable2.id }).from(usersTable2).where(eq3(usersTable2.id, userId)).limit(1);
373
+ return !!user;
374
+ }
375
+ async getRoleNameById(userId) {
376
+ const [role] = await this.db.select({
377
+ roleName: rolesTable3.name
378
+ }).from(usersTable2).leftJoin(rolesTable3, eq3(usersTable2.roleId, rolesTable3.id)).where(eq3(usersTable2.id, userId)).limit(1);
379
+ return role?.roleName;
380
+ }
381
+ async getUserPermissions(userId) {
382
+ const [user] = await this.db.select({ roleId: usersTable2.roleId }).from(usersTable2).where(eq3(usersTable2.id, userId)).limit(1);
383
+ if (!user || !user.roleId)
384
+ return [];
385
+ const userPermissions = await this.db.select({
386
+ name: permissionsTable2.name
387
+ }).from(rolePermissionsTable2).leftJoin(permissionsTable2, eq3(rolePermissionsTable2.permissionId, permissionsTable2.id)).where(eq3(rolePermissionsTable2.roleId, user.roleId));
388
+ return userPermissions.map((p) => p.name).filter((name) => name);
389
+ }
390
+ async getUser(userId) {
391
+ const [user] = await this.db.select({
392
+ id: usersTable2.id,
393
+ email: usersTable2.email,
394
+ status: usersTable2.status,
395
+ roleId: usersTable2.roleId,
396
+ roleName: rolesTable3.name,
397
+ createdAt: usersTable2.createdAt,
398
+ updatedAt: usersTable2.updatedAt
399
+ }).from(usersTable2).leftJoin(rolesTable3, eq3(usersTable2.roleId, rolesTable3.id)).where(eq3(usersTable2.id, userId)).limit(1);
400
+ return user ? { ...user, role: user.roleName } : null;
401
+ }
402
+ }
403
+ TokenRepository = __legacyDecorateClassTS([
404
+ Repository3()
405
+ ], TokenRepository);
406
+ // src/tokens/TokenService.ts
407
+ import { t as t3 } from "najm-api";
408
+ import { getCookie, Injectable as Injectable5 } from "najm-api";
409
+ import jwt from "jsonwebtoken";
410
+ import { jwtDecode } from "jwt-decode";
411
+ import timestring2 from "timestring";
412
+ class TokenService {
413
+ tokenRepository;
414
+ accessSecretKey = process.env.JWT_ACCESS_SECRET;
415
+ accessExpiresIn = process.env.ACCESS_EXPIRES_IN;
416
+ refreshSecretKey = process.env.JWT_REFRESH_SECRET;
417
+ refreshExpiresIn = process.env.REFRESH_EXPIRES_IN;
418
+ constructor(tokenRepository) {
419
+ this.tokenRepository = tokenRepository;
420
+ }
421
+ extractAccessToken(authorization) {
422
+ if (authorization && authorization.startsWith("Bearer")) {
423
+ return authorization.split(" ")[1];
424
+ }
425
+ throw new Error(t3("auth.errors.tokenMissing"));
426
+ }
427
+ verifyAccessToken(token) {
428
+ try {
429
+ return jwt.verify(token, this.accessSecretKey);
430
+ } catch (error) {
431
+ throw new Error(t3("auth.errors.tokenVerificationFailed"));
432
+ }
433
+ }
434
+ verifyRefreshToken(token) {
435
+ try {
436
+ const decoded = jwt.verify(token, this.refreshSecretKey);
437
+ return decoded.userId;
438
+ } catch (error) {
439
+ throw new Error(t3("auth.errors.tokenVerificationFailed"));
440
+ }
441
+ }
442
+ async getUserIdByAccessToken(header) {
443
+ const token = this.extractAccessToken(header);
444
+ const decodedToken = this.verifyAccessToken(token);
445
+ const userId = decodedToken.userId;
446
+ const userExists = await this.tokenRepository.isUserExists(userId);
447
+ if (!userExists) {
448
+ throw new Error(t3("users.errors.notFound"));
449
+ }
450
+ return userId;
451
+ }
452
+ async storeRefreshToken(userId, refreshToken) {
453
+ const expireInSecond = timestring2(this.refreshExpiresIn, "s");
454
+ const tokenData = {
455
+ userId,
456
+ token: refreshToken,
457
+ expiresAt: new Date(Date.now() + expireInSecond * 1000).toISOString()
458
+ };
459
+ await this.tokenRepository.storeRefreshToken(tokenData);
460
+ }
461
+ getTokenExpire(token) {
462
+ return jwtDecode(token).exp;
463
+ }
464
+ generateAccessToken(data) {
465
+ const options = { expiresIn: this.accessExpiresIn };
466
+ return jwt.sign(data, this.accessSecretKey, options);
467
+ }
468
+ generateRefreshToken(data) {
469
+ const options = { expiresIn: this.refreshExpiresIn };
470
+ return jwt.sign(data, this.refreshSecretKey, options);
471
+ }
472
+ async generateTokens(userId) {
473
+ const tokenData = { userId };
474
+ const accessToken = await this.generateAccessToken(tokenData);
475
+ const refreshToken = await this.generateRefreshToken(tokenData);
476
+ const accessTokenExpiresAt = this.getTokenExpire(accessToken);
477
+ const refreshTokenExpiresAt = this.getTokenExpire(refreshToken);
478
+ await this.storeRefreshToken(userId, refreshToken);
479
+ return {
480
+ accessToken,
481
+ refreshToken,
482
+ accessTokenExpiresAt,
483
+ refreshTokenExpiresAt
484
+ };
485
+ }
486
+ async refreshTokens() {
487
+ const newRefreshToken = getCookie("refreshToken");
488
+ const userId = this.verifyRefreshToken(newRefreshToken);
489
+ const userExists = await this.tokenRepository.isUserExists(userId);
490
+ if (!userExists) {
491
+ throw new Error(t3("users.errors.notFound"));
492
+ }
493
+ const storedRefreshToken = await this.tokenRepository.getRefreshToken(userId);
494
+ if (newRefreshToken != storedRefreshToken) {
495
+ throw new Error(t3("auth.errors.refreshTokenInvalid"));
496
+ }
497
+ return await this.generateTokens(userId);
498
+ }
499
+ async revokeToken(userId) {
500
+ return await this.tokenRepository.revokeToken(userId);
501
+ }
502
+ async getUserPermissions(auth) {
503
+ if (!auth)
504
+ return;
505
+ const userId = await this.getUserIdByAccessToken(auth);
506
+ const permissions = await this.tokenRepository.getUserPermissions(userId);
507
+ return permissions;
508
+ }
509
+ async getUserRole(auth) {
510
+ if (!auth)
511
+ return;
512
+ const userId = await this.getUserIdByAccessToken(auth);
513
+ const roleName = await this.tokenRepository.getRoleNameById(userId);
514
+ return roleName;
515
+ }
516
+ async getUser(auth) {
517
+ if (!auth)
518
+ return;
519
+ const userId = await this.getUserIdByAccessToken(auth);
520
+ const user = await this.tokenRepository.getUser(userId);
521
+ if (!user)
522
+ return null;
523
+ return user;
524
+ }
525
+ async storeUserInCache(auth, ctx) {
526
+ const cachedUser = ctx.get("user");
527
+ if (cachedUser)
528
+ return cachedUser;
529
+ const user = await this.getUser(auth);
530
+ if (user) {
531
+ ctx.set("user", user);
532
+ }
533
+ return user;
534
+ }
535
+ }
536
+ TokenService = __legacyDecorateClassTS([
537
+ Injectable5(),
538
+ __legacyMetadataTS("design:paramtypes", [
539
+ typeof TokenRepository === "undefined" ? Object : TokenRepository
540
+ ])
541
+ ], TokenService);
542
+ // src/roles/RoleGuards.ts
543
+ var ROLES = {
544
+ ADMIN: "admin",
545
+ PRINCIPAL: "principal",
546
+ ACCOUNTING: "accounting",
547
+ SECRETARY: "secretary",
548
+ TEACHER: "teacher",
549
+ STUDENT: "student",
550
+ PARENT: "parent"
551
+ };
552
+ var ROLE_GROUPS = {
553
+ ADMINISTRATORS: [ROLES.ADMIN, ROLES.PRINCIPAL],
554
+ FINANCIAL: [ROLES.ADMIN, ROLES.ACCOUNTING],
555
+ STAFF: [ROLES.ADMIN, ROLES.PRINCIPAL, ROLES.ACCOUNTING, ROLES.SECRETARY, ROLES.TEACHER],
556
+ END_USERS: [ROLES.STUDENT, ROLES.PARENT],
557
+ ALL: [ROLES.ADMIN, ROLES.PRINCIPAL, ROLES.ACCOUNTING, ROLES.SECRETARY, ROLES.TEACHER, ROLES.STUDENT, ROLES.PARENT]
558
+ };
559
+
560
+ class RoleChecker {
561
+ isInGroup(userRole, group) {
562
+ return group.includes(userRole?.toLowerCase());
563
+ }
564
+ isAdministrator(userRole) {
565
+ return this.isInGroup(userRole, ROLE_GROUPS.ADMINISTRATORS);
566
+ }
567
+ isStaff(userRole) {
568
+ return this.isInGroup(userRole, ROLE_GROUPS.STAFF);
569
+ }
570
+ hasAnyRole(userRole, roles) {
571
+ return roles.includes(userRole?.toLowerCase());
572
+ }
573
+ hasExactRole(userRole, requiredRole) {
574
+ return userRole?.toLowerCase() === requiredRole?.toLowerCase();
575
+ }
576
+ }
577
+ RoleChecker = __legacyDecorateClassTS([
578
+ Injectable6()
579
+ ], RoleChecker);
580
+
581
+ class RoleGuards {
582
+ roleChecker;
583
+ tokenService;
584
+ constructor(roleChecker, tokenService) {
585
+ this.roleChecker = roleChecker;
586
+ this.tokenService = tokenService;
587
+ }
588
+ async isAuth(auth, ctx) {
589
+ const user = await this.tokenService.storeUserInCache(auth, ctx);
590
+ return !!user;
591
+ }
592
+ async hasRoles(auth, ctx, roles) {
593
+ try {
594
+ const user = await this.tokenService.storeUserInCache(auth, ctx);
595
+ if (!user?.role)
596
+ return false;
597
+ const roleArray = Array.isArray(roles) ? roles : [roles];
598
+ return this.roleChecker.hasAnyRole(user.role, roleArray);
599
+ } catch {
600
+ return false;
601
+ }
602
+ }
603
+ }
604
+ __legacyDecorateClassTS([
605
+ __legacyDecorateParamTS(0, Headers2("authorization")),
606
+ __legacyDecorateParamTS(1, Ctx2()),
607
+ __legacyMetadataTS("design:type", Function),
608
+ __legacyMetadataTS("design:paramtypes", [
609
+ Object,
610
+ Object
611
+ ]),
612
+ __legacyMetadataTS("design:returntype", Object)
613
+ ], RoleGuards.prototype, "isAuth", null);
614
+ __legacyDecorateClassTS([
615
+ __legacyDecorateParamTS(0, Headers2("authorization")),
616
+ __legacyDecorateParamTS(1, Ctx2()),
617
+ __legacyDecorateParamTS(2, GuardParams()),
618
+ __legacyMetadataTS("design:type", Function),
619
+ __legacyMetadataTS("design:paramtypes", [
620
+ Object,
621
+ Object,
622
+ Object
623
+ ]),
624
+ __legacyMetadataTS("design:returntype", Object)
625
+ ], RoleGuards.prototype, "hasRoles", null);
626
+ RoleGuards = __legacyDecorateClassTS([
627
+ Injectable6(),
628
+ __legacyMetadataTS("design:paramtypes", [
629
+ typeof RoleChecker === "undefined" ? Object : RoleChecker,
630
+ typeof TokenService === "undefined" ? Object : TokenService
631
+ ])
632
+ ], RoleGuards);
633
+ var isAdmin = () => Role("admin");
634
+ var isPrincipal = () => Role("principal");
635
+ var isAccounting = () => Role("accounting");
636
+ var isSecretary = () => Role("secretary");
637
+ var isTeacher = () => Role("teacher");
638
+ var isParent = () => Role("parent");
639
+ var isStudent = () => Role("student");
640
+ var isAdministrator = () => Role("admin", "principal");
641
+ var isFinancial = () => Role("admin", "accounting");
642
+ var isStaff = () => Role("admin", "principal", "accounting", "secretary", "teacher");
643
+ var isAuth = createGuard(RoleGuards, "isAuth");
644
+ var Role = (...roles) => createGuard(RoleGuards, "hasRoles")(...roles);
645
+ // src/roles/RoleController.ts
646
+ import { Controller, Get, Post, Put, Delete, Params, Body, t as t4 } from "najm-api";
647
+
648
+ // src/roles/RoleService.ts
649
+ import { Injectable as Injectable7 } from "najm-api";
650
+ class RoleService {
651
+ roleRepository;
652
+ roleValidator;
653
+ constructor(roleRepository, roleValidator) {
654
+ this.roleRepository = roleRepository;
655
+ this.roleValidator = roleValidator;
656
+ }
657
+ async getAll() {
658
+ return await this.roleRepository.getAll();
659
+ }
660
+ async getById(id) {
661
+ await this.roleValidator.checkRoleExists(id);
662
+ return await this.roleRepository.getById(id);
663
+ }
664
+ async getByName(name) {
665
+ return await this.roleRepository.getByName(name);
666
+ }
667
+ async create(data) {
668
+ await this.roleValidator.validateCreateRole(data);
669
+ await this.roleValidator.checkNameUnique(data.name);
670
+ return await this.roleRepository.create(data);
671
+ }
672
+ async update(id, data) {
673
+ await this.roleValidator.checkRoleExists(id);
674
+ await this.roleValidator.checkNameUnique(data.name, id);
675
+ return await this.roleRepository.update(id, data);
676
+ }
677
+ async delete(id) {
678
+ await this.roleValidator.checkRoleExists(id);
679
+ return await this.roleRepository.delete(id);
680
+ }
681
+ async seedDefaultRoles(defaultRoles) {
682
+ const rolesToCreate = [];
683
+ for (const role of defaultRoles) {
684
+ const exists = await this.roleValidator.isRoleNameExists(role.name);
685
+ if (!exists) {
686
+ rolesToCreate.push(role);
687
+ }
688
+ }
689
+ const createdRoles = await Promise.all(rolesToCreate.map((role) => this.roleRepository.create(role)));
690
+ return createdRoles;
691
+ }
692
+ async getRoleIdByName(name) {
693
+ const teacherRole = await this.getByName(name);
694
+ return teacherRole?.id;
695
+ }
696
+ }
697
+ RoleService = __legacyDecorateClassTS([
698
+ Injectable7(),
699
+ __legacyMetadataTS("design:paramtypes", [
700
+ typeof RoleRepository === "undefined" ? Object : RoleRepository,
701
+ typeof RoleValidator === "undefined" ? Object : RoleValidator
702
+ ])
703
+ ], RoleService);
704
+
705
+ // src/roles/RoleController.ts
706
+ class RoleController {
707
+ roleService;
708
+ constructor(roleService) {
709
+ this.roleService = roleService;
710
+ }
711
+ async getRoles() {
712
+ const roles = await this.roleService.getAll();
713
+ return {
714
+ data: roles,
715
+ message: t4("roles.success.retrieved"),
716
+ status: "success"
717
+ };
718
+ }
719
+ async getRole(id) {
720
+ const role = await this.roleService.getById(id);
721
+ return {
722
+ data: role,
723
+ message: t4("roles.success.retrieved"),
724
+ status: "success"
725
+ };
726
+ }
727
+ async createRole(body) {
728
+ const newRole = await this.roleService.create(body);
729
+ return {
730
+ data: newRole,
731
+ message: t4("roles.success.created"),
732
+ status: "success"
733
+ };
734
+ }
735
+ async updateRole(id, body) {
736
+ const updatedRole = await this.roleService.update(id, body);
737
+ return {
738
+ data: updatedRole,
739
+ message: t4("roles.success.updated"),
740
+ status: "success"
741
+ };
742
+ }
743
+ async deleteRole(id) {
744
+ const result = await this.roleService.delete(id);
745
+ return {
746
+ data: result,
747
+ message: t4("roles.success.deleted"),
748
+ status: "success"
749
+ };
750
+ }
751
+ }
752
+ __legacyDecorateClassTS([
753
+ Get(),
754
+ isAdmin(),
755
+ __legacyMetadataTS("design:type", Function),
756
+ __legacyMetadataTS("design:paramtypes", []),
757
+ __legacyMetadataTS("design:returntype", Promise)
758
+ ], RoleController.prototype, "getRoles", null);
759
+ __legacyDecorateClassTS([
760
+ Get("/:id"),
761
+ isAdmin(),
762
+ __legacyDecorateParamTS(0, Params("id")),
763
+ __legacyMetadataTS("design:type", Function),
764
+ __legacyMetadataTS("design:paramtypes", [
765
+ Object
766
+ ]),
767
+ __legacyMetadataTS("design:returntype", Promise)
768
+ ], RoleController.prototype, "getRole", null);
769
+ __legacyDecorateClassTS([
770
+ Post(),
771
+ isAdmin(),
772
+ __legacyDecorateParamTS(0, Body()),
773
+ __legacyMetadataTS("design:type", Function),
774
+ __legacyMetadataTS("design:paramtypes", [
775
+ Object
776
+ ]),
777
+ __legacyMetadataTS("design:returntype", Promise)
778
+ ], RoleController.prototype, "createRole", null);
779
+ __legacyDecorateClassTS([
780
+ Put("/:id"),
781
+ isAdmin(),
782
+ __legacyDecorateParamTS(0, Params("id")),
783
+ __legacyDecorateParamTS(1, Body()),
784
+ __legacyMetadataTS("design:type", Function),
785
+ __legacyMetadataTS("design:paramtypes", [
786
+ Object,
787
+ Object
788
+ ]),
789
+ __legacyMetadataTS("design:returntype", Promise)
790
+ ], RoleController.prototype, "updateRole", null);
791
+ __legacyDecorateClassTS([
792
+ Delete("/:id"),
793
+ isAdmin(),
794
+ __legacyDecorateParamTS(0, Params("id")),
795
+ __legacyMetadataTS("design:type", Function),
796
+ __legacyMetadataTS("design:paramtypes", [
797
+ Object
798
+ ]),
799
+ __legacyMetadataTS("design:returntype", Promise)
800
+ ], RoleController.prototype, "deleteRole", null);
801
+ RoleController = __legacyDecorateClassTS([
802
+ Controller("/roles"),
803
+ __legacyMetadataTS("design:paramtypes", [
804
+ typeof RoleService === "undefined" ? Object : RoleService
805
+ ])
806
+ ], RoleController);
807
+ // src/users/UserService.ts
808
+ import { nanoid } from "nanoid";
809
+ import { clean } from "@/shared";
810
+ class UserService {
811
+ roleValidator;
812
+ roleService;
813
+ userRepository;
814
+ userValidator;
815
+ encryptionService;
816
+ constructor(roleValidator, roleService, userRepository, userValidator, encryptionService) {
817
+ this.roleValidator = roleValidator;
818
+ this.roleService = roleService;
819
+ this.userRepository = userRepository;
820
+ this.userValidator = userValidator;
821
+ this.encryptionService = encryptionService;
822
+ }
823
+ sanitizeUser(user) {
824
+ if (!user)
825
+ return user;
826
+ const { password, ...sanitizedUser } = user;
827
+ return sanitizedUser;
828
+ }
829
+ sanitizeUsers(users) {
830
+ return users.map((user) => this.sanitizeUser(user));
831
+ }
832
+ async resolveUserRole(roleId, roleName) {
833
+ if (roleId) {
834
+ await this.roleValidator.checkRoleExists(roleId);
835
+ return roleId;
836
+ }
837
+ if (roleName) {
838
+ const roleByName = await this.roleService.getByName(roleName);
839
+ if (roleByName) {
840
+ return roleByName.id;
841
+ }
842
+ throw new Error(`Role '${roleName}' not found`);
843
+ }
844
+ const defaultRole = await this.roleService.getByName("Student");
845
+ return defaultRole.id;
846
+ }
847
+ async getAll() {
848
+ const users = await this.userRepository.getAll();
849
+ return this.sanitizeUsers(users);
850
+ }
851
+ async getById(id) {
852
+ await this.userValidator.checkUserExists(id);
853
+ const user = await this.userRepository.getById(id);
854
+ return this.sanitizeUser(user);
855
+ }
856
+ async getByEmail(email) {
857
+ const user = await this.userValidator.checkUserExistsByEmail(email);
858
+ return this.sanitizeUser(user);
859
+ }
860
+ async create(data) {
861
+ const { id, email, image, emailVerified, password, roleId, role } = data;
862
+ let userId = id || nanoid(5);
863
+ let pass = password || "12345678";
864
+ await this.userValidator.checkEmailUnique(data.email);
865
+ await this.userValidator.checkUserIdIsUnique(id);
866
+ const hashedPassword = await this.encryptionService.hashPassword(pass);
867
+ const resolvedRoleId = await this.resolveUserRole(roleId, role);
868
+ const userDetails = {
869
+ id: userId,
870
+ email,
871
+ image,
872
+ password: hashedPassword,
873
+ roleId: resolvedRoleId,
874
+ emailVerified,
875
+ status: "pending"
876
+ };
877
+ await this.userValidator.validateCreateUser(userDetails);
878
+ const newUser = await this.userRepository.create(userDetails);
879
+ return this.sanitizeUser(newUser);
880
+ }
881
+ async update(id, data) {
882
+ const { password, image } = data;
883
+ await this.userValidator.checkUserExists(id);
884
+ await this.userValidator.checkEmailUnique(data.email, id);
885
+ const currentUser = await this.userRepository.getById(id);
886
+ const hashedPassword = await this.encryptionService.hashPassword(password);
887
+ const updateData = {
888
+ ...data,
889
+ image,
890
+ ...hashedPassword && { password: hashedPassword }
891
+ };
892
+ const cleanedUpdateData = clean(updateData);
893
+ const updatedUser = await this.userRepository.update(id, cleanedUpdateData);
894
+ return this.sanitizeUser(updatedUser);
895
+ }
896
+ async delete(id) {
897
+ await this.userValidator.checkUserExists(id);
898
+ const user = await this.userRepository.delete(id);
899
+ return this.sanitizeUser(user);
900
+ }
901
+ async deleteAll() {
902
+ const deletedUsers = await this.userRepository.deleteAll();
903
+ return this.sanitizeUsers(deletedUsers);
904
+ }
905
+ async getRoleName(id) {
906
+ await this.userValidator.checkUserExists(id);
907
+ return await this.userRepository.getRoleNameById(id);
908
+ }
909
+ async getPassword(email) {
910
+ await this.userValidator.checkUserExistsByEmail(email);
911
+ return await this.userRepository.getUserPassword(email);
912
+ }
913
+ async assignRole(id, roleId, roleName) {
914
+ await this.userValidator.checkUserExists(id);
915
+ const resolvedRoleId = await this.resolveUserRole(roleId, roleName);
916
+ const updatedUser = await this.userRepository.update(id, { roleId: resolvedRoleId });
917
+ return this.sanitizeUser(updatedUser);
918
+ }
919
+ async removeRole(id) {
920
+ await this.userValidator.checkUserExists(id);
921
+ const updatedUser = await this.userRepository.update(id, { roleId: null });
922
+ return this.sanitizeUser(updatedUser);
923
+ }
924
+ async seedAdminUser() {
925
+ const email = "admin@admin.com";
926
+ const adminRole = await this.roleValidator.checkAdminRoleExists();
927
+ const existingUser = await this.userRepository.getByEmail(email);
928
+ if (existingUser) {
929
+ await this.delete(existingUser.id);
930
+ }
931
+ const newAdminUser = await this.create({
932
+ id: "USR00",
933
+ name: "System Administrator",
934
+ email,
935
+ password: "12345678",
936
+ image: null,
937
+ roleId: adminRole.id,
938
+ status: "active",
939
+ emailVerified: true
940
+ });
941
+ return this.sanitizeUser(newAdminUser);
942
+ }
943
+ async updateLang(language) {
944
+ setLanguage(language);
945
+ return language;
946
+ }
947
+ async getLang() {
948
+ return getCurrentLanguage();
949
+ }
950
+ }
951
+ __legacyDecorateClassTS([
952
+ Transactional(),
953
+ __legacyMetadataTS("design:type", Function),
954
+ __legacyMetadataTS("design:paramtypes", [
955
+ Object
956
+ ]),
957
+ __legacyMetadataTS("design:returntype", Promise)
958
+ ], UserService.prototype, "create", null);
959
+ UserService = __legacyDecorateClassTS([
960
+ Injectable8(),
961
+ __legacyMetadataTS("design:paramtypes", [
962
+ typeof RoleValidator === "undefined" ? Object : RoleValidator,
963
+ typeof RoleService === "undefined" ? Object : RoleService,
964
+ typeof UserRepository === "undefined" ? Object : UserRepository,
965
+ typeof UserValidator === "undefined" ? Object : UserValidator,
966
+ typeof EncryptionService === "undefined" ? Object : EncryptionService
967
+ ])
968
+ ], UserService);
969
+ // src/users/UserController.ts
970
+ import { Controller as Controller2, Get as Get2, Post as Post2, Put as Put2, Delete as Delete2, Params as Params2, Body as Body2, t as t5 } from "najm-api";
971
+ import { isAdmin as isAdmin2, isAuth as isAuth2 } from "@/roles/RoleGuards";
972
+ class UserController {
973
+ userService;
974
+ constructor(userService) {
975
+ this.userService = userService;
976
+ }
977
+ async getUsers() {
978
+ const users = await this.userService.getAll();
979
+ return {
980
+ data: users,
981
+ message: t5("users.success.retrieved"),
982
+ status: "success"
983
+ };
984
+ }
985
+ async getLang() {
986
+ const language = await this.userService.getLang();
987
+ return {
988
+ data: { language },
989
+ message: t5("users.success.retrieved"),
990
+ status: "success"
991
+ };
992
+ }
993
+ async updateLang(language) {
994
+ const data = await this.userService.updateLang(language);
995
+ return {
996
+ data,
997
+ message: t5("users.success.updated"),
998
+ status: "success"
999
+ };
1000
+ }
1001
+ async getUser(id) {
1002
+ const user = await this.userService.getById(id);
1003
+ return {
1004
+ data: user,
1005
+ message: t5("users.success.retrieved"),
1006
+ status: "success"
1007
+ };
1008
+ }
1009
+ async getByEmail(email) {
1010
+ const user = await this.userService.getByEmail(email);
1011
+ return {
1012
+ data: user,
1013
+ message: t5("users.success.retrieved"),
1014
+ status: "success"
1015
+ };
1016
+ }
1017
+ async getRole(userId) {
1018
+ const role = await this.userService.getRoleName(userId);
1019
+ return {
1020
+ data: role,
1021
+ message: t5("users.success.retrieved"),
1022
+ status: "success"
1023
+ };
1024
+ }
1025
+ async create(body) {
1026
+ const newUser = await this.userService.create(body);
1027
+ return {
1028
+ data: newUser,
1029
+ message: t5("users.success.created"),
1030
+ status: "success"
1031
+ };
1032
+ }
1033
+ async update(id, body) {
1034
+ const updatedUser = await this.userService.update(id, body);
1035
+ return {
1036
+ data: updatedUser,
1037
+ message: t5("users.success.updated"),
1038
+ status: "success"
1039
+ };
1040
+ }
1041
+ async delete(id) {
1042
+ const result = await this.userService.delete(id);
1043
+ return {
1044
+ data: result,
1045
+ message: t5("users.success.deleted"),
1046
+ status: "success"
1047
+ };
1048
+ }
1049
+ async deleteAll() {
1050
+ const result = await this.userService.deleteAll();
1051
+ return {
1052
+ data: result,
1053
+ message: t5("users.success.allDeleted"),
1054
+ status: "success"
1055
+ };
1056
+ }
1057
+ async assignRole(userId, roleId) {
1058
+ await this.userService.assignRole(userId, roleId);
1059
+ return {
1060
+ message: t5("users.success.updated"),
1061
+ status: "success"
1062
+ };
1063
+ }
1064
+ async removeRole(userId) {
1065
+ await this.userService.removeRole(userId);
1066
+ return {
1067
+ message: t5("users.success.updated"),
1068
+ status: "success"
1069
+ };
1070
+ }
1071
+ }
1072
+ __legacyDecorateClassTS([
1073
+ Get2(),
1074
+ isAdmin2(),
1075
+ __legacyMetadataTS("design:type", Function),
1076
+ __legacyMetadataTS("design:paramtypes", []),
1077
+ __legacyMetadataTS("design:returntype", Promise)
1078
+ ], UserController.prototype, "getUsers", null);
1079
+ __legacyDecorateClassTS([
1080
+ Get2("/lang"),
1081
+ isAuth2(),
1082
+ __legacyMetadataTS("design:type", Function),
1083
+ __legacyMetadataTS("design:paramtypes", []),
1084
+ __legacyMetadataTS("design:returntype", Promise)
1085
+ ], UserController.prototype, "getLang", null);
1086
+ __legacyDecorateClassTS([
1087
+ Post2("/lang/:language"),
1088
+ isAuth2(),
1089
+ __legacyDecorateParamTS(0, Params2("language")),
1090
+ __legacyMetadataTS("design:type", Function),
1091
+ __legacyMetadataTS("design:paramtypes", [
1092
+ Object
1093
+ ]),
1094
+ __legacyMetadataTS("design:returntype", Promise)
1095
+ ], UserController.prototype, "updateLang", null);
1096
+ __legacyDecorateClassTS([
1097
+ Get2("/:id"),
1098
+ isAdmin2(),
1099
+ __legacyDecorateParamTS(0, Params2("id")),
1100
+ __legacyMetadataTS("design:type", Function),
1101
+ __legacyMetadataTS("design:paramtypes", [
1102
+ Object
1103
+ ]),
1104
+ __legacyMetadataTS("design:returntype", Promise)
1105
+ ], UserController.prototype, "getUser", null);
1106
+ __legacyDecorateClassTS([
1107
+ Get2("/email/:email"),
1108
+ isAdmin2(),
1109
+ __legacyDecorateParamTS(0, Params2("email")),
1110
+ __legacyMetadataTS("design:type", Function),
1111
+ __legacyMetadataTS("design:paramtypes", [
1112
+ Object
1113
+ ]),
1114
+ __legacyMetadataTS("design:returntype", Promise)
1115
+ ], UserController.prototype, "getByEmail", null);
1116
+ __legacyDecorateClassTS([
1117
+ Get2("/role/:userId"),
1118
+ isAdmin2(),
1119
+ __legacyDecorateParamTS(0, Params2("userId")),
1120
+ __legacyMetadataTS("design:type", Function),
1121
+ __legacyMetadataTS("design:paramtypes", [
1122
+ Object
1123
+ ]),
1124
+ __legacyMetadataTS("design:returntype", Promise)
1125
+ ], UserController.prototype, "getRole", null);
1126
+ __legacyDecorateClassTS([
1127
+ Post2(),
1128
+ isAdmin2(),
1129
+ __legacyDecorateParamTS(0, Body2()),
1130
+ __legacyMetadataTS("design:type", Function),
1131
+ __legacyMetadataTS("design:paramtypes", [
1132
+ Object
1133
+ ]),
1134
+ __legacyMetadataTS("design:returntype", Promise)
1135
+ ], UserController.prototype, "create", null);
1136
+ __legacyDecorateClassTS([
1137
+ Put2("/:id"),
1138
+ isAdmin2(),
1139
+ __legacyDecorateParamTS(0, Params2("id")),
1140
+ __legacyDecorateParamTS(1, Body2()),
1141
+ __legacyMetadataTS("design:type", Function),
1142
+ __legacyMetadataTS("design:paramtypes", [
1143
+ Object,
1144
+ Object
1145
+ ]),
1146
+ __legacyMetadataTS("design:returntype", Promise)
1147
+ ], UserController.prototype, "update", null);
1148
+ __legacyDecorateClassTS([
1149
+ Delete2("/:id"),
1150
+ isAdmin2(),
1151
+ __legacyDecorateParamTS(0, Params2("id")),
1152
+ __legacyMetadataTS("design:type", Function),
1153
+ __legacyMetadataTS("design:paramtypes", [
1154
+ Object
1155
+ ]),
1156
+ __legacyMetadataTS("design:returntype", Promise)
1157
+ ], UserController.prototype, "delete", null);
1158
+ __legacyDecorateClassTS([
1159
+ Delete2(),
1160
+ isAdmin2(),
1161
+ __legacyMetadataTS("design:type", Function),
1162
+ __legacyMetadataTS("design:paramtypes", []),
1163
+ __legacyMetadataTS("design:returntype", Promise)
1164
+ ], UserController.prototype, "deleteAll", null);
1165
+ __legacyDecorateClassTS([
1166
+ Post2("/assign/:userId/:roleId"),
1167
+ isAdmin2(),
1168
+ __legacyDecorateParamTS(0, Params2("userId")),
1169
+ __legacyDecorateParamTS(1, Params2("roleId")),
1170
+ __legacyMetadataTS("design:type", Function),
1171
+ __legacyMetadataTS("design:paramtypes", [
1172
+ Object,
1173
+ Object
1174
+ ]),
1175
+ __legacyMetadataTS("design:returntype", Promise)
1176
+ ], UserController.prototype, "assignRole", null);
1177
+ __legacyDecorateClassTS([
1178
+ Delete2("/remove/:userId"),
1179
+ isAdmin2(),
1180
+ __legacyDecorateParamTS(0, Params2("userId")),
1181
+ __legacyMetadataTS("design:type", Function),
1182
+ __legacyMetadataTS("design:paramtypes", [
1183
+ Object
1184
+ ]),
1185
+ __legacyMetadataTS("design:returntype", Promise)
1186
+ ], UserController.prototype, "removeRole", null);
1187
+ UserController = __legacyDecorateClassTS([
1188
+ Controller2("/users"),
1189
+ __legacyMetadataTS("design:paramtypes", [
1190
+ typeof UserService === "undefined" ? Object : UserService
1191
+ ])
1192
+ ], UserController);
1193
+ // src/auth/AuthService.ts
1194
+ class AuthService {
1195
+ tokenService;
1196
+ userService;
1197
+ userValidator;
1198
+ cookieService;
1199
+ constructor(tokenService, userService, userValidator, cookieService) {
1200
+ this.tokenService = tokenService;
1201
+ this.userService = userService;
1202
+ this.userValidator = userValidator;
1203
+ this.cookieService = cookieService;
1204
+ }
1205
+ async registerUser(body) {
1206
+ return await this.userService.create(body);
1207
+ }
1208
+ async loginUser(body) {
1209
+ const { email, password } = body;
1210
+ if (!email || !password) {
1211
+ throw new Error(t6("auth.errors.invalidCredentials"));
1212
+ }
1213
+ const existingPassword = await this.userService.getPassword(email);
1214
+ const { id } = await this.userService.getByEmail(email);
1215
+ await this.userValidator.checkPasswordValid(password, existingPassword);
1216
+ const data = await this.tokenService.generateTokens(id);
1217
+ this.cookieService.setRefreshCookie(data.refreshToken);
1218
+ return data;
1219
+ }
1220
+ async refreshTokens() {
1221
+ const data = await this.tokenService.refreshTokens();
1222
+ this.cookieService.setRefreshCookie(data.refreshToken);
1223
+ return data;
1224
+ }
1225
+ async logoutUser(userId) {
1226
+ await this.userValidator.checkUserExists(userId);
1227
+ await this.tokenService.revokeToken(userId);
1228
+ this.cookieService.clearRefreshCookie();
1229
+ return { data: null, message: t6("auth.success.logout") };
1230
+ }
1231
+ async getUserProfile(userData) {
1232
+ const lang = getCurrentLanguage2();
1233
+ return {
1234
+ ...userData,
1235
+ language: lang
1236
+ };
1237
+ }
1238
+ async forgotPassword(email) {}
1239
+ }
1240
+ AuthService = __legacyDecorateClassTS([
1241
+ Injectable9(),
1242
+ __legacyMetadataTS("design:paramtypes", [
1243
+ typeof TokenService === "undefined" ? Object : TokenService,
1244
+ typeof UserService === "undefined" ? Object : UserService,
1245
+ typeof UserValidator === "undefined" ? Object : UserValidator,
1246
+ typeof CookieService === "undefined" ? Object : CookieService
1247
+ ])
1248
+ ], AuthService);
1249
+
1250
+ // src/auth/AuthController.ts
1251
+ import { isAuth as isAuth3 } from "@/roles/RoleGuards";
1252
+ class AuthController {
1253
+ authService;
1254
+ constructor(authService) {
1255
+ this.authService = authService;
1256
+ }
1257
+ async registerUser(body) {
1258
+ const data = await this.authService.registerUser(body);
1259
+ return {
1260
+ data,
1261
+ message: t7("auth.success.register"),
1262
+ status: "success"
1263
+ };
1264
+ }
1265
+ async loginUser(body) {
1266
+ const data = await this.authService.loginUser(body);
1267
+ return {
1268
+ data,
1269
+ message: t7("auth.success.login"),
1270
+ status: "success"
1271
+ };
1272
+ }
1273
+ async refreshTokens() {
1274
+ const data = await this.authService.refreshTokens();
1275
+ return {
1276
+ data,
1277
+ message: t7("auth.success.tokenRefreshed"),
1278
+ status: "success"
1279
+ };
1280
+ }
1281
+ async logoutUser(id) {
1282
+ const data = await this.authService.logoutUser(id);
1283
+ return {
1284
+ data,
1285
+ message: t7("auth.success.logout"),
1286
+ status: "success"
1287
+ };
1288
+ }
1289
+ async userProfile(user) {
1290
+ const data = await this.authService.getUserProfile(user);
1291
+ return {
1292
+ data,
1293
+ message: t7("users.success.retrieved"),
1294
+ status: "success"
1295
+ };
1296
+ }
1297
+ async forgotPassword(body) {
1298
+ const data = await this.authService.forgotPassword(body.email);
1299
+ return {
1300
+ data,
1301
+ message: t7("auth.success.passwordReset"),
1302
+ status: "success"
1303
+ };
1304
+ }
1305
+ }
1306
+ __legacyDecorateClassTS([
1307
+ Post3("/register"),
1308
+ __legacyDecorateParamTS(0, Body3()),
1309
+ __legacyMetadataTS("design:type", Function),
1310
+ __legacyMetadataTS("design:paramtypes", [
1311
+ Object
1312
+ ]),
1313
+ __legacyMetadataTS("design:returntype", Promise)
1314
+ ], AuthController.prototype, "registerUser", null);
1315
+ __legacyDecorateClassTS([
1316
+ Post3("/login"),
1317
+ __legacyDecorateParamTS(0, Body3()),
1318
+ __legacyMetadataTS("design:type", Function),
1319
+ __legacyMetadataTS("design:paramtypes", [
1320
+ Object
1321
+ ]),
1322
+ __legacyMetadataTS("design:returntype", Promise)
1323
+ ], AuthController.prototype, "loginUser", null);
1324
+ __legacyDecorateClassTS([
1325
+ Get3("/refresh"),
1326
+ __legacyMetadataTS("design:type", Function),
1327
+ __legacyMetadataTS("design:paramtypes", []),
1328
+ __legacyMetadataTS("design:returntype", Promise)
1329
+ ], AuthController.prototype, "refreshTokens", null);
1330
+ __legacyDecorateClassTS([
1331
+ Get3("/logout/:id"),
1332
+ __legacyDecorateParamTS(0, Params3("id")),
1333
+ __legacyMetadataTS("design:type", Function),
1334
+ __legacyMetadataTS("design:paramtypes", [
1335
+ Object
1336
+ ]),
1337
+ __legacyMetadataTS("design:returntype", Promise)
1338
+ ], AuthController.prototype, "logoutUser", null);
1339
+ __legacyDecorateClassTS([
1340
+ Get3("/me"),
1341
+ isAuth3(),
1342
+ __legacyDecorateParamTS(0, User2()),
1343
+ __legacyMetadataTS("design:type", Function),
1344
+ __legacyMetadataTS("design:paramtypes", [
1345
+ Object
1346
+ ]),
1347
+ __legacyMetadataTS("design:returntype", Promise)
1348
+ ], AuthController.prototype, "userProfile", null);
1349
+ __legacyDecorateClassTS([
1350
+ Post3("/forgot-password"),
1351
+ isAuth3(),
1352
+ __legacyDecorateParamTS(0, Body3()),
1353
+ __legacyMetadataTS("design:type", Function),
1354
+ __legacyMetadataTS("design:paramtypes", [
1355
+ Object
1356
+ ]),
1357
+ __legacyMetadataTS("design:returntype", Promise)
1358
+ ], AuthController.prototype, "forgotPassword", null);
1359
+ AuthController = __legacyDecorateClassTS([
1360
+ Controller3("/auth"),
1361
+ __legacyMetadataTS("design:paramtypes", [
1362
+ typeof AuthService === "undefined" ? Object : AuthService
1363
+ ])
1364
+ ], AuthController);
1365
+ // src/lib/ENUMS.ts
1366
+ var ENUMS = {
1367
+ userType: {
1368
+ values: ["admin", "teacher", "student", "parent"],
1369
+ translationKey: "enums.userType"
1370
+ },
1371
+ userStatus: {
1372
+ values: ["active", "inactive", "pending"],
1373
+ translationKey: "enums.userStatus"
1374
+ },
1375
+ tokenStatus: {
1376
+ values: ["active", "revoked", "expired"],
1377
+ translationKey: "enums.tokenStatus"
1378
+ },
1379
+ tokenType: {
1380
+ values: ["access", "refresh"],
1381
+ translationKey: "enums.tokenType"
1382
+ },
1383
+ fileStatus: {
1384
+ values: ["active", "deleted", "archived"],
1385
+ translationKey: "enums.fileStatus"
1386
+ },
1387
+ gender: {
1388
+ values: ["M", "F"],
1389
+ translationKey: "common.gender"
1390
+ },
1391
+ studentStatus: {
1392
+ values: ["active", "inactive", "graduated", "transferred"],
1393
+ translationKey: "students.status"
1394
+ },
1395
+ teacherStatus: {
1396
+ values: ["active", "inactive", "onLeave"],
1397
+ translationKey: "teachers.status"
1398
+ },
1399
+ employmentType: {
1400
+ values: ["fullTime", "partTime", "contract", "temporary"],
1401
+ translationKey: "teachers.employmentType"
1402
+ },
1403
+ relationshipType: {
1404
+ values: ["father", "mother", "guardian", "stepparent", "grandparent", "other"],
1405
+ translationKey: "parents.relationships"
1406
+ },
1407
+ semester: {
1408
+ values: ["spring", "summer", "fall", "winter"],
1409
+ translationKey: "academic.semester"
1410
+ },
1411
+ classStatus: {
1412
+ values: ["active", "completed", "cancelled"],
1413
+ translationKey: "classes.status"
1414
+ },
1415
+ sectionStatus: {
1416
+ values: ["active", "inactive", "archived"],
1417
+ translationKey: "sections.status"
1418
+ },
1419
+ language: {
1420
+ values: ["en", "fr", "ar", "es"],
1421
+ translationKey: "common.languages"
1422
+ },
1423
+ enrollmentStatus: {
1424
+ values: ["enrolled", "completed", "dropped", "failed"],
1425
+ translationKey: "enrollments.status"
1426
+ },
1427
+ assignmentStatus: {
1428
+ values: ["active", "completed", "cancelled"],
1429
+ translationKey: "assignments.status"
1430
+ },
1431
+ calendarSystem: {
1432
+ values: ["SEMESTER", "TRIMESTER", "QUARTER"],
1433
+ translationKey: "settings.calendarSystem"
1434
+ },
1435
+ assessmentType: {
1436
+ values: ["quiz", "assignment", "project", "participation", "test", "presentation"],
1437
+ translationKey: "assessments.type"
1438
+ },
1439
+ assessmentStatus: {
1440
+ values: ["scheduled", "active", "completed", "cancelled"],
1441
+ translationKey: "assessments.status"
1442
+ },
1443
+ submissionType: {
1444
+ values: ["online", "paper", "presentation", "practical", "discussion"],
1445
+ translationKey: "assessments.submissionType"
1446
+ },
1447
+ examType: {
1448
+ values: ["midterm", "final", "standardized"],
1449
+ translationKey: "exams.type"
1450
+ },
1451
+ examSecurity: {
1452
+ values: ["low", "medium", "high"],
1453
+ translationKey: "exams.security"
1454
+ },
1455
+ examStatus: {
1456
+ values: ["scheduled", "active", "completed", "cancelled", "rescheduled"],
1457
+ translationKey: "exams.status"
1458
+ },
1459
+ gradeStatus: {
1460
+ values: ["graded", "pending", "draft", "reviewed"],
1461
+ translationKey: "grades.status"
1462
+ },
1463
+ attendanceStatus: {
1464
+ values: ["present", "absent", "late", "excused"],
1465
+ translationKey: "attendance.status"
1466
+ },
1467
+ proficiencyLevel: {
1468
+ values: ["beginner", "intermediate", "advanced", "expert"],
1469
+ translationKey: "common.proficiencyLevel"
1470
+ },
1471
+ dayOfWeek: {
1472
+ values: ["monday", "tuesday", "wednesday", "thursday", "friday", "saturday", "sunday"],
1473
+ translationKey: "common.days"
1474
+ },
1475
+ alertType: {
1476
+ values: ["academic", "attendance", "behavioral", "health", "system", "announcement", "reminder", "emergency"],
1477
+ translationKey: "alerts.type"
1478
+ },
1479
+ alertPriority: {
1480
+ values: ["low", "medium", "high", "critical"],
1481
+ translationKey: "alerts.priority"
1482
+ },
1483
+ alertStatus: {
1484
+ values: ["active", "acknowledged", "resolved", "dismissed"],
1485
+ translationKey: "alerts.status"
1486
+ },
1487
+ feeTypeStatus: {
1488
+ values: ["active", "inactive", "archived"],
1489
+ translationKey: "fees.typeStatus"
1490
+ },
1491
+ feeCategory: {
1492
+ values: ["tuition", "registration", "transport", "cafeteria", "books", "sports", "uniform", "technology", "fieldtrip", "other"],
1493
+ translationKey: "feeTypes.category"
1494
+ },
1495
+ paymentType: {
1496
+ values: ["recurring", "oneTime"],
1497
+ translationKey: "payments.type"
1498
+ },
1499
+ schedule: {
1500
+ values: ["monthly", "quarterly", "semester", "annually", "oneTime"],
1501
+ translationKey: "fees.schedule"
1502
+ },
1503
+ feeStatus: {
1504
+ values: ["pending", "partiallyPaid", "paid", "overdue"],
1505
+ translationKey: "fees.status"
1506
+ },
1507
+ feeInstallmentStatus: {
1508
+ values: ["pending", "partiallyPaid", "paid", "overdue"],
1509
+ translationKey: "fees.installmentStatus"
1510
+ },
1511
+ paymentMethod: {
1512
+ values: ["cash", "bankTransfer", "check", "creditCard", "debitCard", "online", "mobilePayment"],
1513
+ translationKey: "payments.methods"
1514
+ },
1515
+ paymentStatus: {
1516
+ values: ["completed", "pending", "failed", "refunded"],
1517
+ translationKey: "payments.status"
1518
+ },
1519
+ eventType: {
1520
+ values: ["academic", "sports", "cultural", "holiday", "exam", "meeting", "workshop", "fieldtrip", "ceremony", "conference", "other"],
1521
+ translationKey: "events.type"
1522
+ },
1523
+ eventStatus: {
1524
+ values: ["scheduled", "ongoing", "completed", "cancelled", "postponed"],
1525
+ translationKey: "events.status"
1526
+ },
1527
+ eventVisibility: {
1528
+ values: ["public", "private", "teachers", "students", "parents", "staff"],
1529
+ translationKey: "events.visibility"
1530
+ },
1531
+ participantType: {
1532
+ values: ["student", "teacher", "parent", "staff"],
1533
+ translationKey: "events.participantType"
1534
+ },
1535
+ expenseCategory: {
1536
+ values: ["salary", "utilities", "maintenance", "supplies", "equipment", "transport", "food", "security", "cleaning", "insurance", "rent", "tax", "marketing", "training", "technology", "miscellaneous"],
1537
+ translationKey: "expenses.categories"
1538
+ },
1539
+ expenseStatus: {
1540
+ values: ["pending", "approved", "paid", "rejected", "cancelled"],
1541
+ translationKey: "expenses.status"
1542
+ },
1543
+ trackerMode: {
1544
+ values: ["tracking", "gprs", "sms", "sleepTime", "sleepShock", "sleepDeep"],
1545
+ translationKey: "tracker.mode"
1546
+ },
1547
+ driverStatus: {
1548
+ values: ["active", "inactive", "onLeave", "suspended"],
1549
+ translationKey: "transport.driverStatus"
1550
+ },
1551
+ vehicleStatus: {
1552
+ values: ["active", "inactive", "maintenance", "retired"],
1553
+ translationKey: "transport.vehicleStatus"
1554
+ },
1555
+ vehicleType: {
1556
+ values: ["sedan", "minibus", "fullbus", "shuttle"],
1557
+ translationKey: "transport.vehicleType"
1558
+ },
1559
+ vehicleDocumentType: {
1560
+ values: ["insurance", "registration", "inspection", "emission", "license"],
1561
+ translationKey: "transport.documentType"
1562
+ },
1563
+ busStatus: {
1564
+ values: ["active", "inactive", "maintenance", "retired"],
1565
+ translationKey: "transport.busStatus"
1566
+ },
1567
+ refuelStatus: {
1568
+ values: ["pending", "completed", "cancelled"],
1569
+ translationKey: "transport.refuelStatus"
1570
+ },
1571
+ fuelType: {
1572
+ values: ["gasoline", "diesel", "electric", "hybrid", "lpg", "cng"],
1573
+ translationKey: "transport.fuelType"
1574
+ },
1575
+ maintenanceType: {
1576
+ values: ["scheduled", "repair", "inspection", "oilChange", "filterChange", "other"],
1577
+ translationKey: "transport.maintenanceType"
1578
+ },
1579
+ maintenanceStatus: {
1580
+ values: ["scheduled", "inProgress", "completed", "cancelled", "overdue"],
1581
+ translationKey: "transport.maintenanceStatus"
1582
+ },
1583
+ maritalStatus: {
1584
+ values: ["single", "married", "divorced", "widowed", "separated"],
1585
+ translationKey: "parents.maritalStatus"
1586
+ }
1587
+ };
1588
+ var getEnumConfig = (enumKey) => ENUMS[enumKey];
1589
+ var getEnumValues = (enumKey) => ENUMS[enumKey]?.values || [];
1590
+ // src/lib/validations.ts
1591
+ import { z as z2 } from "zod";
1592
+
1593
+ // src/lib/ZodEnum.ts
1594
+ import { z } from "zod";
1595
+ var createZodEnum = (enumKey) => {
1596
+ const values = ENUMS[enumKey]?.values;
1597
+ if (!values)
1598
+ throw new Error(`Enum ${enumKey} not found`);
1599
+ return z.enum(values);
1600
+ };
1601
+ var userTypeEnum = createZodEnum("userType");
1602
+ var userStatusEnum = createZodEnum("userStatus");
1603
+ var tokenStatusEnum = createZodEnum("tokenStatus");
1604
+ var tokenTypeEnum = createZodEnum("tokenType");
1605
+ var fileStatusEnum = createZodEnum("fileStatus");
1606
+ var genderEnum = createZodEnum("gender");
1607
+ var studentStatusEnum = createZodEnum("studentStatus");
1608
+ var teacherStatusEnum = createZodEnum("teacherStatus");
1609
+ var employmentTypeEnum = createZodEnum("employmentType");
1610
+ var relationshipTypeEnum = createZodEnum("relationshipType");
1611
+ var semesterEnum = createZodEnum("semester");
1612
+ var classStatusEnum = createZodEnum("classStatus");
1613
+ var sectionStatusEnum = createZodEnum("sectionStatus");
1614
+ var languageEnum = createZodEnum("language");
1615
+ var enrollmentStatusEnum = createZodEnum("enrollmentStatus");
1616
+ var assignmentStatusEnum = createZodEnum("assignmentStatus");
1617
+ var calendarSystemEnum = createZodEnum("calendarSystem");
1618
+ var assessmentTypeEnum = createZodEnum("assessmentType");
1619
+ var assessmentStatusEnum = createZodEnum("assessmentStatus");
1620
+ var submissionTypeEnum = createZodEnum("submissionType");
1621
+ var examTypeEnum = createZodEnum("examType");
1622
+ var examSecurityEnum = createZodEnum("examSecurity");
1623
+ var examStatusEnum = createZodEnum("examStatus");
1624
+ var gradeStatusEnum = createZodEnum("gradeStatus");
1625
+ var attendanceStatusEnum = createZodEnum("attendanceStatus");
1626
+ var proficiencyLevelEnum = createZodEnum("proficiencyLevel");
1627
+ var dayOfWeekEnum = createZodEnum("dayOfWeek");
1628
+ var alertTypeEnum = createZodEnum("alertType");
1629
+ var alertPriorityEnum = createZodEnum("alertPriority");
1630
+ var alertStatusEnum = createZodEnum("alertStatus");
1631
+ var feeTypeStatusEnum = createZodEnum("feeTypeStatus");
1632
+ var paymentTypeEnum = createZodEnum("paymentType");
1633
+ var scheduleEnum = createZodEnum("schedule");
1634
+ var feeStatusEnum = createZodEnum("feeStatus");
1635
+ var feeInstallmentStatusEnum = createZodEnum("feeInstallmentStatus");
1636
+ var paymentMethodEnum = createZodEnum("paymentMethod");
1637
+ var paymentStatusEnum = createZodEnum("paymentStatus");
1638
+ var eventTypeEnum = createZodEnum("eventType");
1639
+ var eventStatusEnum = createZodEnum("eventStatus");
1640
+ var eventVisibilityEnum = createZodEnum("eventVisibility");
1641
+ var participantTypeEnum = createZodEnum("participantType");
1642
+ var expenseCategoryEnum = createZodEnum("expenseCategory");
1643
+ var expenseStatusEnum = createZodEnum("expenseStatus");
1644
+ var trackerModeEnum = createZodEnum("trackerMode");
1645
+ var driverStatusEnum = createZodEnum("driverStatus");
1646
+ var vehicleStatusEnum = createZodEnum("vehicleStatus");
1647
+ var vehicleTypeEnum = createZodEnum("vehicleType");
1648
+ var vehicleDocumentTypeEnum = createZodEnum("vehicleDocumentType");
1649
+ var busStatusEnum = createZodEnum("busStatus");
1650
+ var refuelStatusEnum = createZodEnum("refuelStatus");
1651
+ var fuelTypeEnum = createZodEnum("fuelType");
1652
+ var maintenanceTypeEnum = createZodEnum("maintenanceType");
1653
+ var maintenanceStatusEnum = createZodEnum("maintenanceStatus");
1654
+ var maritalStatusEnum = createZodEnum("maritalStatus");
1655
+
1656
+ // src/lib/validations.ts
1657
+ var requiredId = z2.preprocess((val) => val ?? "", z2.string().min(1, "ID is required"));
1658
+ var optionalId = z2.string().min(1, "ID cannot be empty").nullish().optional();
1659
+ var emailField = z2.string().email("Invalid email format").or(z2.literal(""));
1660
+ var phoneField = z2.string().regex(/^[\+]?[1-9][\d]{0,15}$/, "Invalid phone number");
1661
+ var nameField = z2.string().min(2, "Name must be at least 2 characters").max(100, "Name too long");
1662
+ var dateField = z2.string().regex(/^(\d{4}-\d{2}-\d{2}|\d{2}\/\d{2}\/\d{4}|\d{2}-\d{2}-\d{2}|\d{2}-\d{2}-\d{4})$/, "Date must be in YYYY-MM-DD, MM/DD/YYYY, DD/MM/YYYY, DD-MM-YY, or DD-MM-YYYY format");
1663
+ var optionalDateField = z2.string().regex(/^\d{4}-\d{2}-\d{2}$/, "Date must be in YYYY-MM-DD format").nullable().optional();
1664
+ var timeField = z2.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, "Time must be in HH:MM format").optional().nullable();
1665
+ var cinField = z2.string().min(8, "CIN must be at least 8 characters").max(20, "CIN too long");
1666
+ var addressField = z2.string().max(500, "Address too long").optional();
1667
+ var academicYearField = z2.string().min(9, "Academic year is required").regex(/^\d{4}-\d{4}$/, "Academic year must be in YYYY-YYYY format");
1668
+ var num = () => {
1669
+ const createChainable = (currentSchema) => {
1670
+ const methods = {
1671
+ positive: (msg = "Must be positive") => createChainable(currentSchema.refine((val) => val > 0, { message: msg })),
1672
+ min: (value, msg) => createChainable(currentSchema.refine((val) => val >= value, { message: msg || `Must be at least ${value}` })),
1673
+ max: (value, msg) => createChainable(currentSchema.refine((val) => val <= value, { message: msg || `Cannot exceed ${value}` })),
1674
+ int: (msg = "Must be an integer") => createChainable(currentSchema.refine((val) => Number.isInteger(val), { message: msg }))
1675
+ };
1676
+ return Object.assign(currentSchema, methods);
1677
+ };
1678
+ const isValidNumber = (val) => {
1679
+ if (val === null || val === undefined || Number.isNaN(val))
1680
+ return false;
1681
+ if (typeof val === "number")
1682
+ return true;
1683
+ if (typeof val === "string") {
1684
+ const trimmed = val.trim();
1685
+ return trimmed !== "" && !isNaN(Number(trimmed));
1686
+ }
1687
+ return false;
1688
+ };
1689
+ const baseSchema = z2.any().refine(isValidNumber, { message: "Must be a valid number" }).transform((val) => typeof val === "string" ? Number(val) : val);
1690
+ return createChainable(baseSchema);
1691
+ };
1692
+ var userSchema2 = z2.object({
1693
+ id: optionalId,
1694
+ username: nameField.max(50).optional(),
1695
+ email: emailField,
1696
+ password: z2.string().min(8, "Password must be at least 8 characters"),
1697
+ roleId: optionalId,
1698
+ roleName: nameField.max(50).optional(),
1699
+ lastLogin: optionalDateField,
1700
+ image: z2.union([z2.string(), z2.instanceof(File), z2.undefined()]).optional(),
1701
+ emailVerified: z2.boolean().default(false),
1702
+ status: userStatusEnum,
1703
+ createdAt: optionalDateField
1704
+ });
1705
+ var roleSchema2 = z2.object({
1706
+ id: optionalId,
1707
+ name: nameField.max(50),
1708
+ description: z2.string().max(255, "Description too long").optional(),
1709
+ createdAt: optionalDateField
1710
+ });
1711
+ var studentSchema = z2.object({
1712
+ id: optionalId,
1713
+ classId: requiredId,
1714
+ sectionId: requiredId,
1715
+ studentCode: z2.string(),
1716
+ name: nameField,
1717
+ email: emailField,
1718
+ phone: phoneField.nullish(),
1719
+ address: addressField,
1720
+ dateOfBirth: optionalDateField,
1721
+ gender: genderEnum,
1722
+ enrollmentDate: dateField,
1723
+ medicalConditions: z2.string().max(1000, "Medical conditions description too long").nullish().optional(),
1724
+ previousSchool: z2.string().max(500, "Previous school name too long").optional().nullable(),
1725
+ image: z2.union([z2.string(), z2.instanceof(File), z2.null()]).optional(),
1726
+ status: studentStatusEnum.default("active")
1727
+ });
1728
+ var parentSchema = z2.object({
1729
+ id: optionalId,
1730
+ name: nameField,
1731
+ email: emailField.optional(),
1732
+ phone: phoneField,
1733
+ gender: genderEnum.optional(),
1734
+ address: addressField,
1735
+ dateOfBirth: optionalDateField,
1736
+ cin: cinField,
1737
+ occupation: z2.string().max(100, "Occupation too long").optional(),
1738
+ nationality: z2.string().max(100, "Nationality too long").optional(),
1739
+ maritalStatus: z2.string().max(50, "Marital status too long").optional(),
1740
+ relationshipType: relationshipTypeEnum,
1741
+ image: z2.union([z2.string(), z2.instanceof(File), z2.null()]).optional(),
1742
+ isEmergencyContact: z2.boolean().optional().default(false),
1743
+ financialResponsibility: z2.boolean().optional().default(false)
1744
+ });
1745
+ var driverSchema = z2.object({
1746
+ id: optionalId,
1747
+ name: nameField,
1748
+ email: emailField,
1749
+ cin: cinField,
1750
+ phone: phoneField,
1751
+ address: addressField,
1752
+ gender: genderEnum.optional(),
1753
+ licenseNumber: z2.string().min(5, "License number must be at least 5 characters").max(20, "License number too long"),
1754
+ licenseType: z2.string().max(10, "License type too long"),
1755
+ licenseExpiry: dateField,
1756
+ hireDate: dateField,
1757
+ salary: num().positive("Salary must be positive").optional(),
1758
+ yearsOfExperience: num().int().min(0, "Years of experience must be non-negative").optional(),
1759
+ emergencyContact: nameField.optional(),
1760
+ emergencyPhone: phoneField.optional(),
1761
+ image: z2.union([z2.string(), z2.instanceof(File), z2.null()]).optional(),
1762
+ status: driverStatusEnum.default("active"),
1763
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
1764
+ });
1765
+ var teacherPersonalSchema = z2.object({
1766
+ id: optionalId,
1767
+ name: nameField,
1768
+ cin: cinField,
1769
+ email: emailField,
1770
+ phone: phoneField,
1771
+ address: addressField,
1772
+ gender: genderEnum.optional(),
1773
+ emergencyContact: nameField.optional(),
1774
+ emergencyPhone: phoneField,
1775
+ status: teacherStatusEnum.default("active"),
1776
+ image: z2.union([z2.string(), z2.instanceof(File), z2.null()]).optional()
1777
+ });
1778
+ var teacherProfessionalSchema = z2.object({
1779
+ specialization: z2.string().max(100, "Specialization too long").optional(),
1780
+ yearsOfExperience: num().int().min(0, "Years of experience must be non-negative").optional(),
1781
+ salary: num().positive("Salary must be positive").optional(),
1782
+ hireDate: dateField,
1783
+ bankAccount: z2.coerce.string().max(100, { message: "Bank account too long" }).optional(),
1784
+ employmentType: employmentTypeEnum.optional(),
1785
+ workloadHours: num().int().min(0, "Workload hours must be non-negative").max(60, "Workload hours cannot exceed 60").optional(),
1786
+ academicDegrees: z2.string().max(500, "Academic degrees description too long").optional()
1787
+ });
1788
+ var assignmentSchema = z2.object({
1789
+ classId: z2.string().min(1, "Class is required"),
1790
+ sectionIds: z2.array(z2.string()).min(1, "At least one section is required"),
1791
+ subjectIds: z2.array(z2.string()).min(1, "At least one subject is required"),
1792
+ academicYear: z2.string().optional()
1793
+ });
1794
+ var assignmentsSchema = z2.object({
1795
+ assignments: z2.array(assignmentSchema).min(1, "At least one parent is required")
1796
+ });
1797
+ var teacherFullSchema = z2.object({
1798
+ ...teacherPersonalSchema.shape,
1799
+ ...teacherProfessionalSchema.shape,
1800
+ ...assignmentsSchema.shape
1801
+ });
1802
+ var feeTypeSchema = z2.object({
1803
+ id: optionalId,
1804
+ name: z2.string().min(2, "Fee type name must be at least 2 characters").max(100, "Fee type name too long"),
1805
+ description: z2.string().max(500, "Description too long").optional().nullable(),
1806
+ category: z2.string(),
1807
+ amount: num().positive("Amount must be greater than 0").max(1e5, "Amount too large"),
1808
+ paymentType: paymentTypeEnum.default("recurring"),
1809
+ status: feeTypeStatusEnum.default("active").optional()
1810
+ });
1811
+ var feeSchema = z2.object({
1812
+ id: optionalId,
1813
+ studentId: requiredId,
1814
+ feeTypeId: requiredId,
1815
+ academicYear: academicYearField.optional(),
1816
+ status: feeStatusEnum.optional(),
1817
+ schedule: scheduleEnum,
1818
+ baseAmount: num().positive("Base amount must be positive").optional(),
1819
+ grossAmount: num().positive("Gross amount must be positive").optional(),
1820
+ netAmount: num().optional(),
1821
+ paidAmount: num().min(0, "Paid amount cannot be negative").optional(),
1822
+ discountAmount: num().min(0, "Discount cannot be negative").optional(),
1823
+ discountReason: z2.string().max(500, "Discount reason too long").optional().nullable(),
1824
+ assignedBy: optionalId.nullable(),
1825
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
1826
+ });
1827
+ var bulkFeeItemSchema = feeSchema.omit({ studentId: true });
1828
+ var bulkFeeFormSchema = z2.object({
1829
+ studentId: requiredId,
1830
+ fees: z2.array(bulkFeeItemSchema).min(1, "At least one fee is required")
1831
+ });
1832
+ var feeInstallmentSchema = z2.object({
1833
+ feeId: optionalId,
1834
+ number: num().int().min(1, "Installment must be at least 1"),
1835
+ dueDate: dateField,
1836
+ amount: num().positive("Amount must be greater than 0").max(1e5, "Amount too large"),
1837
+ paidAmount: num().min(0, "Paid amount cannot be negative").max(1e5, "Amount too large").optional(),
1838
+ status: feeInstallmentStatusEnum.optional()
1839
+ });
1840
+ var feePaymentSchema = z2.object({
1841
+ studentId: optionalId,
1842
+ amount: num().positive("Amount must be greater than 0").max(1e5, "Amount too large").optional(),
1843
+ paymentMethod: paymentMethodEnum,
1844
+ paymentDate: dateField,
1845
+ checkNumber: z2.preprocess((val) => val === "" ? null : val, z2.string().max(50, "Check number too long").optional().nullable()),
1846
+ checkDueDate: z2.preprocess((val) => val === "" ? null : val, optionalDateField.nullable()),
1847
+ transactionRef: z2.preprocess((val) => val === "" ? null : val, z2.string().max(100, "Transaction reference too long").optional().nullable()),
1848
+ receiptNumber: z2.preprocess((val) => val === "" ? null : val, z2.string().max(50, "Receipt number too long").optional().nullable()),
1849
+ status: paymentStatusEnum.default("completed"),
1850
+ processedBy: optionalId,
1851
+ notes: z2.preprocess((val) => val === "" ? null : val, z2.string().max(1000, "Notes too long").optional().nullable()),
1852
+ allocations: z2.array(z2.object({
1853
+ feeId: optionalId,
1854
+ number: z2.number().int().positive("Installment must be a positive number"),
1855
+ amount: num().positive("Amount must be greater than 0")
1856
+ })).optional().nullable()
1857
+ });
1858
+ var paymentAllocationSchema = z2.object({
1859
+ paymentId: z2.string().min(1, "Payment ID is required"),
1860
+ feeId: z2.string().min(1, "Fee ID is required"),
1861
+ installmentId: z2.string().optional(),
1862
+ amount: z2.number().positive("Amount must be positive"),
1863
+ type: z2.enum(["fee", "installment"]).default("installment"),
1864
+ notes: z2.string().optional()
1865
+ });
1866
+ var parentsSchema = z2.object({
1867
+ parents: z2.array(parentSchema).optional().default([])
1868
+ });
1869
+ var feesSchema = z2.object({
1870
+ fees: z2.array(bulkFeeItemSchema).min(1, "At least one fee is required")
1871
+ });
1872
+ var fullStudentSchema = z2.object({
1873
+ ...studentSchema.shape,
1874
+ ...parentsSchema.shape,
1875
+ ...feesSchema.shape
1876
+ });
1877
+ var subjectSchema = z2.object({
1878
+ id: optionalId,
1879
+ code: z2.string().min(2, "Subject code must be at least 2 characters").max(10, "Subject code too long"),
1880
+ name: z2.string().min(2, "Subject name must be at least 2 characters").max(100, "Subject name too long"),
1881
+ description: z2.string().max(500, "Description too long").optional(),
1882
+ gradeLevel: num().int().min(1).max(12).optional()
1883
+ });
1884
+ var sectionSchema = z2.object({
1885
+ id: optionalId,
1886
+ classId: requiredId,
1887
+ name: z2.string().min(1, "Section name is required").max(10, "Section name too long"),
1888
+ maxStudents: num().int().min(1, "Max students must be at least 1").max(100, "Max students cannot exceed 100").default(30),
1889
+ roomNumber: num().max(1e4, "Room number too long").optional(),
1890
+ status: sectionStatusEnum.default("active")
1891
+ });
1892
+ var classSchema = z2.object({
1893
+ id: optionalId,
1894
+ name: z2.string().min(1, "Class name is required").max(50, "Class name too long"),
1895
+ description: z2.string().max(500, "Description too long").optional(),
1896
+ academicYear: academicYearField,
1897
+ level: z2.string().min(1, "Class level is required")
1898
+ });
1899
+ var attendanceSchema = z2.object({
1900
+ studentId: requiredId,
1901
+ teacherId: requiredId,
1902
+ subjectId: requiredId,
1903
+ sectionId: requiredId,
1904
+ date: dateField,
1905
+ status: attendanceStatusEnum.default("present"),
1906
+ notes: z2.string().max(500, "Notes too long").optional()
1907
+ });
1908
+ var assessmentSchema = z2.object({
1909
+ classId: requiredId,
1910
+ sectionId: requiredId,
1911
+ subjectId: requiredId,
1912
+ teacherId: requiredId,
1913
+ teacherAssignmentId: optionalId,
1914
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
1915
+ description: z2.string().max(1000, "Description too long").optional().nullable(),
1916
+ type: assessmentTypeEnum.default("quiz"),
1917
+ date: dateField,
1918
+ duration: num().int().min(1, "Duration must be at least 1 minute").max(480, "Duration cannot exceed 8 hours"),
1919
+ totalMarks: num().positive("Total marks must be greater than 0").max(1000, "Total marks cannot exceed 1000"),
1920
+ passingMarks: num().min(0, "Passing marks must be non-negative").max(1000, "Passing marks cannot exceed 1000"),
1921
+ instructions: z2.string().max(2000, "Instructions too long").optional().nullable(),
1922
+ status: assessmentStatusEnum.default("scheduled"),
1923
+ assessmentId: optionalId
1924
+ });
1925
+ var bulkAssessmentSchema = z2.object({
1926
+ assessments: z2.array(assessmentSchema).min(1, "At least one assessment is required").max(50, "Cannot create more than 50 assessments at once")
1927
+ });
1928
+ var gradeSchema = z2.object({
1929
+ assessmentId: requiredId,
1930
+ teacherId: requiredId,
1931
+ subjectId: requiredId,
1932
+ sectionId: requiredId,
1933
+ studentId: requiredId,
1934
+ gradeId: requiredId,
1935
+ assessmentTitle: z2.string().min(3, "Assessment title must be at least 3 characters").max(200, "Assessment title too long").optional(),
1936
+ marksObtained: num().min(0, "Marks obtained must be non-negative").max(1000, "Marks obtained cannot exceed 1000"),
1937
+ feedback: z2.string().max(1000, "Feedback too long").optional().nullable(),
1938
+ status: gradeStatusEnum.default("graded")
1939
+ });
1940
+ var examSchema = z2.object({
1941
+ classId: optionalId,
1942
+ sectionId: requiredId,
1943
+ subjectId: requiredId,
1944
+ teacherId: requiredId,
1945
+ examId: requiredId,
1946
+ teacherAssignmentId: requiredId,
1947
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
1948
+ description: z2.string().max(1000, "Description too long").optional().nullable(),
1949
+ type: examTypeEnum.default("midterm"),
1950
+ date: dateField,
1951
+ startTime: timeField,
1952
+ endTime: timeField,
1953
+ duration: num().int().min(30, "Exam duration must be at least 30 minutes").max(480, "Duration cannot exceed 8 hours"),
1954
+ totalMarks: num().positive("Total marks must be greater than 0").max(1000, "Total marks cannot exceed 1000"),
1955
+ passingMarks: num().min(0, "Passing marks must be non-negative").max(1000, "Passing marks cannot exceed 1000"),
1956
+ roomNumber: num().max(50, "Room number too long").optional().nullable(),
1957
+ allowedMaterials: z2.string().max(500, "Allowed materials description too long").optional().nullable(),
1958
+ instructions: z2.string().max(2000, "Instructions too long").optional().nullable(),
1959
+ status: examStatusEnum.default("scheduled")
1960
+ });
1961
+ var announcementSchema = z2.object({
1962
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
1963
+ content: z2.string().min(10, "Content must be at least 10 characters").max(5000, "Content too long"),
1964
+ authorId: optionalId,
1965
+ targetAudience: z2.enum(["all", "students", "teachers", "parents", "class"]),
1966
+ classId: optionalId,
1967
+ isPublished: z2.boolean().default(false),
1968
+ publishDate: z2.string().datetime("Invalid publish date").optional(),
1969
+ expiryDate: z2.string().datetime("Invalid expiry date").optional()
1970
+ });
1971
+ var alertSchema = z2.object({
1972
+ type: alertTypeEnum,
1973
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
1974
+ message: z2.string().min(10, "Message must be at least 10 characters").max(2000, "Message too long"),
1975
+ priority: alertPriorityEnum.default("medium"),
1976
+ status: alertStatusEnum.default("active"),
1977
+ studentId: optionalId,
1978
+ teacherId: optionalId,
1979
+ classId: optionalId,
1980
+ subjectId: optionalId,
1981
+ targetAudience: z2.enum(["all", "students", "teachers", "parents"]).optional(),
1982
+ authorId: optionalId,
1983
+ isRead: z2.boolean().default(false)
1984
+ });
1985
+ var eventSchema = z2.object({
1986
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
1987
+ description: z2.string().max(5000, "Description too long").optional().nullable(),
1988
+ type: eventTypeEnum,
1989
+ startDate: dateField,
1990
+ endDate: dateField,
1991
+ startTime: timeField,
1992
+ endTime: timeField,
1993
+ location: z2.string().max(200, "Location too long").optional().nullable(),
1994
+ venue: z2.string().max(200, "Venue too long").optional().nullable(),
1995
+ organizerId: optionalId,
1996
+ classId: optionalId.nullable(),
1997
+ sectionId: optionalId.nullable(),
1998
+ visibility: eventVisibilityEnum.default("public"),
1999
+ status: eventStatusEnum.default("scheduled"),
2000
+ capacity: num().int().positive("Capacity must be positive").optional().nullable(),
2001
+ registrationRequired: z2.boolean().default(false),
2002
+ registrationDeadline: optionalDateField.nullable(),
2003
+ attachments: z2.any().optional().nullable(),
2004
+ notes: z2.string().max(2000, "Notes too long").optional().nullable()
2005
+ });
2006
+ var eventParticipantSchema = z2.object({
2007
+ eventId: optionalId,
2008
+ participantId: optionalId,
2009
+ participantType: participantTypeEnum,
2010
+ attendanceStatus: attendanceStatusEnum.optional().nullable(),
2011
+ notes: z2.string().max(500, "Notes too long").optional().nullable()
2012
+ });
2013
+ var expenseSchema = z2.object({
2014
+ id: optionalId,
2015
+ category: expenseCategoryEnum,
2016
+ title: z2.string().min(3, "Title must be at least 3 characters").max(200, "Title too long"),
2017
+ amount: num().positive("Amount must be greater than 0").max(1e7, "Amount too large"),
2018
+ expenseDate: dateField,
2019
+ paymentMethod: paymentMethodEnum.optional().nullable(),
2020
+ paymentDate: optionalDateField.nullable(),
2021
+ vendor: z2.string().max(200, "Vendor name too long").optional().nullable(),
2022
+ invoiceNumber: z2.string().max(100, "Invoice number too long").optional().nullable(),
2023
+ receiptNumber: z2.string().max(100, "Receipt number too long").optional().nullable(),
2024
+ checkNumber: z2.string().max(50, "Check number too long").optional().nullable(),
2025
+ transactionRef: z2.string().max(100, "Transaction reference too long").optional().nullable(),
2026
+ status: expenseStatusEnum.default("pending"),
2027
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
2028
+ });
2029
+ var expenseApprovalSchema = z2.object({
2030
+ action: z2.enum(["approve", "reject"]),
2031
+ rejectionReason: z2.string().min(10, "Rejection reason must be at least 10 characters").max(1000, "Rejection reason too long").optional().nullable()
2032
+ });
2033
+ var expensePaymentSchema = z2.object({
2034
+ paymentMethod: paymentMethodEnum,
2035
+ paymentDate: dateField,
2036
+ checkNumber: z2.string().max(50, "Check number too long").optional().nullable(),
2037
+ transactionRef: z2.string().max(100, "Transaction reference too long").optional().nullable(),
2038
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
2039
+ });
2040
+ var vehicleSchema = z2.object({
2041
+ id: optionalId,
2042
+ name: z2.string().min(2, "Vehicle name must be at least 2 characters").max(100, "Vehicle name too long"),
2043
+ brand: z2.string().min(2, "Brand must be at least 2 characters").max(100, "Brand too long"),
2044
+ model: z2.string().min(2, "Model must be at least 2 characters").max(100, "Model too long"),
2045
+ year: num().int().min(1900, "Year must be after 1900").max(new Date().getFullYear() + 1, "Year cannot be in future"),
2046
+ type: vehicleTypeEnum.default("fullbus"),
2047
+ capacity: num().int().min(1, "Capacity must be at least 1").max(200, "Capacity cannot exceed 200"),
2048
+ licensePlate: z2.string().min(2, "License plate must be at least 2 characters").max(50, "License plate too long"),
2049
+ driverId: optionalId.nullable(),
2050
+ image: z2.string().max(500, "Image path too long").optional().nullable().default("novehicle.png"),
2051
+ purchaseDate: optionalDateField.nullable(),
2052
+ purchasePrice: num().min(0, "Purchase price must be non-negative").max(1e7, "Purchase price too large").optional().nullable(),
2053
+ initialMileage: num().min(0, "Initial mileage must be non-negative").max(1e7, "Initial mileage too large").optional().nullable(),
2054
+ currentMileage: num().min(0, "Current mileage must be non-negative").max(1e7, "Current mileage too large").optional().nullable(),
2055
+ status: vehicleStatusEnum.default("active"),
2056
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
2057
+ });
2058
+ var refuelSchema = z2.object({
2059
+ id: optionalId,
2060
+ busId: optionalId,
2061
+ refuelDate: dateField,
2062
+ quantity: num().positive("Quantity must be greater than 0").max(1e4, "Quantity too large"),
2063
+ unitPrice: num().positive("Unit price must be greater than 0").max(1e5, "Unit price too large"),
2064
+ totalCost: num().positive("Total cost must be greater than 0").max(1e7, "Total cost too large").optional(),
2065
+ fuelType: fuelTypeEnum.default("diesel"),
2066
+ odometer: num().min(0, "Odometer must be non-negative").max(1e7, "Odometer value too large").optional().nullable(),
2067
+ fuelStation: z2.string().max(200, "Fuel station name too long").optional().nullable(),
2068
+ invoiceNumber: z2.string().max(100, "Invoice number too long").optional().nullable(),
2069
+ paymentMethod: paymentMethodEnum.optional().nullable(),
2070
+ paidBy: optionalId.nullable(),
2071
+ status: refuelStatusEnum.default("completed"),
2072
+ notes: z2.string().max(1000, "Notes too long").optional().nullable()
2073
+ });
2074
+ var settingsSchema = z2.object({
2075
+ schoolName: z2.string().min(2, "School name must be at least 2 characters").max(200, "School name too long"),
2076
+ schoolAddress: z2.string().max(500, "School address too long").optional(),
2077
+ schoolPhone: phoneField,
2078
+ schoolEmail: emailField,
2079
+ schoolWebsite: z2.string().url("Must be a valid URL").max(255, "School website URL too long").optional(),
2080
+ schoolLogo: z2.string().url("Must be a valid image URL").max(255, "School logo URL too long").optional(),
2081
+ currentAcademicYear: academicYearField,
2082
+ gradingScale: z2.any().optional(),
2083
+ attendanceRequirement: num().min(0, "Attendance requirement must be non-negative").max(100, "Attendance requirement cannot exceed 100").default(75),
2084
+ maxClassSize: num().int("Max class size must be an integer").min(1, "Max class size must be at least 1").max(200, "Max class size cannot exceed 200").default(34),
2085
+ minimumPassingGrade: num().min(0, "Minimum passing grade must be non-negative").max(100, "Minimum passing grade cannot exceed 100").default(60),
2086
+ defaultExamDuration: num().int("Default exam duration must be in minutes").min(15, "Exam duration must be at least 15 minutes").max(480, "Exam duration cannot exceed 480 minutes").default(120),
2087
+ calendarSystem: calendarSystemEnum.default("SEMESTER"),
2088
+ startMonth: z2.string().default("september"),
2089
+ endMonth: z2.string().default("june"),
2090
+ academicAlerts: z2.boolean().default(true),
2091
+ attendanceAlerts: z2.boolean().default(true),
2092
+ eventAlerts: z2.boolean().default(true),
2093
+ homeworkAlerts: z2.boolean().default(true),
2094
+ feesReminder: z2.boolean().default(true),
2095
+ feesOverdueAlerts: z2.boolean().default(true),
2096
+ emailNotifications: z2.boolean().default(true),
2097
+ smsNotifications: z2.boolean().default(false),
2098
+ parentNotifications: z2.boolean().default(true),
2099
+ lowGradeAlerts: z2.boolean().default(true),
2100
+ allowLateSubmission: z2.boolean().default(true),
2101
+ examResultsAlerts: z2.boolean().default(true),
2102
+ disciplinaryAlerts: z2.boolean().default(true),
2103
+ achievementAlerts: z2.boolean().default(true),
2104
+ maintenanceNotifications: z2.boolean().default(true),
2105
+ twoFactorEnabled: z2.boolean().default(false),
2106
+ sessionTimeout: z2.string().regex(/^\d{1,4}$/, "Session timeout must be a number between 1-9999 minutes").default("60"),
2107
+ passwordRequireSymbols: z2.boolean().default(true),
2108
+ loginNotifications: z2.boolean().default(true),
2109
+ parentAccessEnabled: z2.boolean().default(true),
2110
+ teacherAccessEnabled: z2.boolean().default(true),
2111
+ studentAccessEnabled: z2.boolean().default(true),
2112
+ timeZone: z2.string().min(1, "Time zone is required").default("UTC"),
2113
+ language: languageEnum.default("en"),
2114
+ theme: z2.enum(["light", "dark", "system"]).default("system"),
2115
+ dateFormat: z2.enum(["YYYY-MM-DD", "MM/DD/YYYY", "DD/MM/YYYY", "DD-MM-YY", "DD-MM-YYYY"]).default("MM/DD/YYYY"),
2116
+ timeFormat: z2.enum(["12", "24"]).default("12"),
2117
+ currency: z2.string().length(3, "Currency must be a 3-letter ISO code").regex(/^[A-Z]{3}$/, "Currency must be uppercase ISO code").default("USD"),
2118
+ gradingPeriods: num().int("Grading periods must be an integer").min(1, "Grading periods must be at least 1").max(12, "Grading periods cannot exceed 12").default(4),
2119
+ schoolStartTime: z2.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, "Invalid start time format (HH:MM)").default("08:00"),
2120
+ schoolEndTime: z2.string().regex(/^([01]?[0-9]|2[0-3]):[0-5][0-9]$/, "Invalid end time format (HH:MM)").default("15:00"),
2121
+ lunchBreakDuration: num().int("Lunch break duration must be in minutes").min(15, "Lunch break must be at least 15 minutes").max(120, "Lunch break cannot exceed 120 minutes").default(30),
2122
+ maintenanceMode: z2.boolean().default(false),
2123
+ autoBackup: z2.boolean().default(true)
2124
+ });
2125
+ var idParamSchema = z2.object({
2126
+ id: optionalId
2127
+ });
2128
+ var paginationSchema = z2.object({
2129
+ page: num().int().min(1).default(1),
2130
+ limit: num().int().min(1).max(100).default(10)
2131
+ });
2132
+ var dateRangeSchema = z2.object({
2133
+ dateFrom: dateField,
2134
+ dateTo: dateField
2135
+ });
2136
+ // src/permissions/PermissionRepository.ts
2137
+ import { permissionsTable as permissionsTable3, rolePermissionsTable as rolePermissionsTable3, rolesTable as rolesTable4 } from "@/database/schema";
2138
+ import { eq as eq4, and } from "drizzle-orm";
2139
+ import { Repository as Repository4 } from "najm-api";
2140
+ class PermissionRepository {
2141
+ async getAll() {
2142
+ return await this.db.select().from(permissionsTable3);
2143
+ }
2144
+ async getById(id) {
2145
+ const [existingPermission] = await this.db.select().from(permissionsTable3).where(eq4(permissionsTable3.id, id));
2146
+ return existingPermission;
2147
+ }
2148
+ async getByName(name) {
2149
+ const [existingPermission] = await this.db.select().from(permissionsTable3).where(eq4(permissionsTable3.name, name));
2150
+ return existingPermission;
2151
+ }
2152
+ async create(data) {
2153
+ const [newPermission] = await this.db.insert(permissionsTable3).values(data).returning();
2154
+ return newPermission;
2155
+ }
2156
+ async update(id, data) {
2157
+ const [updatedPermission] = await this.db.update(permissionsTable3).set(data).where(eq4(permissionsTable3.id, id)).returning();
2158
+ return updatedPermission;
2159
+ }
2160
+ async delete(id) {
2161
+ const [deletedPermission] = await this.db.delete(permissionsTable3).where(eq4(permissionsTable3.id, id)).returning();
2162
+ return deletedPermission;
2163
+ }
2164
+ async getPermissionsByRole(roleId) {
2165
+ return await this.db.select({
2166
+ id: permissionsTable3.id,
2167
+ name: permissionsTable3.name,
2168
+ description: permissionsTable3.description,
2169
+ resource: permissionsTable3.resource,
2170
+ action: permissionsTable3.action
2171
+ }).from(rolePermissionsTable3).leftJoin(permissionsTable3, eq4(rolePermissionsTable3.permissionId, permissionsTable3.id)).where(eq4(rolePermissionsTable3.roleId, roleId));
2172
+ }
2173
+ async getRolesByPermission(permissionId) {
2174
+ return await this.db.select({
2175
+ id: rolesTable4.id,
2176
+ name: rolesTable4.name,
2177
+ description: rolesTable4.description
2178
+ }).from(rolePermissionsTable3).leftJoin(rolesTable4, eq4(rolePermissionsTable3.roleId, rolesTable4.id)).where(eq4(rolePermissionsTable3.permissionId, permissionId));
2179
+ }
2180
+ async assignPermissionToRole(roleId, permissionId) {
2181
+ const [newRolePermission] = await this.db.insert(rolePermissionsTable3).values({ roleId, permissionId }).returning();
2182
+ return newRolePermission;
2183
+ }
2184
+ async removePermissionFromRole(roleId, permissionId) {
2185
+ const [deletedRolePermission] = await this.db.delete(rolePermissionsTable3).where(and(eq4(rolePermissionsTable3.roleId, roleId), eq4(rolePermissionsTable3.permissionId, permissionId))).returning();
2186
+ return deletedRolePermission;
2187
+ }
2188
+ async checkRoleHasPermission(roleId, permissionId) {
2189
+ const [rolePermission] = await this.db.select().from(rolePermissionsTable3).where(and(eq4(rolePermissionsTable3.roleId, roleId), eq4(rolePermissionsTable3.permissionId, permissionId)));
2190
+ return !!rolePermission;
2191
+ }
2192
+ async deleteAll() {
2193
+ await this.db.delete(rolePermissionsTable3);
2194
+ const deletedPermissions = await this.db.delete(permissionsTable3).returning();
2195
+ return deletedPermissions;
2196
+ }
2197
+ }
2198
+ PermissionRepository = __legacyDecorateClassTS([
2199
+ Repository4()
2200
+ ], PermissionRepository);
2201
+ // src/permissions/PermissionGuards.ts
2202
+ import { createGuard as createGuard2, Injectable as Injectable10, GuardParams as GuardParams2, Headers as Headers3, Ctx as Ctx3 } from "najm-api";
2203
+ class PermissionGuards {
2204
+ tokenService;
2205
+ constructor(tokenService) {
2206
+ this.tokenService = tokenService;
2207
+ }
2208
+ async getUserPermissions(auth) {
2209
+ const permissions = await this.tokenService.getUserPermissions(auth);
2210
+ if (!permissions || !Array.isArray(permissions))
2211
+ return null;
2212
+ return permissions;
2213
+ }
2214
+ checkPermissionMatch(permissions, requiredPermission) {
2215
+ if (permissions.includes(requiredPermission)) {
2216
+ return true;
2217
+ }
2218
+ const [requiredAction, requiredResource] = requiredPermission.split(":");
2219
+ if (requiredAction && requiredResource) {
2220
+ if (permissions.includes(`${requiredAction}:*`)) {
2221
+ return true;
2222
+ }
2223
+ if (permissions.includes(`*:${requiredResource}`)) {
2224
+ return true;
2225
+ }
2226
+ }
2227
+ if (permissions.includes("*:*")) {
2228
+ return true;
2229
+ }
2230
+ return false;
2231
+ }
2232
+ async hasPermission(auth, ctx, requiredPermission) {
2233
+ await this.tokenService.storeUserInCache(auth, ctx);
2234
+ const permissions = await this.getUserPermissions(auth);
2235
+ if (!permissions)
2236
+ return false;
2237
+ const check = this.checkPermissionMatch(permissions, requiredPermission);
2238
+ return check;
2239
+ }
2240
+ }
2241
+ __legacyDecorateClassTS([
2242
+ __legacyDecorateParamTS(0, Headers3("authorization")),
2243
+ __legacyDecorateParamTS(1, Ctx3()),
2244
+ __legacyDecorateParamTS(2, GuardParams2()),
2245
+ __legacyMetadataTS("design:type", Function),
2246
+ __legacyMetadataTS("design:paramtypes", [
2247
+ Object,
2248
+ Object,
2249
+ Object
2250
+ ]),
2251
+ __legacyMetadataTS("design:returntype", Object)
2252
+ ], PermissionGuards.prototype, "hasPermission", null);
2253
+ PermissionGuards = __legacyDecorateClassTS([
2254
+ Injectable10(),
2255
+ __legacyMetadataTS("design:paramtypes", [
2256
+ typeof TokenService === "undefined" ? Object : TokenService
2257
+ ])
2258
+ ], PermissionGuards);
2259
+ var Permission = (...permissions) => createGuard2(PermissionGuards, "hasPermission")(...permissions);
2260
+ // src/permissions/PermissionController.ts
2261
+ import { Controller as Controller4, Get as Get4, Post as Post4, Put as Put3, Delete as Delete3, Params as Params4, Body as Body4, t as t9 } from "najm-api";
2262
+
2263
+ // src/permissions/PermissionService.ts
2264
+ import { Injectable as Injectable12 } from "najm-api";
2265
+
2266
+ // src/permissions/PermissionValidator.ts
2267
+ import { Injectable as Injectable11, t as t8 } from "najm-api";
2268
+ import { parseSchema as parseSchema3 } from "@/shared";
2269
+ import { z as z3 } from "zod";
2270
+ var permissionSchema = z3.object({
2271
+ name: z3.string().min(1, "Permission name is required"),
2272
+ description: z3.string().optional(),
2273
+ resource: z3.string().min(1, "Resource is required"),
2274
+ action: z3.string().min(1, "Action is required")
2275
+ });
2276
+
2277
+ class PermissionValidator {
2278
+ permissionRepository;
2279
+ roleValidator;
2280
+ constructor(permissionRepository, roleValidator) {
2281
+ this.permissionRepository = permissionRepository;
2282
+ this.roleValidator = roleValidator;
2283
+ }
2284
+ async validateCreatePermission(data) {
2285
+ return parseSchema3(permissionSchema, data);
2286
+ }
2287
+ async isPermissionExists(id) {
2288
+ const existingPermission = await this.permissionRepository.getById(id);
2289
+ return !!existingPermission;
2290
+ }
2291
+ async isPermissionNameExists(name) {
2292
+ const existingPermission = await this.permissionRepository.getByName(name);
2293
+ return !!existingPermission;
2294
+ }
2295
+ async checkPermissionExists(id) {
2296
+ const permissionExists = await this.isPermissionExists(id);
2297
+ if (!permissionExists) {
2298
+ throw new Error(t8("permissions.errors.notFound"));
2299
+ }
2300
+ return permissionExists;
2301
+ }
2302
+ async checkPermissionExistsByName(name) {
2303
+ const permissionExists = await this.isPermissionNameExists(name);
2304
+ if (!permissionExists) {
2305
+ throw new Error(t8("permissions.errors.notFound"));
2306
+ }
2307
+ return permissionExists;
2308
+ }
2309
+ async checkPermissionNameUnique(name, excludeId = null) {
2310
+ if (!name)
2311
+ return;
2312
+ const existingPermission = await this.permissionRepository.getByName(name);
2313
+ if (existingPermission && existingPermission.id !== excludeId) {
2314
+ throw new Error(t8("permissions.errors.nameExists"));
2315
+ }
2316
+ }
2317
+ async checkRoleExists(id) {
2318
+ return await this.roleValidator.checkRoleExists(id);
2319
+ }
2320
+ async checkRoleExistsByName(name) {
2321
+ return await this.roleValidator.checkRoleExistsByName(name);
2322
+ }
2323
+ async checkRoleHasPermission(roleId, permissionId) {
2324
+ await this.roleValidator.checkRoleExists(roleId);
2325
+ await this.checkPermissionExists(permissionId);
2326
+ const hasPermission = await this.permissionRepository.checkRoleHasPermission(roleId, permissionId);
2327
+ if (hasPermission) {
2328
+ throw new Error(t8("permissions.errors.roleAlreadyHasPermission"));
2329
+ }
2330
+ }
2331
+ }
2332
+ PermissionValidator = __legacyDecorateClassTS([
2333
+ Injectable11(),
2334
+ __legacyMetadataTS("design:paramtypes", [
2335
+ typeof PermissionRepository === "undefined" ? Object : PermissionRepository,
2336
+ typeof RoleValidator === "undefined" ? Object : RoleValidator
2337
+ ])
2338
+ ], PermissionValidator);
2339
+
2340
+ // src/permissions/PermissionService.ts
2341
+ class PermissionService {
2342
+ permissionRepository;
2343
+ permissionValidator;
2344
+ roleService;
2345
+ constructor(permissionRepository, permissionValidator, roleService) {
2346
+ this.permissionRepository = permissionRepository;
2347
+ this.permissionValidator = permissionValidator;
2348
+ this.roleService = roleService;
2349
+ }
2350
+ async getAll() {
2351
+ return await this.permissionRepository.getAll();
2352
+ }
2353
+ async getById(id) {
2354
+ await this.permissionValidator.checkPermissionExists(id);
2355
+ return await this.permissionRepository.getById(id);
2356
+ }
2357
+ async getByName(name) {
2358
+ return await this.permissionRepository.getByName(name);
2359
+ }
2360
+ async getByResource(resource) {
2361
+ return await this.permissionRepository.getAll().then((permissions) => permissions.filter((p) => p.resource === resource));
2362
+ }
2363
+ async create(data) {
2364
+ await this.permissionValidator.validateCreatePermission(data);
2365
+ await this.permissionValidator.checkPermissionNameUnique(data.name);
2366
+ return await this.permissionRepository.create(data);
2367
+ }
2368
+ async update(id, data) {
2369
+ await this.permissionValidator.checkPermissionExists(id);
2370
+ await this.permissionValidator.checkPermissionNameUnique(data.name, id);
2371
+ return await this.permissionRepository.update(id, data);
2372
+ }
2373
+ async delete(id) {
2374
+ await this.permissionValidator.checkPermissionExists(id);
2375
+ return await this.permissionRepository.delete(id);
2376
+ }
2377
+ async getPermissionsByRole(roleId) {
2378
+ return await this.permissionRepository.getPermissionsByRole(roleId);
2379
+ }
2380
+ async getRolesByPermission(permissionId) {
2381
+ await this.permissionValidator.checkPermissionExists(permissionId);
2382
+ return await this.permissionRepository.getRolesByPermission(permissionId);
2383
+ }
2384
+ async assignPermissionToRole(roleId, permissionId) {
2385
+ await this.permissionValidator.checkRoleHasPermission(roleId, permissionId);
2386
+ return await this.permissionRepository.assignPermissionToRole(roleId, permissionId);
2387
+ }
2388
+ async removePermissionFromRole(roleId, permissionId) {
2389
+ return await this.permissionRepository.removePermissionFromRole(roleId, permissionId);
2390
+ }
2391
+ async seedDefaultPermissions(defaultPermissions) {
2392
+ const createdPermissions = [];
2393
+ for (const permission of defaultPermissions) {
2394
+ try {
2395
+ const permissionEntity = await this.create(permission);
2396
+ createdPermissions.push(permissionEntity);
2397
+ } catch (error) {
2398
+ continue;
2399
+ }
2400
+ }
2401
+ return createdPermissions;
2402
+ }
2403
+ async seedDefaultRolePermissions(defaultRolePermissions) {
2404
+ const results = [];
2405
+ for (const { roleName, permissions } of defaultRolePermissions) {
2406
+ try {
2407
+ await this.permissionValidator.checkRoleExistsByName(roleName);
2408
+ const role = await this.roleService.getByName(roleName);
2409
+ for (const permissionName of permissions) {
2410
+ try {
2411
+ await this.permissionValidator.checkPermissionExistsByName(permissionName);
2412
+ const permission = await this.getByName(permissionName);
2413
+ await this.permissionValidator.checkRoleHasPermission(role.id, permission.id);
2414
+ await this.assignPermissionToRole(role.id, permission.id);
2415
+ results.push({ role: roleName, permission: permissionName });
2416
+ } catch (error) {
2417
+ continue;
2418
+ }
2419
+ }
2420
+ } catch (error) {
2421
+ continue;
2422
+ }
2423
+ }
2424
+ return results;
2425
+ }
2426
+ async deleteAll() {
2427
+ return await this.permissionRepository.deleteAll();
2428
+ }
2429
+ }
2430
+ PermissionService = __legacyDecorateClassTS([
2431
+ Injectable12(),
2432
+ __legacyMetadataTS("design:paramtypes", [
2433
+ typeof PermissionRepository === "undefined" ? Object : PermissionRepository,
2434
+ typeof PermissionValidator === "undefined" ? Object : PermissionValidator,
2435
+ typeof RoleService === "undefined" ? Object : RoleService
2436
+ ])
2437
+ ], PermissionService);
2438
+
2439
+ // src/permissions/PermissionController.ts
2440
+ import { isAdmin as isAdmin3 } from "@/roles/RoleGuards";
2441
+ class PermissionController {
2442
+ permissionService;
2443
+ constructor(permissionService) {
2444
+ this.permissionService = permissionService;
2445
+ }
2446
+ async getPermissions() {
2447
+ const permissions = await this.permissionService.getAll();
2448
+ return {
2449
+ data: permissions,
2450
+ message: t9("permissions.success.retrieved"),
2451
+ status: "success"
2452
+ };
2453
+ }
2454
+ async getPermission(id) {
2455
+ const permission = await this.permissionService.getById(id);
2456
+ return {
2457
+ data: permission,
2458
+ message: t9("permissions.success.retrieved"),
2459
+ status: "success"
2460
+ };
2461
+ }
2462
+ async create(body) {
2463
+ const newPermission = await this.permissionService.create(body);
2464
+ return {
2465
+ data: newPermission,
2466
+ message: t9("permissions.success.created"),
2467
+ status: "success"
2468
+ };
2469
+ }
2470
+ async update(id, body) {
2471
+ const updatedPermission = await this.permissionService.update(id, body);
2472
+ return {
2473
+ data: updatedPermission,
2474
+ message: t9("permissions.success.updated"),
2475
+ status: "success"
2476
+ };
2477
+ }
2478
+ async delete(id) {
2479
+ const result = await this.permissionService.delete(id);
2480
+ return {
2481
+ data: result,
2482
+ message: t9("permissions.success.deleted"),
2483
+ status: "success"
2484
+ };
2485
+ }
2486
+ async getByRole(roleId) {
2487
+ const permissions = await this.permissionService.getPermissionsByRole(roleId);
2488
+ return {
2489
+ data: permissions,
2490
+ message: t9("permissions.success.retrieved"),
2491
+ status: "success"
2492
+ };
2493
+ }
2494
+ async getRolesByPermission(permissionId) {
2495
+ const roles = await this.permissionService.getRolesByPermission(permissionId);
2496
+ return {
2497
+ data: roles,
2498
+ message: t9("permissions.success.retrieved"),
2499
+ status: "success"
2500
+ };
2501
+ }
2502
+ async assignToRole(roleId, permissionId) {
2503
+ const result = await this.permissionService.assignPermissionToRole(roleId, permissionId);
2504
+ return {
2505
+ data: result,
2506
+ message: t9("permissions.success.assigned"),
2507
+ status: "success"
2508
+ };
2509
+ }
2510
+ async removeFromRole(roleId, permissionId) {
2511
+ const result = await this.permissionService.removePermissionFromRole(roleId, permissionId);
2512
+ return {
2513
+ data: result,
2514
+ message: t9("permissions.success.removed"),
2515
+ status: "success"
2516
+ };
2517
+ }
2518
+ async deleteAll() {
2519
+ const result = await this.permissionService.deleteAll();
2520
+ return {
2521
+ data: result,
2522
+ message: t9("permissions.success.allDeleted"),
2523
+ status: "success"
2524
+ };
2525
+ }
2526
+ }
2527
+ __legacyDecorateClassTS([
2528
+ Get4(),
2529
+ __legacyMetadataTS("design:type", Function),
2530
+ __legacyMetadataTS("design:paramtypes", []),
2531
+ __legacyMetadataTS("design:returntype", Promise)
2532
+ ], PermissionController.prototype, "getPermissions", null);
2533
+ __legacyDecorateClassTS([
2534
+ Get4("/:id"),
2535
+ __legacyDecorateParamTS(0, Params4("id")),
2536
+ __legacyMetadataTS("design:type", Function),
2537
+ __legacyMetadataTS("design:paramtypes", [
2538
+ String
2539
+ ]),
2540
+ __legacyMetadataTS("design:returntype", Promise)
2541
+ ], PermissionController.prototype, "getPermission", null);
2542
+ __legacyDecorateClassTS([
2543
+ Post4(),
2544
+ __legacyDecorateParamTS(0, Body4()),
2545
+ __legacyMetadataTS("design:type", Function),
2546
+ __legacyMetadataTS("design:paramtypes", [
2547
+ Object
2548
+ ]),
2549
+ __legacyMetadataTS("design:returntype", Promise)
2550
+ ], PermissionController.prototype, "create", null);
2551
+ __legacyDecorateClassTS([
2552
+ Put3("/:id"),
2553
+ __legacyDecorateParamTS(0, Params4("id")),
2554
+ __legacyDecorateParamTS(1, Body4()),
2555
+ __legacyMetadataTS("design:type", Function),
2556
+ __legacyMetadataTS("design:paramtypes", [
2557
+ String,
2558
+ Object
2559
+ ]),
2560
+ __legacyMetadataTS("design:returntype", Promise)
2561
+ ], PermissionController.prototype, "update", null);
2562
+ __legacyDecorateClassTS([
2563
+ Delete3("/:id"),
2564
+ __legacyDecorateParamTS(0, Params4("id")),
2565
+ __legacyMetadataTS("design:type", Function),
2566
+ __legacyMetadataTS("design:paramtypes", [
2567
+ String
2568
+ ]),
2569
+ __legacyMetadataTS("design:returntype", Promise)
2570
+ ], PermissionController.prototype, "delete", null);
2571
+ __legacyDecorateClassTS([
2572
+ Get4("/role/:roleId"),
2573
+ __legacyDecorateParamTS(0, Params4("roleId")),
2574
+ __legacyMetadataTS("design:type", Function),
2575
+ __legacyMetadataTS("design:paramtypes", [
2576
+ String
2577
+ ]),
2578
+ __legacyMetadataTS("design:returntype", Promise)
2579
+ ], PermissionController.prototype, "getByRole", null);
2580
+ __legacyDecorateClassTS([
2581
+ Get4("/roles/:permissionId"),
2582
+ __legacyDecorateParamTS(0, Params4("permissionId")),
2583
+ __legacyMetadataTS("design:type", Function),
2584
+ __legacyMetadataTS("design:paramtypes", [
2585
+ String
2586
+ ]),
2587
+ __legacyMetadataTS("design:returntype", Promise)
2588
+ ], PermissionController.prototype, "getRolesByPermission", null);
2589
+ __legacyDecorateClassTS([
2590
+ Post4("/assign/:roleId/:permissionId"),
2591
+ __legacyDecorateParamTS(0, Params4("roleId")),
2592
+ __legacyDecorateParamTS(1, Params4("permissionId")),
2593
+ __legacyMetadataTS("design:type", Function),
2594
+ __legacyMetadataTS("design:paramtypes", [
2595
+ String,
2596
+ String
2597
+ ]),
2598
+ __legacyMetadataTS("design:returntype", Promise)
2599
+ ], PermissionController.prototype, "assignToRole", null);
2600
+ __legacyDecorateClassTS([
2601
+ Delete3("/remove/:roleId/:permissionId"),
2602
+ __legacyDecorateParamTS(0, Params4("roleId")),
2603
+ __legacyDecorateParamTS(1, Params4("permissionId")),
2604
+ __legacyMetadataTS("design:type", Function),
2605
+ __legacyMetadataTS("design:paramtypes", [
2606
+ String,
2607
+ String
2608
+ ]),
2609
+ __legacyMetadataTS("design:returntype", Promise)
2610
+ ], PermissionController.prototype, "removeFromRole", null);
2611
+ __legacyDecorateClassTS([
2612
+ Delete3(),
2613
+ isAdmin3(),
2614
+ __legacyMetadataTS("design:type", Function),
2615
+ __legacyMetadataTS("design:paramtypes", []),
2616
+ __legacyMetadataTS("design:returntype", Promise)
2617
+ ], PermissionController.prototype, "deleteAll", null);
2618
+ PermissionController = __legacyDecorateClassTS([
2619
+ Controller4("/permissions"),
2620
+ isAdmin3(),
2621
+ __legacyMetadataTS("design:paramtypes", [
2622
+ typeof PermissionService === "undefined" ? Object : PermissionService
2623
+ ])
2624
+ ], PermissionController);
2625
+ // src/shared/index.ts
2626
+ import * as fs from "fs/promises";
2627
+ import * as path from "path";
2628
+ import _isEmpty from "lodash.isempty";
2629
+ var avatarsPath = path.join(process.cwd(), "avatars");
2630
+ var parseSchema4 = async (schema, data) => {
2631
+ try {
2632
+ return await schema.parseAsync(data);
2633
+ } catch (error) {
2634
+ const errors = error.issues || error.errors || [];
2635
+ const errorMessage = errors.map((err) => `${err.path.join(".")}: ${err.message}`).join("; ");
2636
+ throw new Error(errorMessage);
2637
+ }
2638
+ };
2639
+ var clean2 = (obj) => {
2640
+ const cleaned = {};
2641
+ for (const [key, value] of Object.entries(obj)) {
2642
+ if (value !== null && value !== undefined && value !== "") {
2643
+ cleaned[key] = value;
2644
+ }
2645
+ }
2646
+ return cleaned;
2647
+ };
2648
+ var getAvatarFile = async (fileName) => {
2649
+ try {
2650
+ const filePath = path.join(avatarsPath, fileName);
2651
+ const buffer = await fs.readFile(filePath);
2652
+ const file = new File([buffer], fileName, {
2653
+ type: "image/png"
2654
+ });
2655
+ return file;
2656
+ } catch (error) {
2657
+ return null;
2658
+ }
2659
+ };
2660
+ var formatDate = (dateValue) => {
2661
+ if (!dateValue)
2662
+ return null;
2663
+ let date;
2664
+ if (dateValue instanceof Date) {
2665
+ date = dateValue;
2666
+ } else if (typeof dateValue === "string") {
2667
+ date = new Date(dateValue);
2668
+ } else {
2669
+ return null;
2670
+ }
2671
+ if (isNaN(date.getTime()))
2672
+ return null;
2673
+ return date.toISOString().split("T")[0];
2674
+ };
2675
+ function calculateAge(dateOfBirth) {
2676
+ if (!dateOfBirth)
2677
+ return null;
2678
+ const formattedDate = formatDate(dateOfBirth);
2679
+ if (!formattedDate)
2680
+ return null;
2681
+ const birth = new Date(formattedDate);
2682
+ const today = new Date;
2683
+ let age = today.getFullYear() - birth.getFullYear();
2684
+ const monthDiff = today.getMonth() - birth.getMonth();
2685
+ if (monthDiff < 0 || monthDiff === 0 && today.getDate() < birth.getDate()) {
2686
+ age--;
2687
+ }
2688
+ return age;
2689
+ }
2690
+ function calculateYearsOfExperience(hireDate) {
2691
+ if (!hireDate)
2692
+ return null;
2693
+ const formattedDate = formatDate(hireDate);
2694
+ if (!formattedDate)
2695
+ return null;
2696
+ const hire = new Date(formattedDate);
2697
+ const today = new Date;
2698
+ let years = today.getFullYear() - hire.getFullYear();
2699
+ const monthDiff = today.getMonth() - hire.getMonth();
2700
+ if (monthDiff < 0 || monthDiff === 0 && today.getDate() < hire.getDate()) {
2701
+ years--;
2702
+ }
2703
+ return years;
2704
+ }
2705
+ function pickProps(source, keys) {
2706
+ const result = {};
2707
+ for (const key of keys) {
2708
+ if (source[key] !== undefined) {
2709
+ result[key] = source[key];
2710
+ }
2711
+ }
2712
+ return result;
2713
+ }
2714
+ var isEmpty = _isEmpty;
2715
+ var isPath = (img) => typeof img === "string" && img.trim().length > 0 && (img.startsWith("/") || img.startsWith("http") || img.startsWith("storage/"));
2716
+ var isFile = (img) => !!img && typeof img !== "string" && img instanceof File;
2717
+ // src/database/db.ts
2718
+ import { drizzle } from "drizzle-orm/postgres-js";
2719
+
2720
+ // src/database/schema/index.ts
2721
+ var exports_schema = {};
2722
+ __export(exports_schema, {
2723
+ usersTable: () => usersTable3,
2724
+ tokensTable: () => tokensTable2,
2725
+ timestamps: () => timestamps,
2726
+ rolesTable: () => rolesTable5,
2727
+ rolePermissionsTable: () => rolePermissionsTable4,
2728
+ permissionsTable: () => permissionsTable4,
2729
+ idField: () => idField
2730
+ });
2731
+ import { pgTable, text, boolean, timestamp } from "drizzle-orm/pg-core";
2732
+ import { nanoid as nanoid2 } from "nanoid";
2733
+ import { sql } from "drizzle-orm";
2734
+
2735
+ // src/database/schema/PgEnum.ts
2736
+ import { getEnumConfig as getEnumConfig2 } from "@/lib/ENUMS";
2737
+ import { pgEnum } from "drizzle-orm/pg-core";
2738
+ var createPgEnum = (enumKey) => {
2739
+ const config = getEnumConfig2(enumKey);
2740
+ if (!config)
2741
+ throw new Error(`Enum ${enumKey} not found`);
2742
+ const enumName = config.name || enumKey;
2743
+ return pgEnum(enumName, config.values);
2744
+ };
2745
+ var userStatusEnum2 = createPgEnum("userStatus");
2746
+ var tokenStatusEnum2 = createPgEnum("tokenStatus");
2747
+ var tokenTypeEnum2 = createPgEnum("tokenType");
2748
+ var studentStatusEnum2 = createPgEnum("studentStatus");
2749
+
2750
+ // src/database/schema/index.ts
2751
+ var timestamps = {
2752
+ createdAt: timestamp("created_at", { mode: "string" }).defaultNow(),
2753
+ updatedAt: timestamp("updated_at", { mode: "string" }).defaultNow().$onUpdate(() => sql`CURRENT_TIMESTAMP`)
2754
+ };
2755
+ var idField = (length = 5) => text("id").primaryKey().notNull().$defaultFn(() => nanoid2(length));
2756
+ var rolesTable5 = pgTable("roles", {
2757
+ id: idField(),
2758
+ name: text("name").notNull(),
2759
+ description: text("description")
2760
+ });
2761
+ var usersTable3 = pgTable("users", {
2762
+ id: idField(8),
2763
+ email: text("email").notNull().unique(),
2764
+ emailVerified: boolean("email_verified").default(false),
2765
+ password: text("password").notNull(),
2766
+ image: text("image").default("noavatar.png"),
2767
+ status: userStatusEnum2("status").default("pending"),
2768
+ roleId: text("role_id").references(() => rolesTable5.id),
2769
+ lastLogin: timestamp("last_login", { mode: "string" }),
2770
+ ...timestamps
2771
+ });
2772
+ var tokensTable2 = pgTable("tokens", {
2773
+ id: idField(10),
2774
+ userId: text("user_id").references(() => usersTable3.id, { onDelete: "cascade" }).unique().notNull(),
2775
+ token: text("token").notNull(),
2776
+ type: tokenTypeEnum2("type").default("refresh"),
2777
+ status: tokenStatusEnum2("status").default("active"),
2778
+ expiresAt: timestamp("expires_at", { mode: "string" }).notNull(),
2779
+ ...timestamps
2780
+ });
2781
+ var permissionsTable4 = pgTable("permissions", {
2782
+ id: idField(),
2783
+ name: text("name").notNull().unique(),
2784
+ description: text("description"),
2785
+ resource: text("resource").notNull(),
2786
+ action: text("action").notNull(),
2787
+ ...timestamps
2788
+ });
2789
+ var rolePermissionsTable4 = pgTable("role_permissions", {
2790
+ id: idField(),
2791
+ roleId: text("role_id").references(() => rolesTable5.id).notNull(),
2792
+ permissionId: text("permission_id").references(() => permissionsTable4.id).notNull(),
2793
+ ...timestamps
2794
+ });
2795
+
2796
+ // src/database/db.ts
2797
+ var db = process.env.DB_URL ? drizzle(process.env.DB_URL, { schema: exports_schema }) : null;
2798
+ if (db) {
2799
+ console.warn("⚠️ najm-auth: Using standalone DB instance. This should only be used for development.");
2800
+ }
2801
+ // src/plugin.ts
2802
+ var AuthPlugin = {
2803
+ name: "najm-auth",
2804
+ version: "0.1.3",
2805
+ database: "auth",
2806
+ controllers: [
2807
+ UserController,
2808
+ AuthController,
2809
+ RoleController,
2810
+ PermissionController
2811
+ ],
2812
+ services: [
2813
+ UserService,
2814
+ AuthService,
2815
+ CookieService,
2816
+ EncryptionService,
2817
+ RoleService,
2818
+ PermissionService,
2819
+ TokenService
2820
+ ],
2821
+ repositories: [
2822
+ UserRepository,
2823
+ RoleRepository,
2824
+ PermissionRepository,
2825
+ TokenRepository
2826
+ ],
2827
+ providers: [
2828
+ UserValidator,
2829
+ RoleValidator,
2830
+ PermissionValidator,
2831
+ RoleGuards,
2832
+ PermissionGuards
2833
+ ],
2834
+ onSetup: async () => {
2835
+ console.log(" ✓ najm-auth plugin initialized");
2836
+ console.log(" - Authentication endpoints ready");
2837
+ console.log(" - Authorization system active");
2838
+ console.log(" - Database connection established");
2839
+ },
2840
+ onTeardown: async () => {
2841
+ console.log(" ✓ najm-auth plugin cleanup complete");
2842
+ }
2843
+ };
2844
+ export {
2845
+ vehicleTypeEnum,
2846
+ vehicleStatusEnum,
2847
+ vehicleSchema,
2848
+ vehicleDocumentTypeEnum,
2849
+ usersTable3 as usersTable,
2850
+ userTypeEnum,
2851
+ userStatusEnum,
2852
+ userSchema2 as userSchema,
2853
+ trackerModeEnum,
2854
+ tokensTable2 as tokensTable,
2855
+ tokenTypeEnum,
2856
+ tokenStatusEnum,
2857
+ teacherStatusEnum,
2858
+ teacherProfessionalSchema,
2859
+ teacherPersonalSchema,
2860
+ teacherFullSchema,
2861
+ submissionTypeEnum,
2862
+ subjectSchema,
2863
+ studentStatusEnum,
2864
+ studentSchema,
2865
+ settingsSchema,
2866
+ semesterEnum,
2867
+ sectionStatusEnum,
2868
+ sectionSchema,
2869
+ scheduleEnum,
2870
+ rolesTable5 as rolesTable,
2871
+ roleSchema2 as roleSchema,
2872
+ rolePermissionsTable4 as rolePermissionsTable,
2873
+ relationshipTypeEnum,
2874
+ refuelStatusEnum,
2875
+ refuelSchema,
2876
+ proficiencyLevelEnum,
2877
+ pickProps,
2878
+ permissionsTable4 as permissionsTable,
2879
+ paymentTypeEnum,
2880
+ paymentStatusEnum,
2881
+ paymentMethodEnum,
2882
+ paymentAllocationSchema,
2883
+ participantTypeEnum,
2884
+ parseSchema4 as parseSchema,
2885
+ parentsSchema,
2886
+ parentSchema,
2887
+ paginationSchema,
2888
+ maritalStatusEnum,
2889
+ maintenanceTypeEnum,
2890
+ maintenanceStatusEnum,
2891
+ languageEnum,
2892
+ isTeacher,
2893
+ isStudent,
2894
+ isStaff,
2895
+ isSecretary,
2896
+ isPrincipal,
2897
+ isPath,
2898
+ isParent,
2899
+ isFinancial,
2900
+ isFile,
2901
+ isEmpty,
2902
+ isAuth,
2903
+ isAdministrator,
2904
+ isAdmin,
2905
+ isAccounting,
2906
+ idParamSchema,
2907
+ idField,
2908
+ gradeStatusEnum,
2909
+ gradeSchema,
2910
+ getEnumValues,
2911
+ getEnumConfig,
2912
+ getAvatarFile,
2913
+ genderEnum,
2914
+ fullStudentSchema,
2915
+ fuelTypeEnum,
2916
+ formatDate,
2917
+ fileStatusEnum,
2918
+ feesSchema,
2919
+ feeTypeStatusEnum,
2920
+ feeTypeSchema,
2921
+ feeStatusEnum,
2922
+ feeSchema,
2923
+ feePaymentSchema,
2924
+ feeInstallmentStatusEnum,
2925
+ feeInstallmentSchema,
2926
+ expenseStatusEnum,
2927
+ expenseSchema,
2928
+ expensePaymentSchema,
2929
+ expenseCategoryEnum,
2930
+ expenseApprovalSchema,
2931
+ examTypeEnum,
2932
+ examStatusEnum,
2933
+ examSecurityEnum,
2934
+ examSchema,
2935
+ eventVisibilityEnum,
2936
+ eventTypeEnum,
2937
+ eventStatusEnum,
2938
+ eventSchema,
2939
+ eventParticipantSchema,
2940
+ enrollmentStatusEnum,
2941
+ employmentTypeEnum,
2942
+ driverStatusEnum,
2943
+ driverSchema,
2944
+ db,
2945
+ dayOfWeekEnum,
2946
+ dateRangeSchema,
2947
+ clean2 as clean,
2948
+ classStatusEnum,
2949
+ classSchema,
2950
+ calendarSystemEnum,
2951
+ calculateYearsOfExperience,
2952
+ calculateAge,
2953
+ busStatusEnum,
2954
+ bulkFeeItemSchema,
2955
+ bulkFeeFormSchema,
2956
+ bulkAssessmentSchema,
2957
+ avatarsPath,
2958
+ attendanceStatusEnum,
2959
+ attendanceSchema,
2960
+ assignmentsSchema,
2961
+ assignmentStatusEnum,
2962
+ assignmentSchema,
2963
+ assessmentTypeEnum,
2964
+ assessmentStatusEnum,
2965
+ assessmentSchema,
2966
+ announcementSchema,
2967
+ alertTypeEnum,
2968
+ alertStatusEnum,
2969
+ alertSchema,
2970
+ alertPriorityEnum,
2971
+ UserValidator,
2972
+ UserService,
2973
+ UserRepository,
2974
+ UserController,
2975
+ TokenService,
2976
+ TokenRepository,
2977
+ RoleValidator,
2978
+ RoleService,
2979
+ RoleRepository,
2980
+ RoleGuards,
2981
+ RoleController,
2982
+ RoleChecker,
2983
+ Role,
2984
+ ROLE_GROUPS,
2985
+ ROLES,
2986
+ PermissionValidator,
2987
+ PermissionService,
2988
+ PermissionRepository,
2989
+ PermissionGuards,
2990
+ PermissionController,
2991
+ Permission,
2992
+ EncryptionService,
2993
+ ENUMS,
2994
+ CookieService,
2995
+ AuthService,
2996
+ AuthPlugin,
2997
+ AuthController
2998
+ };