@payez/next-mvp 3.0.0
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.
- package/README.md +782 -0
- package/dist/api/auth-handler.d.ts +67 -0
- package/dist/api/auth-handler.js +397 -0
- package/dist/api/index.d.ts +10 -0
- package/dist/api/index.js +19 -0
- package/dist/api-handlers/account/change-password.d.ts +9 -0
- package/dist/api-handlers/account/change-password.js +112 -0
- package/dist/api-handlers/account/masked-info.d.ts +2 -0
- package/dist/api-handlers/account/masked-info.js +41 -0
- package/dist/api-handlers/account/profile.d.ts +3 -0
- package/dist/api-handlers/account/profile.js +63 -0
- package/dist/api-handlers/account/recovery/initiate.d.ts +2 -0
- package/dist/api-handlers/account/recovery/initiate.js +26 -0
- package/dist/api-handlers/account/recovery/send-code.d.ts +2 -0
- package/dist/api-handlers/account/recovery/send-code.js +28 -0
- package/dist/api-handlers/account/recovery/verify-code.d.ts +2 -0
- package/dist/api-handlers/account/recovery/verify-code.js +28 -0
- package/dist/api-handlers/account/reset-password.d.ts +2 -0
- package/dist/api-handlers/account/reset-password.js +26 -0
- package/dist/api-handlers/account/send-code.d.ts +24 -0
- package/dist/api-handlers/account/send-code.js +60 -0
- package/dist/api-handlers/account/update-phone.d.ts +27 -0
- package/dist/api-handlers/account/update-phone.js +64 -0
- package/dist/api-handlers/account/validate-password.d.ts +17 -0
- package/dist/api-handlers/account/validate-password.js +81 -0
- package/dist/api-handlers/account/verify-email.d.ts +26 -0
- package/dist/api-handlers/account/verify-email.js +106 -0
- package/dist/api-handlers/account/verify-sms.d.ts +26 -0
- package/dist/api-handlers/account/verify-sms.js +106 -0
- package/dist/api-handlers/admin/analytics.d.ts +20 -0
- package/dist/api-handlers/admin/analytics.js +379 -0
- package/dist/api-handlers/admin/audit.d.ts +20 -0
- package/dist/api-handlers/admin/audit.js +214 -0
- package/dist/api-handlers/admin/index.d.ts +21 -0
- package/dist/api-handlers/admin/index.js +41 -0
- package/dist/api-handlers/admin/redis-sessions.d.ts +36 -0
- package/dist/api-handlers/admin/redis-sessions.js +204 -0
- package/dist/api-handlers/admin/sessions.d.ts +21 -0
- package/dist/api-handlers/admin/sessions.js +284 -0
- package/dist/api-handlers/admin/site-logs.d.ts +46 -0
- package/dist/api-handlers/admin/site-logs.js +318 -0
- package/dist/api-handlers/admin/users.d.ts +20 -0
- package/dist/api-handlers/admin/users.js +222 -0
- package/dist/api-handlers/admin/vibe-data.d.ts +80 -0
- package/dist/api-handlers/admin/vibe-data.js +268 -0
- package/dist/api-handlers/anon/preferences.d.ts +37 -0
- package/dist/api-handlers/anon/preferences.js +96 -0
- package/dist/api-handlers/auth/jwks.d.ts +2 -0
- package/dist/api-handlers/auth/jwks.js +24 -0
- package/dist/api-handlers/auth/login.d.ts +42 -0
- package/dist/api-handlers/auth/login.js +178 -0
- package/dist/api-handlers/auth/refresh.d.ts +74 -0
- package/dist/api-handlers/auth/refresh.js +635 -0
- package/dist/api-handlers/auth/signout.d.ts +37 -0
- package/dist/api-handlers/auth/signout.js +187 -0
- package/dist/api-handlers/auth/status.d.ts +8 -0
- package/dist/api-handlers/auth/status.js +26 -0
- package/dist/api-handlers/auth/update-session.d.ts +37 -0
- package/dist/api-handlers/auth/update-session.js +95 -0
- package/dist/api-handlers/auth/validate.d.ts +6 -0
- package/dist/api-handlers/auth/validate.js +43 -0
- package/dist/api-handlers/auth/verify-code.d.ts +43 -0
- package/dist/api-handlers/auth/verify-code.js +94 -0
- package/dist/api-handlers/session/refresh-viability.d.ts +14 -0
- package/dist/api-handlers/session/refresh-viability.js +39 -0
- package/dist/api-handlers/session/viability.d.ts +13 -0
- package/dist/api-handlers/session/viability.js +146 -0
- package/dist/api-handlers/test/force-expire.d.ts +23 -0
- package/dist/api-handlers/test/force-expire.js +65 -0
- package/dist/auth/auth-decision.d.ts +39 -0
- package/dist/auth/auth-decision.js +182 -0
- package/dist/auth/auth-options.d.ts +57 -0
- package/dist/auth/auth-options.js +213 -0
- package/dist/auth/callbacks/index.d.ts +6 -0
- package/dist/auth/callbacks/index.js +12 -0
- package/dist/auth/callbacks/jwt.d.ts +45 -0
- package/dist/auth/callbacks/jwt.js +305 -0
- package/dist/auth/callbacks/session.d.ts +60 -0
- package/dist/auth/callbacks/session.js +170 -0
- package/dist/auth/callbacks/signin.d.ts +23 -0
- package/dist/auth/callbacks/signin.js +44 -0
- package/dist/auth/events/index.d.ts +4 -0
- package/dist/auth/events/index.js +8 -0
- package/dist/auth/events/signout.d.ts +17 -0
- package/dist/auth/events/signout.js +32 -0
- package/dist/auth/providers/credentials.d.ts +32 -0
- package/dist/auth/providers/credentials.js +223 -0
- package/dist/auth/providers/index.d.ts +5 -0
- package/dist/auth/providers/index.js +21 -0
- package/dist/auth/providers/oauth.d.ts +26 -0
- package/dist/auth/providers/oauth.js +105 -0
- package/dist/auth/route-config.d.ts +66 -0
- package/dist/auth/route-config.js +190 -0
- package/dist/auth/types/auth-types.d.ts +417 -0
- package/dist/auth/types/auth-types.js +53 -0
- package/dist/auth/types/index.d.ts +6 -0
- package/dist/auth/types/index.js +22 -0
- package/dist/auth/unauthenticated-routes.d.ts +1 -0
- package/dist/auth/unauthenticated-routes.js +19 -0
- package/dist/auth/utils/idp-client.d.ts +94 -0
- package/dist/auth/utils/idp-client.js +383 -0
- package/dist/auth/utils/index.d.ts +5 -0
- package/dist/auth/utils/index.js +21 -0
- package/dist/auth/utils/token-utils.d.ts +84 -0
- package/dist/auth/utils/token-utils.js +219 -0
- package/dist/client/AuthContext.d.ts +19 -0
- package/dist/client/AuthContext.js +112 -0
- package/dist/client/fetch-with-auth.d.ts +11 -0
- package/dist/client/fetch-with-auth.js +44 -0
- package/dist/client/fetchWithSession.d.ts +3 -0
- package/dist/client/fetchWithSession.js +24 -0
- package/dist/client/index.d.ts +9 -0
- package/dist/client/index.js +20 -0
- package/dist/client/useAnonSession.d.ts +36 -0
- package/dist/client/useAnonSession.js +99 -0
- package/dist/components/SessionSync.d.ts +13 -0
- package/dist/components/SessionSync.js +119 -0
- package/dist/components/SignalRHealthCheck.d.ts +10 -0
- package/dist/components/SignalRHealthCheck.js +97 -0
- package/dist/components/account/UserAvatarMenu.d.ts +20 -0
- package/dist/components/account/UserAvatarMenu.js +80 -0
- package/dist/components/account/index.d.ts +7 -0
- package/dist/components/account/index.js +10 -0
- package/dist/components/admin/AlertSettingsTab.d.ts +48 -0
- package/dist/components/admin/AlertSettingsTab.js +351 -0
- package/dist/components/admin/AnalyticsTab.d.ts +22 -0
- package/dist/components/admin/AnalyticsTab.js +167 -0
- package/dist/components/admin/DataBrowserTab.d.ts +19 -0
- package/dist/components/admin/DataBrowserTab.js +252 -0
- package/dist/components/admin/LoggingSettingsTab.d.ts +73 -0
- package/dist/components/admin/LoggingSettingsTab.js +339 -0
- package/dist/components/admin/SessionsTab.d.ts +37 -0
- package/dist/components/admin/SessionsTab.js +165 -0
- package/dist/components/admin/StatsTab.d.ts +53 -0
- package/dist/components/admin/StatsTab.js +161 -0
- package/dist/components/admin/VibeAdminContext.d.ts +32 -0
- package/dist/components/admin/VibeAdminContext.js +38 -0
- package/dist/components/admin/VibeAdminLayout.d.ts +11 -0
- package/dist/components/admin/VibeAdminLayout.js +69 -0
- package/dist/components/admin/index.d.ts +29 -0
- package/dist/components/admin/index.js +44 -0
- package/dist/components/auth/FederatedAuthSection.d.ts +8 -0
- package/dist/components/auth/FederatedAuthSection.js +45 -0
- package/dist/components/auth/ModeAwareLoginPage.d.ts +10 -0
- package/dist/components/auth/ModeAwareLoginPage.js +42 -0
- package/dist/components/auth/ModeAwareSignupPage.d.ts +9 -0
- package/dist/components/auth/ModeAwareSignupPage.js +78 -0
- package/dist/components/auth/TraditionalAuthSection.d.ts +14 -0
- package/dist/components/auth/TraditionalAuthSection.js +20 -0
- package/dist/components/recovery/CompleteStep.d.ts +5 -0
- package/dist/components/recovery/CompleteStep.js +8 -0
- package/dist/components/recovery/InitiateRecoveryStep.d.ts +8 -0
- package/dist/components/recovery/InitiateRecoveryStep.js +20 -0
- package/dist/components/recovery/SelectMethodStep.d.ts +8 -0
- package/dist/components/recovery/SelectMethodStep.js +8 -0
- package/dist/components/recovery/SetPasswordStep.d.ts +6 -0
- package/dist/components/recovery/SetPasswordStep.js +20 -0
- package/dist/components/recovery/VerifyCodeStep.d.ts +10 -0
- package/dist/components/recovery/VerifyCodeStep.js +24 -0
- package/dist/components/reserved/ReservedRecoveryWarning.d.ts +38 -0
- package/dist/components/reserved/ReservedRecoveryWarning.js +92 -0
- package/dist/components/reserved/ReservedStatusBox.d.ts +30 -0
- package/dist/components/reserved/ReservedStatusBox.js +71 -0
- package/dist/components/ui/BetaBadge.d.ts +29 -0
- package/dist/components/ui/BetaBadge.js +38 -0
- package/dist/components/ui/Footer.d.ts +37 -0
- package/dist/components/ui/Footer.js +41 -0
- package/dist/config/env.d.ts +66 -0
- package/dist/config/env.js +57 -0
- package/dist/config/logger.d.ts +57 -0
- package/dist/config/logger.js +73 -0
- package/dist/config/logging-config.d.ts +30 -0
- package/dist/config/logging-config.js +122 -0
- package/dist/config/unauthenticated-routes.d.ts +17 -0
- package/dist/config/unauthenticated-routes.js +24 -0
- package/dist/config/vibe-log-transport.d.ts +79 -0
- package/dist/config/vibe-log-transport.js +203 -0
- package/dist/edge/internal-api-url.d.ts +53 -0
- package/dist/edge/internal-api-url.js +63 -0
- package/dist/edge/middleware.d.ts +14 -0
- package/dist/edge/middleware.js +32 -0
- package/dist/hooks/useAuth.d.ts +23 -0
- package/dist/hooks/useAuth.js +81 -0
- package/dist/hooks/useAuthSettings.d.ts +59 -0
- package/dist/hooks/useAuthSettings.js +93 -0
- package/dist/hooks/useAvailableProviders.d.ts +45 -0
- package/dist/hooks/useAvailableProviders.js +108 -0
- package/dist/hooks/usePasswordValidation.d.ts +27 -0
- package/dist/hooks/usePasswordValidation.js +102 -0
- package/dist/hooks/useProfile.d.ts +15 -0
- package/dist/hooks/useProfile.js +59 -0
- package/dist/hooks/usePublicAuthSettings.d.ts +56 -0
- package/dist/hooks/usePublicAuthSettings.js +131 -0
- package/dist/hooks/useSessionExpiration.d.ts +57 -0
- package/dist/hooks/useSessionExpiration.js +72 -0
- package/dist/hooks/useViabilitySession.d.ts +75 -0
- package/dist/hooks/useViabilitySession.js +268 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.js +54 -0
- package/dist/lib/anon-session.d.ts +74 -0
- package/dist/lib/anon-session.js +169 -0
- package/dist/lib/api-handler.d.ts +123 -0
- package/dist/lib/api-handler.js +478 -0
- package/dist/lib/app-slug.d.ts +95 -0
- package/dist/lib/app-slug.js +172 -0
- package/dist/lib/demo-mode.d.ts +6 -0
- package/dist/lib/demo-mode.js +16 -0
- package/dist/lib/geolocation.d.ts +64 -0
- package/dist/lib/geolocation.js +235 -0
- package/dist/lib/idp-client-config.d.ts +75 -0
- package/dist/lib/idp-client-config.js +351 -0
- package/dist/lib/idp-fetch.d.ts +14 -0
- package/dist/lib/idp-fetch.js +91 -0
- package/dist/lib/internal-api.d.ts +87 -0
- package/dist/lib/internal-api.js +122 -0
- package/dist/lib/jwt-decode-client.d.ts +10 -0
- package/dist/lib/jwt-decode-client.js +46 -0
- package/dist/lib/jwt-decode.d.ts +48 -0
- package/dist/lib/jwt-decode.js +57 -0
- package/dist/lib/nextauth-secret.d.ts +10 -0
- package/dist/lib/nextauth-secret.js +104 -0
- package/dist/lib/rate-limit-service.d.ts +23 -0
- package/dist/lib/rate-limit-service.js +6 -0
- package/dist/lib/redis.d.ts +5 -0
- package/dist/lib/redis.js +28 -0
- package/dist/lib/refresh-token-validator.d.ts +13 -0
- package/dist/lib/refresh-token-validator.js +117 -0
- package/dist/lib/roles.d.ts +145 -0
- package/dist/lib/roles.js +168 -0
- package/dist/lib/secret-validation.d.ts +4 -0
- package/dist/lib/secret-validation.js +14 -0
- package/dist/lib/session-store.d.ts +166 -0
- package/dist/lib/session-store.js +537 -0
- package/dist/lib/session.d.ts +21 -0
- package/dist/lib/session.js +26 -0
- package/dist/lib/site-logger.d.ts +214 -0
- package/dist/lib/site-logger.js +210 -0
- package/dist/lib/standardized-client-api.d.ts +161 -0
- package/dist/lib/standardized-client-api.js +786 -0
- package/dist/lib/startup-init.d.ts +40 -0
- package/dist/lib/startup-init.js +261 -0
- package/dist/lib/test-aware-get-token.d.ts +2 -0
- package/dist/lib/test-aware-get-token.js +81 -0
- package/dist/lib/token-expiry.d.ts +14 -0
- package/dist/lib/token-expiry.js +39 -0
- package/dist/lib/token-lifecycle.d.ts +52 -0
- package/dist/lib/token-lifecycle.js +398 -0
- package/dist/lib/types/api-responses.d.ts +128 -0
- package/dist/lib/types/api-responses.js +171 -0
- package/dist/lib/user-agent-parser.d.ts +50 -0
- package/dist/lib/user-agent-parser.js +220 -0
- package/dist/logging/api/admin-analytics.d.ts +3 -0
- package/dist/logging/api/admin-analytics.js +45 -0
- package/dist/logging/api/audit-log.d.ts +3 -0
- package/dist/logging/api/audit-log.js +52 -0
- package/dist/logging/components/AdminAnalyticsLayout.d.ts +10 -0
- package/dist/logging/components/AdminAnalyticsLayout.js +11 -0
- package/dist/logging/components/AuditLogViewer.d.ts +7 -0
- package/dist/logging/components/AuditLogViewer.js +51 -0
- package/dist/logging/components/ErrorMetricsCard.d.ts +7 -0
- package/dist/logging/components/ErrorMetricsCard.js +16 -0
- package/dist/logging/components/HealthMetricsCard.d.ts +7 -0
- package/dist/logging/components/HealthMetricsCard.js +19 -0
- package/dist/logging/hooks/useAdminAnalytics.d.ts +24 -0
- package/dist/logging/hooks/useAdminAnalytics.js +22 -0
- package/dist/logging/hooks/useAuditLog.d.ts +6 -0
- package/dist/logging/hooks/useAuditLog.js +25 -0
- package/dist/logging/hooks/useErrorMetrics.d.ts +6 -0
- package/dist/logging/hooks/useErrorMetrics.js +38 -0
- package/dist/logging/hooks/useHealthMetrics.d.ts +6 -0
- package/dist/logging/hooks/useHealthMetrics.js +41 -0
- package/dist/logging/index.d.ts +11 -0
- package/dist/logging/index.js +40 -0
- package/dist/logging/types/analytics.d.ts +68 -0
- package/dist/logging/types/analytics.js +3 -0
- package/dist/logging/types/audit.d.ts +29 -0
- package/dist/logging/types/audit.js +2 -0
- package/dist/logging/types/index.d.ts +2 -0
- package/dist/logging/types/index.js +19 -0
- package/dist/middleware/auth-decision.d.ts +33 -0
- package/dist/middleware/auth-decision.js +65 -0
- package/dist/middleware/create-middleware.d.ts +100 -0
- package/dist/middleware/create-middleware.js +445 -0
- package/dist/middleware/rbac-check.d.ts +44 -0
- package/dist/middleware/rbac-check.js +191 -0
- package/dist/middleware/twofa-presets.d.ts +134 -0
- package/dist/middleware/twofa-presets.js +175 -0
- package/dist/models/DecodedAccessToken.d.ts +17 -0
- package/dist/models/DecodedAccessToken.js +2 -0
- package/dist/models/SessionModel.d.ts +122 -0
- package/dist/models/SessionModel.js +136 -0
- package/dist/pages/admin-login/page.d.ts +31 -0
- package/dist/pages/admin-login/page.js +83 -0
- package/dist/pages/admin-roles/RolesAdminPage.d.ts +15 -0
- package/dist/pages/admin-roles/RolesAdminPage.js +78 -0
- package/dist/pages/admin-roles/index.d.ts +8 -0
- package/dist/pages/admin-roles/index.js +15 -0
- package/dist/pages/admin-roles/modals.d.ts +72 -0
- package/dist/pages/admin-roles/modals.js +154 -0
- package/dist/pages/client-admin/ClientSiteAdminPage.d.ts +79 -0
- package/dist/pages/client-admin/ClientSiteAdminPage.js +177 -0
- package/dist/pages/client-admin/index.d.ts +32 -0
- package/dist/pages/client-admin/index.js +37 -0
- package/dist/pages/login/page.d.ts +22 -0
- package/dist/pages/login/page.js +239 -0
- package/dist/pages/profile/EnhancedProfilePage.d.ts +13 -0
- package/dist/pages/profile/EnhancedProfilePage.js +150 -0
- package/dist/pages/profile/index.d.ts +8 -0
- package/dist/pages/profile/index.js +16 -0
- package/dist/pages/profile/page.d.ts +19 -0
- package/dist/pages/profile/page.js +47 -0
- package/dist/pages/profile/profile-patch.d.ts +1 -0
- package/dist/pages/profile/profile-patch.js +281 -0
- package/dist/pages/recovery/page.d.ts +1 -0
- package/dist/pages/recovery/page.js +142 -0
- package/dist/pages/roles/MyRolesPage.d.ts +24 -0
- package/dist/pages/roles/MyRolesPage.js +71 -0
- package/dist/pages/roles/components.d.ts +63 -0
- package/dist/pages/roles/components.js +108 -0
- package/dist/pages/roles/index.d.ts +8 -0
- package/dist/pages/roles/index.js +19 -0
- package/dist/pages/security/EnhancedSecurityPage.d.ts +14 -0
- package/dist/pages/security/EnhancedSecurityPage.js +248 -0
- package/dist/pages/security/index.d.ts +8 -0
- package/dist/pages/security/index.js +16 -0
- package/dist/pages/security/page.d.ts +21 -0
- package/dist/pages/security/page.js +212 -0
- package/dist/pages/security/security-patch.d.ts +1 -0
- package/dist/pages/security/security-patch.js +302 -0
- package/dist/pages/settings/EnhancedSettingsPage.d.ts +46 -0
- package/dist/pages/settings/EnhancedSettingsPage.js +231 -0
- package/dist/pages/settings/index.d.ts +8 -0
- package/dist/pages/settings/index.js +16 -0
- package/dist/pages/settings/page.d.ts +7 -0
- package/dist/pages/settings/page.js +26 -0
- package/dist/pages/showcase/ShowcasePage.d.ts +13 -0
- package/dist/pages/showcase/ShowcasePage.js +140 -0
- package/dist/pages/showcase/index.d.ts +12 -0
- package/dist/pages/showcase/index.js +17 -0
- package/dist/pages/test-env/EmergencyLogoutPage.d.ts +14 -0
- package/dist/pages/test-env/EmergencyLogoutPage.js +98 -0
- package/dist/pages/test-env/JwtInspectPage.d.ts +14 -0
- package/dist/pages/test-env/JwtInspectPage.js +114 -0
- package/dist/pages/test-env/RefreshTokenPage.d.ts +15 -0
- package/dist/pages/test-env/RefreshTokenPage.js +91 -0
- package/dist/pages/test-env/TestEnvPage.d.ts +13 -0
- package/dist/pages/test-env/TestEnvPage.js +49 -0
- package/dist/pages/test-env/index.d.ts +24 -0
- package/dist/pages/test-env/index.js +32 -0
- package/dist/pages/verify-code/page.d.ts +30 -0
- package/dist/pages/verify-code/page.js +408 -0
- package/dist/routes/account/index.d.ts +28 -0
- package/dist/routes/account/index.js +71 -0
- package/dist/routes/account/masked-info.d.ts +33 -0
- package/dist/routes/account/masked-info.js +39 -0
- package/dist/routes/account/send-code.d.ts +37 -0
- package/dist/routes/account/send-code.js +42 -0
- package/dist/routes/account/update-phone.d.ts +13 -0
- package/dist/routes/account/update-phone.js +17 -0
- package/dist/routes/account/verify-email.d.ts +38 -0
- package/dist/routes/account/verify-email.js +43 -0
- package/dist/routes/account/verify-sms.d.ts +38 -0
- package/dist/routes/account/verify-sms.js +43 -0
- package/dist/routes/auth/index.d.ts +19 -0
- package/dist/routes/auth/index.js +64 -0
- package/dist/routes/auth/logout.d.ts +31 -0
- package/dist/routes/auth/logout.js +113 -0
- package/dist/routes/auth/nextauth.d.ts +19 -0
- package/dist/routes/auth/nextauth.js +72 -0
- package/dist/routes/auth/refresh.d.ts +30 -0
- package/dist/routes/auth/refresh.js +51 -0
- package/dist/routes/auth/session.d.ts +72 -0
- package/dist/routes/auth/session.js +180 -0
- package/dist/routes/auth/settings.d.ts +25 -0
- package/dist/routes/auth/settings.js +55 -0
- package/dist/routes/auth/viability.d.ts +52 -0
- package/dist/routes/auth/viability.js +201 -0
- package/dist/routes/index.d.ts +12 -0
- package/dist/routes/index.js +54 -0
- package/dist/routes/session/index.d.ts +6 -0
- package/dist/routes/session/index.js +10 -0
- package/dist/routes/session/refresh-viability.d.ts +16 -0
- package/dist/routes/session/refresh-viability.js +20 -0
- package/dist/services/signalrActivityService.d.ts +44 -0
- package/dist/services/signalrActivityService.js +257 -0
- package/dist/stores/authStore.d.ts +154 -0
- package/dist/stores/authStore.js +1531 -0
- package/dist/theme/ThemeProvider.d.ts +14 -0
- package/dist/theme/ThemeProvider.js +28 -0
- package/dist/theme/default.d.ts +8 -0
- package/dist/theme/default.js +33 -0
- package/dist/theme/index.d.ts +15 -0
- package/dist/theme/index.js +25 -0
- package/dist/theme/types.d.ts +56 -0
- package/dist/theme/types.js +8 -0
- package/dist/theme/useTheme.d.ts +60 -0
- package/dist/theme/useTheme.js +63 -0
- package/dist/theme/utils.d.ts +13 -0
- package/dist/theme/utils.js +39 -0
- package/dist/types/api.d.ts +134 -0
- package/dist/types/api.js +44 -0
- package/dist/types/auth.d.ts +19 -0
- package/dist/types/auth.js +2 -0
- package/dist/types/logging.d.ts +42 -0
- package/dist/types/logging.js +2 -0
- package/dist/types/recovery.d.ts +48 -0
- package/dist/types/recovery.js +2 -0
- package/dist/types/security.d.ts +1 -0
- package/dist/types/security.js +2 -0
- package/dist/utils/api.d.ts +85 -0
- package/dist/utils/api.js +287 -0
- package/dist/utils/circuitBreaker.d.ts +43 -0
- package/dist/utils/circuitBreaker.js +91 -0
- package/dist/utils/error-message.d.ts +1 -0
- package/dist/utils/error-message.js +103 -0
- package/dist/utils/layout/reservedSpace.d.ts +59 -0
- package/dist/utils/layout/reservedSpace.js +102 -0
- package/dist/utils/logout.d.ts +14 -0
- package/dist/utils/logout.js +32 -0
- package/dist/vibe/client.d.ts +261 -0
- package/dist/vibe/client.js +445 -0
- package/dist/vibe/errors.d.ts +83 -0
- package/dist/vibe/errors.js +146 -0
- package/dist/vibe/generic.d.ts +234 -0
- package/dist/vibe/generic.js +369 -0
- package/dist/vibe/hooks/index.d.ts +169 -0
- package/dist/vibe/hooks/index.js +252 -0
- package/dist/vibe/index.d.ts +23 -0
- package/dist/vibe/index.js +67 -0
- package/dist/vibe/sessions.d.ts +161 -0
- package/dist/vibe/sessions.js +391 -0
- package/dist/vibe/types.d.ts +353 -0
- package/dist/vibe/types.js +315 -0
- package/package.json +855 -0
- package/scripts/check-internal-url-usage.sh +73 -0
- package/scripts/dev-broker.ps1 +35 -0
- package/scripts/dev-local.ps1 +45 -0
- package/src/api/auth-handler.ts +550 -0
- package/src/api/index.ts +18 -0
- package/src/api-handlers/account/change-password.ts +145 -0
- package/src/api-handlers/account/masked-info.ts +45 -0
- package/src/api-handlers/account/profile.ts +80 -0
- package/src/api-handlers/account/recovery/initiate.ts +23 -0
- package/src/api-handlers/account/recovery/send-code.ts +25 -0
- package/src/api-handlers/account/recovery/verify-code.ts +25 -0
- package/src/api-handlers/account/reset-password.ts +23 -0
- package/src/api-handlers/account/send-code.ts +76 -0
- package/src/api-handlers/account/update-phone.ts +79 -0
- package/src/api-handlers/account/validate-password.ts +118 -0
- package/src/api-handlers/account/verify-email.ts +125 -0
- package/src/api-handlers/account/verify-sms.ts +125 -0
- package/src/api-handlers/admin/analytics.ts +445 -0
- package/src/api-handlers/admin/audit.ts +225 -0
- package/src/api-handlers/admin/index.ts +59 -0
- package/src/api-handlers/admin/redis-sessions.ts +253 -0
- package/src/api-handlers/admin/sessions.ts +320 -0
- package/src/api-handlers/admin/site-logs.ts +367 -0
- package/src/api-handlers/admin/users.ts +244 -0
- package/src/api-handlers/admin/vibe-data.ts +326 -0
- package/src/api-handlers/anon/preferences.ts +123 -0
- package/src/api-handlers/auth/jwks.ts +20 -0
- package/src/api-handlers/auth/login.ts +240 -0
- package/src/api-handlers/auth/refresh.ts +687 -0
- package/src/api-handlers/auth/signout.ts +212 -0
- package/src/api-handlers/auth/status.ts +23 -0
- package/src/api-handlers/auth/update-session.ts +125 -0
- package/src/api-handlers/auth/validate.ts +44 -0
- package/src/api-handlers/auth/verify-code.ts +129 -0
- package/src/api-handlers/session/refresh-viability.ts +36 -0
- package/src/api-handlers/session/viability.ts +166 -0
- package/src/api-handlers/test/force-expire.ts +67 -0
- package/src/auth/auth-decision.ts +230 -0
- package/src/auth/auth-options.ts +237 -0
- package/src/auth/callbacks/index.ts +7 -0
- package/src/auth/callbacks/jwt.ts +382 -0
- package/src/auth/callbacks/session.ts +243 -0
- package/src/auth/callbacks/signin.ts +56 -0
- package/src/auth/events/index.ts +5 -0
- package/src/auth/events/signout.ts +33 -0
- package/src/auth/providers/credentials.ts +256 -0
- package/src/auth/providers/index.ts +6 -0
- package/src/auth/providers/oauth.ts +114 -0
- package/src/auth/route-config.ts +220 -0
- package/src/auth/types/auth-types.ts +555 -0
- package/src/auth/types/index.ts +7 -0
- package/src/auth/unauthenticated-routes.ts +3 -0
- package/src/auth/utils/idp-client.ts +444 -0
- package/src/auth/utils/index.ts +6 -0
- package/src/auth/utils/token-utils.ts +244 -0
- package/src/client/AuthContext.tsx +140 -0
- package/src/client/fetch-with-auth.ts +48 -0
- package/src/client/fetchWithSession.ts +21 -0
- package/src/client/index.ts +13 -0
- package/src/client/useAnonSession.ts +131 -0
- package/src/components/SessionSync.tsx +137 -0
- package/src/components/SignalRHealthCheck.tsx +131 -0
- package/src/components/account/UserAvatarMenu.tsx +217 -0
- package/src/components/account/index.ts +8 -0
- package/src/components/admin/AlertSettingsTab.tsx +728 -0
- package/src/components/admin/AnalyticsTab.tsx +703 -0
- package/src/components/admin/DataBrowserTab.tsx +505 -0
- package/src/components/admin/LoggingSettingsTab.tsx +665 -0
- package/src/components/admin/SessionsTab.tsx +414 -0
- package/src/components/admin/StatsTab.tsx +379 -0
- package/src/components/admin/VibeAdminContext.tsx +87 -0
- package/src/components/admin/VibeAdminLayout.tsx +185 -0
- package/src/components/admin/index.ts +59 -0
- package/src/components/auth/FederatedAuthSection.tsx +95 -0
- package/src/components/auth/ModeAwareLoginPage.tsx +135 -0
- package/src/components/auth/ModeAwareSignupPage.tsx +267 -0
- package/src/components/auth/TraditionalAuthSection.tsx +99 -0
- package/src/components/recovery/CompleteStep.tsx +36 -0
- package/src/components/recovery/InitiateRecoveryStep.tsx +68 -0
- package/src/components/recovery/SelectMethodStep.tsx +73 -0
- package/src/components/recovery/SetPasswordStep.tsx +97 -0
- package/src/components/recovery/VerifyCodeStep.tsx +90 -0
- package/src/components/reserved/ReservedRecoveryWarning.tsx +160 -0
- package/src/components/reserved/ReservedStatusBox.tsx +118 -0
- package/src/components/ui/BetaBadge.tsx +58 -0
- package/src/components/ui/Footer.tsx +93 -0
- package/src/config/env.ts +57 -0
- package/src/config/logger.ts +62 -0
- package/src/config/logging-config.ts +82 -0
- package/src/config/unauthenticated-routes.ts +19 -0
- package/src/config/vibe-log-transport.ts +250 -0
- package/src/edge/internal-api-url.ts +65 -0
- package/src/edge/middleware.ts +42 -0
- package/src/hooks/useAuth.ts +115 -0
- package/src/hooks/useAuthSettings.ts +97 -0
- package/src/hooks/useAvailableProviders.ts +118 -0
- package/src/hooks/usePasswordValidation.ts +127 -0
- package/src/hooks/useProfile.ts +75 -0
- package/src/hooks/usePublicAuthSettings.ts +149 -0
- package/src/hooks/useSessionExpiration.ts +102 -0
- package/src/hooks/useViabilitySession.ts +335 -0
- package/src/index.ts +63 -0
- package/src/lib/anon-session.ts +213 -0
- package/src/lib/api-handler.ts +625 -0
- package/src/lib/app-slug.ts +178 -0
- package/src/lib/demo-mode.ts +13 -0
- package/src/lib/geolocation.ts +265 -0
- package/src/lib/idp-client-config.ts +442 -0
- package/src/lib/idp-fetch.ts +101 -0
- package/src/lib/internal-api.ts +171 -0
- package/src/lib/jwt-decode-client.ts +45 -0
- package/src/lib/jwt-decode.ts +83 -0
- package/src/lib/nextauth-secret.ts +126 -0
- package/src/lib/rate-limit-service.ts +9 -0
- package/src/lib/redis.ts +27 -0
- package/src/lib/refresh-token-validator.ts +64 -0
- package/src/lib/roles.ts +177 -0
- package/src/lib/secret-validation.ts +8 -0
- package/src/lib/session-store.ts +637 -0
- package/src/lib/session.ts +34 -0
- package/src/lib/site-logger.ts +245 -0
- package/src/lib/standardized-client-api.ts +896 -0
- package/src/lib/startup-init.ts +247 -0
- package/src/lib/test-aware-get-token.ts +30 -0
- package/src/lib/token-expiry.ts +40 -0
- package/src/lib/token-lifecycle.ts +477 -0
- package/src/lib/types/api-responses.ts +336 -0
- package/src/lib/user-agent-parser.ts +252 -0
- package/src/logging/api/admin-analytics.ts +51 -0
- package/src/logging/api/audit-log.ts +53 -0
- package/src/logging/components/AdminAnalyticsLayout.tsx +49 -0
- package/src/logging/components/AuditLogViewer.tsx +125 -0
- package/src/logging/components/ErrorMetricsCard.tsx +98 -0
- package/src/logging/components/HealthMetricsCard.tsx +70 -0
- package/src/logging/hooks/useAdminAnalytics.ts +22 -0
- package/src/logging/hooks/useAuditLog.ts +24 -0
- package/src/logging/hooks/useErrorMetrics.ts +40 -0
- package/src/logging/hooks/useHealthMetrics.ts +44 -0
- package/src/logging/index.ts +18 -0
- package/src/logging/types/analytics.ts +81 -0
- package/src/logging/types/audit.ts +31 -0
- package/src/logging/types/index.ts +3 -0
- package/src/middleware/auth-decision.ts +43 -0
- package/src/middleware/create-middleware.ts +626 -0
- package/src/middleware/rbac-check.ts +244 -0
- package/src/middleware/twofa-presets.ts +224 -0
- package/src/models/DecodedAccessToken.ts +17 -0
- package/src/models/SessionModel.ts +258 -0
- package/src/pages/admin-login/page.tsx +229 -0
- package/src/pages/admin-roles/RolesAdminPage.tsx +357 -0
- package/src/pages/admin-roles/index.ts +9 -0
- package/src/pages/admin-roles/modals.tsx +469 -0
- package/src/pages/client-admin/ClientSiteAdminPage.tsx +380 -0
- package/src/pages/client-admin/index.ts +33 -0
- package/src/pages/login/page.tsx +463 -0
- package/src/pages/profile/EnhancedProfilePage.tsx +479 -0
- package/src/pages/profile/index.ts +9 -0
- package/src/pages/profile/page.tsx +166 -0
- package/src/pages/recovery/page.tsx +234 -0
- package/src/pages/roles/MyRolesPage.tsx +211 -0
- package/src/pages/roles/components.tsx +294 -0
- package/src/pages/roles/index.ts +17 -0
- package/src/pages/security/EnhancedSecurityPage.tsx +574 -0
- package/src/pages/security/index.ts +9 -0
- package/src/pages/security/page.tsx +507 -0
- package/src/pages/settings/EnhancedSettingsPage.tsx +642 -0
- package/src/pages/settings/index.ts +9 -0
- package/src/pages/settings/page.tsx +47 -0
- package/src/pages/showcase/ShowcasePage.tsx +530 -0
- package/src/pages/showcase/index.ts +13 -0
- package/src/pages/test-env/EmergencyLogoutPage.tsx +179 -0
- package/src/pages/test-env/JwtInspectPage.tsx +418 -0
- package/src/pages/test-env/RefreshTokenPage.tsx +155 -0
- package/src/pages/test-env/TestEnvPage.tsx +116 -0
- package/src/pages/test-env/index.ts +25 -0
- package/src/pages/verify-code/page.tsx +648 -0
- package/src/routes/account/index.ts +32 -0
- package/src/routes/account/masked-info.ts +37 -0
- package/src/routes/account/send-code.ts +40 -0
- package/src/routes/account/update-phone.ts +13 -0
- package/src/routes/account/verify-email.ts +41 -0
- package/src/routes/account/verify-sms.ts +41 -0
- package/src/routes/auth/index.ts +23 -0
- package/src/routes/auth/logout.ts +127 -0
- package/src/routes/auth/nextauth.ts +71 -0
- package/src/routes/auth/refresh.ts +54 -0
- package/src/routes/auth/session.ts +193 -0
- package/src/routes/auth/settings.ts +75 -0
- package/src/routes/auth/viability.ts +220 -0
- package/src/routes/index.ts +18 -0
- package/src/routes/session/index.ts +7 -0
- package/src/routes/session/refresh-viability.ts +17 -0
- package/src/services/signalrActivityService.ts +258 -0
- package/src/stores/authStore.ts +1904 -0
- package/src/templates/instrumentation.ts +41 -0
- package/src/theme/ThemeProvider.tsx +39 -0
- package/src/theme/default.ts +33 -0
- package/src/theme/index.ts +31 -0
- package/src/theme/types.ts +69 -0
- package/src/theme/useTheme.ts +57 -0
- package/src/theme/utils.ts +40 -0
- package/src/types/api.ts +13 -0
- package/src/types/auth.d.ts +15 -0
- package/src/types/auth.ts +22 -0
- package/src/types/logging.ts +11 -0
- package/src/types/next-auth.d.ts +15 -0
- package/src/types/recovery.ts +54 -0
- package/src/types/security.ts +1 -0
- package/src/utils/api.ts +353 -0
- package/src/utils/circuitBreaker.ts +40 -0
- package/src/utils/error-message.ts +108 -0
- package/src/utils/layout/reservedSpace.ts +124 -0
- package/src/utils/logout.ts +30 -0
- package/src/vibe/client.ts +590 -0
- package/src/vibe/errors.ts +185 -0
- package/src/vibe/generic.ts +429 -0
- package/src/vibe/hooks/index.ts +367 -0
- package/src/vibe/index.ts +121 -0
- package/src/vibe/sessions.ts +551 -0
- package/src/vibe/types.ts +577 -0
|
@@ -0,0 +1,637 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Session Store for `@payez/next-mvp` using ioredis
|
|
3
|
+
*
|
|
4
|
+
* This module provides a Redis-backed session store that is compatible with the
|
|
5
|
+
* `ioredis` client. It handles the creation, retrieval, and deletion of
|
|
6
|
+
* session data, which is the single source of truth for authentication.
|
|
7
|
+
*
|
|
8
|
+
* Includes advanced distributed refresh coordination with version control.
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import redis from './redis';
|
|
12
|
+
import { randomBytes } from 'crypto';
|
|
13
|
+
import { SessionData } from '../models/SessionModel';
|
|
14
|
+
import { getSessionPrefix, getRefreshLockPrefix } from './app-slug';
|
|
15
|
+
import { extractKidFromToken } from '../auth/utils/token-utils';
|
|
16
|
+
|
|
17
|
+
// Re-export SessionData for consumers
|
|
18
|
+
export type { SessionData } from '../models/SessionModel';
|
|
19
|
+
|
|
20
|
+
// Use app-slug prefixes for multi-app isolation
|
|
21
|
+
const getSessionKey = (token: string) => `${getSessionPrefix()}${token}`;
|
|
22
|
+
const getRefreshLockKey = (token: string) => `${getRefreshLockPrefix()}${token}`;
|
|
23
|
+
const getSessionVersionKey = (token: string) => `${getSessionPrefix()}ver:${token}`;
|
|
24
|
+
|
|
25
|
+
const REFRESH_LOCK_TTL = 60; // 60 seconds
|
|
26
|
+
const SESSION_TTL = 3 * 24 * 60 * 60; // 3 days in seconds (matches refresh token lifetime)
|
|
27
|
+
|
|
28
|
+
export interface RefreshLockInfo {
|
|
29
|
+
sessionToken: string;
|
|
30
|
+
acquiredAt: number;
|
|
31
|
+
acquiredBy: string; // request ID or process identifier
|
|
32
|
+
lockVersion: number;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Generates a new session token.
|
|
37
|
+
* @returns A new session token string.
|
|
38
|
+
*/
|
|
39
|
+
export function generateSessionToken(): string {
|
|
40
|
+
return randomBytes(32).toString('hex');
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
/**
|
|
44
|
+
* Creates a new session in Redis.
|
|
45
|
+
*
|
|
46
|
+
* @param data The session data to store.
|
|
47
|
+
* @returns The generated session token (redisSessionId).
|
|
48
|
+
*/
|
|
49
|
+
export async function createSession(data: SessionData): Promise<string> {
|
|
50
|
+
const sessionToken = randomBytes(32).toString('hex');
|
|
51
|
+
const key = getSessionKey(sessionToken);
|
|
52
|
+
const versionKey = getSessionVersionKey(sessionToken);
|
|
53
|
+
|
|
54
|
+
try {
|
|
55
|
+
await redis.multi()
|
|
56
|
+
.setex(key, SESSION_TTL, JSON.stringify(data))
|
|
57
|
+
.setex(versionKey, SESSION_TTL, '1')
|
|
58
|
+
.exec();
|
|
59
|
+
} catch (error) {
|
|
60
|
+
console.error('[SESSION-STORE] Failed to create session:', error);
|
|
61
|
+
throw error;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return sessionToken;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Retrieves a session from Redis.
|
|
69
|
+
*
|
|
70
|
+
* @param sessionToken The session token (redisSessionId) to look up.
|
|
71
|
+
* @returns The session data, or null if not found.
|
|
72
|
+
*/
|
|
73
|
+
export async function getSession(sessionToken: string): Promise<SessionData | null> {
|
|
74
|
+
if (!sessionToken) {
|
|
75
|
+
return null;
|
|
76
|
+
}
|
|
77
|
+
const key = getSessionKey(sessionToken);
|
|
78
|
+
const json = await redis.get(key);
|
|
79
|
+
if (!json) {
|
|
80
|
+
return null;
|
|
81
|
+
}
|
|
82
|
+
try {
|
|
83
|
+
return JSON.parse(json) as SessionData;
|
|
84
|
+
} catch {
|
|
85
|
+
console.error('[SESSION-STORE] Failed to parse session data');
|
|
86
|
+
return null;
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
/**
|
|
91
|
+
* Retrieves a session along with a version identifier for optimistic locking.
|
|
92
|
+
* @param sessionToken The session token to look up.
|
|
93
|
+
* @returns An object with session and version, or null if not found.
|
|
94
|
+
*/
|
|
95
|
+
export async function getSessionWithVersion(sessionToken: string): Promise<{ session: SessionData; version: string } | null> {
|
|
96
|
+
const session = await getSession(sessionToken);
|
|
97
|
+
if (!session) {
|
|
98
|
+
return null;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
const versionKey = getSessionVersionKey(sessionToken);
|
|
102
|
+
const version = await redis.get(versionKey);
|
|
103
|
+
|
|
104
|
+
if (!version) {
|
|
105
|
+
// Session exists but version key missing - use idpAccessTokenExpires as fallback
|
|
106
|
+
const fallbackVersion = String(session.idpAccessTokenExpires || Date.now());
|
|
107
|
+
return { session, version: fallbackVersion };
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
return { session, version };
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
/**
|
|
114
|
+
* Checks if the access token in a session is still fresh (not expired).
|
|
115
|
+
* @param sessionToken The session token to check.
|
|
116
|
+
* @param currentAccessToken The current access token to compare.
|
|
117
|
+
* @param currentVersion Optional version to check for changes.
|
|
118
|
+
* @returns Object with freshness status and latest token info.
|
|
119
|
+
*/
|
|
120
|
+
export async function isAccessTokenFresh(
|
|
121
|
+
sessionToken: string,
|
|
122
|
+
currentAccessToken: string,
|
|
123
|
+
currentVersion?: string
|
|
124
|
+
): Promise<{
|
|
125
|
+
isFresh: boolean;
|
|
126
|
+
latestAccessToken?: string;
|
|
127
|
+
latestVersion?: string;
|
|
128
|
+
versionChanged: boolean;
|
|
129
|
+
}> {
|
|
130
|
+
const sessionWithVersion = await getSessionWithVersion(sessionToken);
|
|
131
|
+
|
|
132
|
+
if (!sessionWithVersion) {
|
|
133
|
+
return { isFresh: false, versionChanged: true };
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
const { session, version } = sessionWithVersion;
|
|
137
|
+
const versionChanged = currentVersion ? version !== currentVersion : false;
|
|
138
|
+
|
|
139
|
+
// If version changed, the token might be stale
|
|
140
|
+
// Use normalized field name (idpAccessToken)
|
|
141
|
+
const isFresh = !versionChanged && session.idpAccessToken === currentAccessToken;
|
|
142
|
+
|
|
143
|
+
return {
|
|
144
|
+
isFresh,
|
|
145
|
+
latestAccessToken: session.idpAccessToken || undefined,
|
|
146
|
+
latestVersion: version,
|
|
147
|
+
versionChanged
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
/**
|
|
152
|
+
* Deletes a session from Redis.
|
|
153
|
+
* @param sessionToken The session token to delete.
|
|
154
|
+
*/
|
|
155
|
+
export async function deleteSession(sessionToken: string): Promise<void> {
|
|
156
|
+
if (!sessionToken) {
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
const key = getSessionKey(sessionToken);
|
|
160
|
+
const versionKey = getSessionVersionKey(sessionToken);
|
|
161
|
+
await redis.del(key, versionKey);
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
/**
|
|
165
|
+
* Sets a session directly (for testing or migrations).
|
|
166
|
+
* @param sessionToken The session token.
|
|
167
|
+
* @param data The session data.
|
|
168
|
+
*/
|
|
169
|
+
export async function setSession(sessionToken: string, data: SessionData): Promise<void> {
|
|
170
|
+
const key = getSessionKey(sessionToken);
|
|
171
|
+
const versionKey = getSessionVersionKey(sessionToken);
|
|
172
|
+
await redis.multi()
|
|
173
|
+
.setex(key, SESSION_TTL, JSON.stringify(data))
|
|
174
|
+
.incr(versionKey)
|
|
175
|
+
.expire(versionKey, SESSION_TTL)
|
|
176
|
+
.exec();
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
/**
|
|
180
|
+
* Updates tokens within an existing session.
|
|
181
|
+
* @param sessionToken The session token to update.
|
|
182
|
+
* @param updates Partial session data to update.
|
|
183
|
+
* @returns The updated session data, or null if the session was not found.
|
|
184
|
+
*/
|
|
185
|
+
export async function updateSession(
|
|
186
|
+
sessionToken: string,
|
|
187
|
+
updates: Partial<SessionData>
|
|
188
|
+
): Promise<SessionData | null> {
|
|
189
|
+
const key = getSessionKey(sessionToken);
|
|
190
|
+
const versionKey = getSessionVersionKey(sessionToken);
|
|
191
|
+
|
|
192
|
+
// Get current session to merge with updates
|
|
193
|
+
const currentSession = await getSession(sessionToken);
|
|
194
|
+
if (!currentSession) {
|
|
195
|
+
return null;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
// CRITICAL: Track any refresh token changes (check both old and new field names)
|
|
199
|
+
const hadRefreshToken = !!(currentSession.idpRefreshToken || (currentSession as any).refreshToken);
|
|
200
|
+
const willHaveRefreshToken = (updates as any).idpRefreshToken !== undefined
|
|
201
|
+
? !!(updates as any).idpRefreshToken
|
|
202
|
+
: (updates as any).refreshToken !== undefined
|
|
203
|
+
? !!(updates as any).refreshToken
|
|
204
|
+
: hadRefreshToken;
|
|
205
|
+
|
|
206
|
+
if (hadRefreshToken && !willHaveRefreshToken) {
|
|
207
|
+
console.error('[SESSION-STORE] ⚠️ REFRESH_TOKEN_BEING_CLEARED', {
|
|
208
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
209
|
+
userId: currentSession.userId,
|
|
210
|
+
updateKeys: Object.keys(updates),
|
|
211
|
+
refreshTokenInUpdate: (updates as any).idpRefreshToken || (updates as any).refreshToken,
|
|
212
|
+
refreshTokenClearedReason: (updates as any).refreshTokenClearedReason || 'UNKNOWN',
|
|
213
|
+
stack: new Error().stack
|
|
214
|
+
});
|
|
215
|
+
}
|
|
216
|
+
|
|
217
|
+
// Merge current session with updates
|
|
218
|
+
const updatedSession = { ...currentSession, ...updates };
|
|
219
|
+
|
|
220
|
+
// Write the entire updated session back to Redis with version increment
|
|
221
|
+
await redis.multi()
|
|
222
|
+
.setex(key, SESSION_TTL, JSON.stringify(updatedSession))
|
|
223
|
+
.incr(versionKey)
|
|
224
|
+
.expire(versionKey, SESSION_TTL)
|
|
225
|
+
.exec();
|
|
226
|
+
|
|
227
|
+
return updatedSession;
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
/**
|
|
231
|
+
* Transitions a session to a MFA-completed state.
|
|
232
|
+
* @param sessionToken The session token to update.
|
|
233
|
+
* @param tokens The new tokens received after MFA completion.
|
|
234
|
+
* @param mfaMethod The MFA method used (email, sms, totp) - required for token refresh.
|
|
235
|
+
* @returns The updated session data.
|
|
236
|
+
*/
|
|
237
|
+
export async function transitionTo2FASession(
|
|
238
|
+
sessionToken: string,
|
|
239
|
+
tokens: {
|
|
240
|
+
accessToken?: string;
|
|
241
|
+
refreshToken?: string;
|
|
242
|
+
accessTokenExpires?: number;
|
|
243
|
+
refreshTokenExpires?: number;
|
|
244
|
+
// Support new field names
|
|
245
|
+
idpAccessToken?: string;
|
|
246
|
+
idpRefreshToken?: string;
|
|
247
|
+
idpAccessTokenExpires?: number;
|
|
248
|
+
idpRefreshTokenExpires?: number;
|
|
249
|
+
},
|
|
250
|
+
mfaMethod?: 'email' | 'sms' | 'totp'
|
|
251
|
+
): Promise<SessionData | null> {
|
|
252
|
+
const newAccessToken = tokens.idpAccessToken || tokens.accessToken;
|
|
253
|
+
|
|
254
|
+
console.log('[transitionTo2FASession] Called with:', {
|
|
255
|
+
sessionToken: sessionToken?.substring(0, 8) + '...',
|
|
256
|
+
mfaMethod,
|
|
257
|
+
hasAccessToken: !!newAccessToken,
|
|
258
|
+
hasRefreshToken: !!(tokens.idpRefreshToken || tokens.refreshToken),
|
|
259
|
+
});
|
|
260
|
+
|
|
261
|
+
// Extract bearerKeyId from the new access token (IDP may use different key after 2FA)
|
|
262
|
+
let bearerKeyId: string | undefined;
|
|
263
|
+
if (newAccessToken) {
|
|
264
|
+
bearerKeyId = extractKidFromToken(newAccessToken);
|
|
265
|
+
if (bearerKeyId) {
|
|
266
|
+
console.log('[transitionTo2FASession] Extracted bearerKeyId (kid) from new JWT header:', bearerKeyId);
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
|
|
270
|
+
// Support both old and new field names in input
|
|
271
|
+
// CRITICAL: Set BOTH mfaVerified (new) AND twoFactorComplete (legacy) for compatibility
|
|
272
|
+
// auth.ts session callback reads twoFactorComplete, so we must set it here
|
|
273
|
+
const updates: Partial<SessionData> = {
|
|
274
|
+
idpAccessToken: newAccessToken,
|
|
275
|
+
idpRefreshToken: tokens.idpRefreshToken || tokens.refreshToken,
|
|
276
|
+
idpAccessTokenExpires: tokens.idpAccessTokenExpires || tokens.accessTokenExpires,
|
|
277
|
+
mfaVerified: true,
|
|
278
|
+
twoFactorComplete: true, // Legacy field - required by auth.ts session callback
|
|
279
|
+
mfaMethod: mfaMethod,
|
|
280
|
+
// Update bearerKeyId if extracted from new token
|
|
281
|
+
...(bearerKeyId && { bearerKeyId }),
|
|
282
|
+
};
|
|
283
|
+
|
|
284
|
+
const refreshExpires = tokens.idpRefreshTokenExpires || tokens.refreshTokenExpires;
|
|
285
|
+
if (refreshExpires !== undefined) {
|
|
286
|
+
updates.idpRefreshTokenExpires = refreshExpires;
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
console.log('[transitionTo2FASession] Updates to apply:', {
|
|
290
|
+
mfaVerified: updates.mfaVerified,
|
|
291
|
+
twoFactorComplete: (updates as any).twoFactorComplete,
|
|
292
|
+
mfaMethod: updates.mfaMethod,
|
|
293
|
+
hasIdpAccessToken: !!updates.idpAccessToken,
|
|
294
|
+
});
|
|
295
|
+
|
|
296
|
+
const result = await updateSession(sessionToken, updates);
|
|
297
|
+
|
|
298
|
+
console.log('[transitionTo2FASession] Result:', {
|
|
299
|
+
success: !!result,
|
|
300
|
+
resultMfaVerified: result?.mfaVerified,
|
|
301
|
+
});
|
|
302
|
+
|
|
303
|
+
return result;
|
|
304
|
+
}
|
|
305
|
+
|
|
306
|
+
/**
|
|
307
|
+
* Updates IDP tokens and their expiries in an existing session.
|
|
308
|
+
* @param sessionToken The session token to update.
|
|
309
|
+
* @param idpAccessToken The new IDP access token.
|
|
310
|
+
* @param idpRefreshToken The new IDP refresh token.
|
|
311
|
+
* @param idpAccessTokenExpires The access token expiry timestamp.
|
|
312
|
+
* @param idpRefreshTokenExpires The refresh token expiry timestamp (optional).
|
|
313
|
+
* @returns The updated session data.
|
|
314
|
+
*/
|
|
315
|
+
export async function updateTokens(
|
|
316
|
+
sessionToken: string,
|
|
317
|
+
idpAccessToken: string,
|
|
318
|
+
idpRefreshToken: string,
|
|
319
|
+
idpAccessTokenExpires: number,
|
|
320
|
+
idpRefreshTokenExpires?: number
|
|
321
|
+
): Promise<SessionData | null> {
|
|
322
|
+
return updateSession(sessionToken, {
|
|
323
|
+
idpAccessToken,
|
|
324
|
+
idpRefreshToken,
|
|
325
|
+
idpAccessTokenExpires,
|
|
326
|
+
idpRefreshTokenExpires,
|
|
327
|
+
} as Partial<SessionData>);
|
|
328
|
+
}
|
|
329
|
+
|
|
330
|
+
/**
|
|
331
|
+
* Marks a session as having completed MFA.
|
|
332
|
+
* @param sessionToken The session token to update.
|
|
333
|
+
* @returns The updated session data.
|
|
334
|
+
*/
|
|
335
|
+
export async function mark2FAComplete(sessionToken: string): Promise<SessionData | null> {
|
|
336
|
+
return updateSession(sessionToken, {
|
|
337
|
+
mfaVerified: true,
|
|
338
|
+
});
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
/**
|
|
342
|
+
* Checks if MFA is complete for a session.
|
|
343
|
+
* @param sessionToken The session token.
|
|
344
|
+
* @returns True if MFA is complete.
|
|
345
|
+
*/
|
|
346
|
+
export async function is2FAComplete(sessionToken: string): Promise<boolean> {
|
|
347
|
+
const session = await getSession(sessionToken);
|
|
348
|
+
return session?.mfaVerified === true;
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
/**
|
|
352
|
+
* Gets IDP tokens from a session.
|
|
353
|
+
* @param sessionToken The session token.
|
|
354
|
+
* @returns The tokens or null if session not found.
|
|
355
|
+
*/
|
|
356
|
+
export async function getTokens(sessionToken: string): Promise<{
|
|
357
|
+
accessToken: string;
|
|
358
|
+
refreshToken: string;
|
|
359
|
+
// Also expose new names for clarity
|
|
360
|
+
idpAccessToken: string;
|
|
361
|
+
idpRefreshToken: string;
|
|
362
|
+
} | null> {
|
|
363
|
+
const session = await getSession(sessionToken);
|
|
364
|
+
if (!session || !session.idpAccessToken || !session.idpRefreshToken) {
|
|
365
|
+
return null;
|
|
366
|
+
}
|
|
367
|
+
return {
|
|
368
|
+
// Legacy names for backward compatibility
|
|
369
|
+
accessToken: session.idpAccessToken,
|
|
370
|
+
refreshToken: session.idpRefreshToken,
|
|
371
|
+
// New normalized names
|
|
372
|
+
idpAccessToken: session.idpAccessToken,
|
|
373
|
+
idpRefreshToken: session.idpRefreshToken,
|
|
374
|
+
};
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
/**
|
|
378
|
+
* Refreshes a JWT session (placeholder for compatibility).
|
|
379
|
+
* @param sessionToken The session token.
|
|
380
|
+
* @returns The session data or null.
|
|
381
|
+
*/
|
|
382
|
+
export async function refreshJWTSession(sessionToken: string): Promise<SessionData | null> {
|
|
383
|
+
return getSession(sessionToken);
|
|
384
|
+
}
|
|
385
|
+
|
|
386
|
+
/**
|
|
387
|
+
* Clears all sessions (for testing only).
|
|
388
|
+
*/
|
|
389
|
+
export async function clearAllSessions(): Promise<void> {
|
|
390
|
+
console.warn('[SESSION-STORE] clearAllSessions called - this should only be used in testing');
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
/**
|
|
394
|
+
* Lists all sessions (for testing/debugging only).
|
|
395
|
+
* @returns An empty array (placeholder).
|
|
396
|
+
*/
|
|
397
|
+
export async function listAllSessions(): Promise<string[]> {
|
|
398
|
+
console.warn('[SESSION-STORE] listAllSessions called - this should only be used in testing');
|
|
399
|
+
return [];
|
|
400
|
+
}
|
|
401
|
+
|
|
402
|
+
// ===============================
|
|
403
|
+
// DISTRIBUTED REFRESH COORDINATION
|
|
404
|
+
// ===============================
|
|
405
|
+
|
|
406
|
+
/**
|
|
407
|
+
* Attempt to acquire a refresh lock for a session
|
|
408
|
+
* Uses Redis SET with NX (Not eXists) for atomic lock acquisition
|
|
409
|
+
*/
|
|
410
|
+
export async function acquireRefreshLock(
|
|
411
|
+
sessionToken: string,
|
|
412
|
+
requestId: string,
|
|
413
|
+
maxWaitMs: number = 5000
|
|
414
|
+
): Promise<{ acquired: boolean; lockInfo?: RefreshLockInfo }> {
|
|
415
|
+
const lockKey = getRefreshLockKey(sessionToken);
|
|
416
|
+
const acquiredAt = Date.now();
|
|
417
|
+
const lockVersion = Math.floor(Math.random() * 1000000);
|
|
418
|
+
|
|
419
|
+
const lockInfo: RefreshLockInfo = {
|
|
420
|
+
sessionToken,
|
|
421
|
+
acquiredAt,
|
|
422
|
+
acquiredBy: requestId,
|
|
423
|
+
lockVersion
|
|
424
|
+
};
|
|
425
|
+
|
|
426
|
+
try {
|
|
427
|
+
// Try to acquire the lock atomically
|
|
428
|
+
const result = await redis.set(lockKey, JSON.stringify(lockInfo), 'PX', REFRESH_LOCK_TTL * 1000, 'NX');
|
|
429
|
+
|
|
430
|
+
if (result === 'OK') {
|
|
431
|
+
console.log('[SESSION-STORE] Refresh lock acquired', {
|
|
432
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
433
|
+
requestId,
|
|
434
|
+
lockVersion
|
|
435
|
+
});
|
|
436
|
+
|
|
437
|
+
return { acquired: true, lockInfo };
|
|
438
|
+
} else {
|
|
439
|
+
// Lock already exists, check if we should wait
|
|
440
|
+
if (maxWaitMs > 0) {
|
|
441
|
+
console.log('[SESSION-STORE] Refresh lock already exists, waiting for release', {
|
|
442
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
443
|
+
requestId,
|
|
444
|
+
maxWaitMs
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
return await waitForRefreshLockRelease(sessionToken, requestId, maxWaitMs);
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
console.warn('[SESSION-STORE] Refresh lock already exists, not waiting', {
|
|
451
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
452
|
+
requestId
|
|
453
|
+
});
|
|
454
|
+
|
|
455
|
+
return { acquired: false };
|
|
456
|
+
}
|
|
457
|
+
} catch (error) {
|
|
458
|
+
console.error('[SESSION-STORE] Failed to acquire refresh lock', {
|
|
459
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
460
|
+
requestId,
|
|
461
|
+
error: error instanceof Error ? error.message : String(error)
|
|
462
|
+
});
|
|
463
|
+
|
|
464
|
+
return { acquired: false };
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
|
|
468
|
+
/**
|
|
469
|
+
* Wait for a refresh lock to be released
|
|
470
|
+
*/
|
|
471
|
+
async function waitForRefreshLockRelease(
|
|
472
|
+
sessionToken: string,
|
|
473
|
+
requestId: string,
|
|
474
|
+
maxWaitMs: number
|
|
475
|
+
): Promise<{ acquired: boolean; lockInfo?: RefreshLockInfo }> {
|
|
476
|
+
const lockKey = getRefreshLockKey(sessionToken);
|
|
477
|
+
const startTime = Date.now();
|
|
478
|
+
const pollInterval = 100;
|
|
479
|
+
|
|
480
|
+
while (Date.now() - startTime < maxWaitMs) {
|
|
481
|
+
try {
|
|
482
|
+
const lockExists = await redis.exists(lockKey);
|
|
483
|
+
|
|
484
|
+
if (!lockExists) {
|
|
485
|
+
// Lock released - do not reacquire here to avoid double refresh
|
|
486
|
+
return { acquired: false };
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
// Wait before next check
|
|
490
|
+
await new Promise(resolve => setTimeout(resolve, pollInterval));
|
|
491
|
+
} catch (error) {
|
|
492
|
+
console.error('[SESSION-STORE] Error while waiting for refresh lock release', {
|
|
493
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
494
|
+
requestId,
|
|
495
|
+
error: error instanceof Error ? error.message : String(error)
|
|
496
|
+
});
|
|
497
|
+
break;
|
|
498
|
+
}
|
|
499
|
+
}
|
|
500
|
+
|
|
501
|
+
console.warn('[SESSION-STORE] Timeout waiting for refresh lock release', {
|
|
502
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
503
|
+
requestId,
|
|
504
|
+
waitedMs: Date.now() - startTime
|
|
505
|
+
});
|
|
506
|
+
|
|
507
|
+
return { acquired: false };
|
|
508
|
+
}
|
|
509
|
+
|
|
510
|
+
/**
|
|
511
|
+
* Release a refresh lock
|
|
512
|
+
* Uses Lua script to ensure atomic validation and release
|
|
513
|
+
*/
|
|
514
|
+
export async function releaseRefreshLock(
|
|
515
|
+
sessionToken: string,
|
|
516
|
+
requestId: string,
|
|
517
|
+
lockVersion?: number
|
|
518
|
+
): Promise<boolean> {
|
|
519
|
+
const lockKey = getRefreshLockKey(sessionToken);
|
|
520
|
+
|
|
521
|
+
try {
|
|
522
|
+
// Lua script for atomic lock validation and release
|
|
523
|
+
const luaScript = `
|
|
524
|
+
local lockKey = KEYS[1]
|
|
525
|
+
local expectedRequestId = ARGV[1]
|
|
526
|
+
local expectedVersion = ARGV[2]
|
|
527
|
+
|
|
528
|
+
local lockData = redis.call('GET', lockKey)
|
|
529
|
+
if not lockData then
|
|
530
|
+
return 0 -- Lock doesn't exist
|
|
531
|
+
end
|
|
532
|
+
|
|
533
|
+
local lockInfo = cjson.decode(lockData)
|
|
534
|
+
if lockInfo.acquiredBy == expectedRequestId then
|
|
535
|
+
if not expectedVersion or expectedVersion == '' or tostring(lockInfo.lockVersion) == expectedVersion then
|
|
536
|
+
redis.call('DEL', lockKey)
|
|
537
|
+
return 1 -- Successfully released
|
|
538
|
+
else
|
|
539
|
+
return -2 -- Version mismatch
|
|
540
|
+
end
|
|
541
|
+
else
|
|
542
|
+
return -1 -- Wrong owner
|
|
543
|
+
end
|
|
544
|
+
`;
|
|
545
|
+
|
|
546
|
+
const result = await redis.eval(
|
|
547
|
+
luaScript,
|
|
548
|
+
1,
|
|
549
|
+
lockKey,
|
|
550
|
+
requestId,
|
|
551
|
+
lockVersion ? lockVersion.toString() : ''
|
|
552
|
+
) as number;
|
|
553
|
+
|
|
554
|
+
if (result === 1) {
|
|
555
|
+
console.log('[SESSION-STORE] Refresh lock released successfully', {
|
|
556
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
557
|
+
requestId,
|
|
558
|
+
lockVersion
|
|
559
|
+
});
|
|
560
|
+
return true;
|
|
561
|
+
} else if (result === 0) {
|
|
562
|
+
console.warn('[SESSION-STORE] Attempted to release non-existent refresh lock', {
|
|
563
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
564
|
+
requestId
|
|
565
|
+
});
|
|
566
|
+
return false;
|
|
567
|
+
} else if (result === -1) {
|
|
568
|
+
console.error('[SESSION-STORE] Attempted to release refresh lock owned by another request', {
|
|
569
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
570
|
+
requestId
|
|
571
|
+
});
|
|
572
|
+
return false;
|
|
573
|
+
} else if (result === -2) {
|
|
574
|
+
console.error('[SESSION-STORE] Lock version mismatch during release', {
|
|
575
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
576
|
+
requestId,
|
|
577
|
+
lockVersion
|
|
578
|
+
});
|
|
579
|
+
return false;
|
|
580
|
+
} else {
|
|
581
|
+
console.error('[SESSION-STORE] Unexpected result from lock release script', {
|
|
582
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
583
|
+
requestId,
|
|
584
|
+
result
|
|
585
|
+
});
|
|
586
|
+
return false;
|
|
587
|
+
}
|
|
588
|
+
} catch (error) {
|
|
589
|
+
console.error('[SESSION-STORE] Failed to release refresh lock', {
|
|
590
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
591
|
+
requestId,
|
|
592
|
+
error: error instanceof Error ? error.message : String(error)
|
|
593
|
+
});
|
|
594
|
+
|
|
595
|
+
return false;
|
|
596
|
+
}
|
|
597
|
+
}
|
|
598
|
+
|
|
599
|
+
/**
|
|
600
|
+
* Check if a refresh lock exists for a session
|
|
601
|
+
*/
|
|
602
|
+
export async function checkRefreshLock(sessionToken: string): Promise<RefreshLockInfo | null> {
|
|
603
|
+
const lockKey = getRefreshLockKey(sessionToken);
|
|
604
|
+
|
|
605
|
+
try {
|
|
606
|
+
const lockData = await redis.get(lockKey);
|
|
607
|
+
|
|
608
|
+
if (!lockData) {
|
|
609
|
+
return null;
|
|
610
|
+
}
|
|
611
|
+
|
|
612
|
+
return JSON.parse(lockData) as RefreshLockInfo;
|
|
613
|
+
} catch (error) {
|
|
614
|
+
console.error('[SESSION-STORE] Failed to check refresh lock', {
|
|
615
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
616
|
+
error: error instanceof Error ? error.message : String(error)
|
|
617
|
+
});
|
|
618
|
+
|
|
619
|
+
return null;
|
|
620
|
+
}
|
|
621
|
+
}
|
|
622
|
+
|
|
623
|
+
/**
|
|
624
|
+
* Simple check if a refresh is currently in progress for a session
|
|
625
|
+
*/
|
|
626
|
+
export async function isRefreshInProgress(sessionToken: string): Promise<boolean> {
|
|
627
|
+
const lock = await checkRefreshLock(sessionToken);
|
|
628
|
+
return lock !== null;
|
|
629
|
+
}
|
|
630
|
+
|
|
631
|
+
/**
|
|
632
|
+
* Force cleanup of expired or orphaned refresh locks
|
|
633
|
+
*/
|
|
634
|
+
export async function cleanupRefreshLocks(): Promise<number> {
|
|
635
|
+
console.warn('[SESSION-STORE] cleanupRefreshLocks called - scanning for expired locks');
|
|
636
|
+
return 0;
|
|
637
|
+
}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
export type MinimalSession = {
|
|
2
|
+
user?: {
|
|
3
|
+
id?: string | null;
|
|
4
|
+
email?: string | null;
|
|
5
|
+
} | null;
|
|
6
|
+
accessToken?: string | null;
|
|
7
|
+
expires?: string;
|
|
8
|
+
[k: string]: any;
|
|
9
|
+
};
|
|
10
|
+
|
|
11
|
+
// AppSession is used by authStore - alias to MinimalSession for compatibility
|
|
12
|
+
export type AppSession = MinimalSession;
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* Strict session validation for client-side guards.
|
|
16
|
+
* A session is considered valid when:
|
|
17
|
+
* - a user object exists with non-empty id and email
|
|
18
|
+
* - an accessToken string exists
|
|
19
|
+
*/
|
|
20
|
+
export function isValidSession(session: MinimalSession | null | undefined): boolean {
|
|
21
|
+
if (!session) return false;
|
|
22
|
+
const u = session.user as any;
|
|
23
|
+
const hasUser = !!u && typeof u.id === 'string' && u.id.length > 0 && typeof u.email === 'string' && u.email.length > 0;
|
|
24
|
+
const hasAccessToken = typeof session.accessToken === 'string' && session.accessToken.length > 0;
|
|
25
|
+
return hasUser && hasAccessToken;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Sanitize session data - returns null if session is invalid
|
|
30
|
+
*/
|
|
31
|
+
export function sanitizeSession(session: MinimalSession | null | undefined): MinimalSession | null {
|
|
32
|
+
if (!isValidSession(session)) return null;
|
|
33
|
+
return session as MinimalSession;
|
|
34
|
+
}
|