@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,444 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* IDP Client Utilities
|
|
3
|
+
*
|
|
4
|
+
* Functions for calling PayEz IDP API endpoints.
|
|
5
|
+
* Handles login, OAuth callback, token refresh, and 2FA verification.
|
|
6
|
+
*
|
|
7
|
+
* URL USAGE:
|
|
8
|
+
* - IDP_URL: Used for all calls to the PayEz Identity Provider
|
|
9
|
+
* - INTERNAL_API_URL: NOT used here - that's for calling THIS app's own endpoints
|
|
10
|
+
*
|
|
11
|
+
* @version 1.0.0
|
|
12
|
+
* @since auth-refactor-2026-01
|
|
13
|
+
*/
|
|
14
|
+
|
|
15
|
+
import type {
|
|
16
|
+
IdpLoginResponse,
|
|
17
|
+
IdpOAuthCallbackResponse,
|
|
18
|
+
IdpRefreshResponse,
|
|
19
|
+
LoginCredentials,
|
|
20
|
+
} from '../types/auth-types';
|
|
21
|
+
|
|
22
|
+
// ============================================================================
|
|
23
|
+
// CONFIGURATION
|
|
24
|
+
// ============================================================================
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Get IDP base URL. Throws if not configured.
|
|
28
|
+
*/
|
|
29
|
+
export function getIdpUrl(): string {
|
|
30
|
+
const url = process.env.IDP_URL;
|
|
31
|
+
if (!url) {
|
|
32
|
+
throw new Error('[IDP_CLIENT] FATAL: IDP_URL environment variable is REQUIRED');
|
|
33
|
+
}
|
|
34
|
+
return url.replace(/\/$/, ''); // Remove trailing slash
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Get client ID for this application.
|
|
39
|
+
*/
|
|
40
|
+
export function getClientId(): string {
|
|
41
|
+
const clientId = process.env.CLIENT_ID || process.env.NEXT_PUBLIC_CLIENT_ID;
|
|
42
|
+
if (!clientId) {
|
|
43
|
+
throw new Error('[IDP_CLIENT] FATAL: CLIENT_ID environment variable is REQUIRED');
|
|
44
|
+
}
|
|
45
|
+
return clientId;
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
// ============================================================================
|
|
49
|
+
// LOGIN
|
|
50
|
+
// ============================================================================
|
|
51
|
+
|
|
52
|
+
/**
|
|
53
|
+
* Authenticate user with email/password via IDP.
|
|
54
|
+
*
|
|
55
|
+
* @param credentials - User's email and password
|
|
56
|
+
* @param clientHeaders - Headers to forward (IP, User-Agent for audit)
|
|
57
|
+
* @returns IDP login response with tokens or error
|
|
58
|
+
*/
|
|
59
|
+
export async function idpLogin(
|
|
60
|
+
credentials: LoginCredentials,
|
|
61
|
+
clientHeaders?: { ip?: string; userAgent?: string }
|
|
62
|
+
): Promise<IdpLoginResponse> {
|
|
63
|
+
const idpUrl = getIdpUrl();
|
|
64
|
+
const clientId = getClientId();
|
|
65
|
+
|
|
66
|
+
const headers: Record<string, string> = {
|
|
67
|
+
'Content-Type': 'application/json',
|
|
68
|
+
'X-Client-Id': clientId,
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
// Forward client IP for audit logging
|
|
72
|
+
if (clientHeaders?.ip) {
|
|
73
|
+
headers['X-Forwarded-For'] = clientHeaders.ip;
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
// Forward User-Agent for audit logging
|
|
77
|
+
if (clientHeaders?.userAgent) {
|
|
78
|
+
headers['User-Agent'] = clientHeaders.userAgent;
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
try {
|
|
82
|
+
const response = await fetch(`${idpUrl}/api/ExternalAuth/login`, {
|
|
83
|
+
method: 'POST',
|
|
84
|
+
headers,
|
|
85
|
+
body: JSON.stringify({
|
|
86
|
+
username_or_email: credentials.email,
|
|
87
|
+
password: credentials.password,
|
|
88
|
+
client_id: clientId,
|
|
89
|
+
}),
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
const data = await response.json();
|
|
93
|
+
|
|
94
|
+
// Unwrap PayEz response envelope if present
|
|
95
|
+
const responseData = data.data || data;
|
|
96
|
+
|
|
97
|
+
if (!response.ok || !responseData.result || !responseData.success) {
|
|
98
|
+
return {
|
|
99
|
+
success: false,
|
|
100
|
+
error: responseData.error || {
|
|
101
|
+
code: `HTTP_${response.status}`,
|
|
102
|
+
message: getLoginErrorMessage(response.status, responseData),
|
|
103
|
+
},
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
return {
|
|
108
|
+
success: true,
|
|
109
|
+
result: responseData.result,
|
|
110
|
+
};
|
|
111
|
+
} catch (error) {
|
|
112
|
+
console.error('[IDP_CLIENT] Login request failed:', error);
|
|
113
|
+
return {
|
|
114
|
+
success: false,
|
|
115
|
+
error: {
|
|
116
|
+
code: 'NETWORK_ERROR',
|
|
117
|
+
message: 'Failed to connect to authentication service',
|
|
118
|
+
},
|
|
119
|
+
};
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
/**
|
|
124
|
+
* Get user-friendly error message for login failures.
|
|
125
|
+
*/
|
|
126
|
+
function getLoginErrorMessage(status: number, responseData: any): string {
|
|
127
|
+
// Check for structured error from IDP
|
|
128
|
+
if (responseData?.error?.message) {
|
|
129
|
+
return responseData.error.message;
|
|
130
|
+
}
|
|
131
|
+
|
|
132
|
+
// Fallback to HTTP status-based messages
|
|
133
|
+
switch (status) {
|
|
134
|
+
case 401:
|
|
135
|
+
return 'Invalid email or password. Please try again.';
|
|
136
|
+
case 403:
|
|
137
|
+
return 'Account access denied. Please contact support.';
|
|
138
|
+
case 429:
|
|
139
|
+
return 'Too many login attempts. Please try again later.';
|
|
140
|
+
default:
|
|
141
|
+
if (status >= 500) {
|
|
142
|
+
return 'Authentication service is temporarily unavailable.';
|
|
143
|
+
}
|
|
144
|
+
return 'Authentication failed. Please try again.';
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// ============================================================================
|
|
149
|
+
// OAUTH CALLBACK
|
|
150
|
+
// ============================================================================
|
|
151
|
+
|
|
152
|
+
/**
|
|
153
|
+
* Register/authenticate OAuth user with IDP.
|
|
154
|
+
*
|
|
155
|
+
* Called after OAuth provider (Google, etc.) redirects back.
|
|
156
|
+
* Creates or retrieves IDP user and returns IDP tokens.
|
|
157
|
+
*
|
|
158
|
+
* @param oauthData - Data from OAuth provider
|
|
159
|
+
* @returns IDP response with tokens and user info
|
|
160
|
+
*/
|
|
161
|
+
export async function idpOAuthCallback(oauthData: {
|
|
162
|
+
provider: string;
|
|
163
|
+
providerAccountId: string;
|
|
164
|
+
email: string;
|
|
165
|
+
name?: string;
|
|
166
|
+
image?: string;
|
|
167
|
+
accessToken?: string;
|
|
168
|
+
refreshToken?: string;
|
|
169
|
+
expiresAt?: number;
|
|
170
|
+
}): Promise<IdpOAuthCallbackResponse> {
|
|
171
|
+
const idpUrl = getIdpUrl();
|
|
172
|
+
const clientId = getClientId();
|
|
173
|
+
|
|
174
|
+
try {
|
|
175
|
+
const response = await fetch(`${idpUrl}/api/ExternalAuth/oauth-callback`, {
|
|
176
|
+
method: 'POST',
|
|
177
|
+
headers: {
|
|
178
|
+
'Content-Type': 'application/json',
|
|
179
|
+
'X-Client-Id': clientId,
|
|
180
|
+
},
|
|
181
|
+
body: JSON.stringify({
|
|
182
|
+
provider: oauthData.provider,
|
|
183
|
+
provider_account_id: oauthData.providerAccountId,
|
|
184
|
+
email: oauthData.email,
|
|
185
|
+
name: oauthData.name || '',
|
|
186
|
+
image: oauthData.image || '',
|
|
187
|
+
access_token: oauthData.accessToken || '',
|
|
188
|
+
refresh_token: oauthData.refreshToken || '',
|
|
189
|
+
expires_at: oauthData.expiresAt || 0,
|
|
190
|
+
}),
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
if (!response.ok) {
|
|
194
|
+
const errorText = await response.text().catch(() => 'Unknown error');
|
|
195
|
+
console.error('[IDP_CLIENT] OAuth callback failed:', response.status, errorText);
|
|
196
|
+
return {
|
|
197
|
+
success: false,
|
|
198
|
+
error: {
|
|
199
|
+
code: `HTTP_${response.status}`,
|
|
200
|
+
message: 'OAuth registration failed',
|
|
201
|
+
},
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
const data = await response.json();
|
|
206
|
+
const responseData = data.data || data;
|
|
207
|
+
|
|
208
|
+
// Normalize snake_case to camelCase
|
|
209
|
+
return {
|
|
210
|
+
success: responseData.success !== false,
|
|
211
|
+
data: responseData.success !== false
|
|
212
|
+
? {
|
|
213
|
+
accessToken: responseData.accessToken || responseData.access_token,
|
|
214
|
+
refreshToken: responseData.refreshToken || responseData.refresh_token,
|
|
215
|
+
isNewUser: responseData.isNewUser ?? responseData.is_new_user ?? false,
|
|
216
|
+
user: responseData.user
|
|
217
|
+
? {
|
|
218
|
+
userId: responseData.user.userId || responseData.user.user_id,
|
|
219
|
+
email: responseData.user.email || responseData.user.Email,
|
|
220
|
+
fullName: responseData.user.fullName || responseData.user.full_name || responseData.user.name,
|
|
221
|
+
roles: responseData.user.roles || [],
|
|
222
|
+
}
|
|
223
|
+
: undefined,
|
|
224
|
+
}
|
|
225
|
+
: undefined,
|
|
226
|
+
error: responseData.error,
|
|
227
|
+
};
|
|
228
|
+
} catch (error) {
|
|
229
|
+
console.error('[IDP_CLIENT] OAuth callback request failed:', error);
|
|
230
|
+
return {
|
|
231
|
+
success: false,
|
|
232
|
+
error: {
|
|
233
|
+
code: 'NETWORK_ERROR',
|
|
234
|
+
message: 'Failed to connect to authentication service',
|
|
235
|
+
},
|
|
236
|
+
};
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
// ============================================================================
|
|
241
|
+
// TOKEN REFRESH
|
|
242
|
+
// ============================================================================
|
|
243
|
+
|
|
244
|
+
/**
|
|
245
|
+
* Refresh an expired access token using the refresh token.
|
|
246
|
+
*
|
|
247
|
+
* @param refreshToken - The refresh token from previous login
|
|
248
|
+
* @param mfaContext - MFA context to preserve across refresh
|
|
249
|
+
* @returns New tokens or error
|
|
250
|
+
*/
|
|
251
|
+
export async function idpRefreshToken(
|
|
252
|
+
refreshToken: string,
|
|
253
|
+
mfaContext?: {
|
|
254
|
+
amr?: string[];
|
|
255
|
+
acr?: string;
|
|
256
|
+
twoFactorVerified?: boolean;
|
|
257
|
+
twoFactorMethod?: string;
|
|
258
|
+
twoFactorCompletedAt?: number;
|
|
259
|
+
}
|
|
260
|
+
): Promise<IdpRefreshResponse> {
|
|
261
|
+
const idpUrl = getIdpUrl();
|
|
262
|
+
const clientId = getClientId();
|
|
263
|
+
|
|
264
|
+
const requestBody: Record<string, any> = {
|
|
265
|
+
refresh_token: refreshToken,
|
|
266
|
+
};
|
|
267
|
+
|
|
268
|
+
// Include MFA context so new token preserves authentication level
|
|
269
|
+
if (mfaContext) {
|
|
270
|
+
if (mfaContext.amr) {
|
|
271
|
+
requestBody.amr = mfaContext.amr;
|
|
272
|
+
}
|
|
273
|
+
if (mfaContext.acr) {
|
|
274
|
+
requestBody.acr = mfaContext.acr;
|
|
275
|
+
}
|
|
276
|
+
if (mfaContext.twoFactorVerified) {
|
|
277
|
+
requestBody.two_factor_verified = true;
|
|
278
|
+
}
|
|
279
|
+
if (mfaContext.twoFactorMethod) {
|
|
280
|
+
requestBody.two_factor_method = mfaContext.twoFactorMethod;
|
|
281
|
+
}
|
|
282
|
+
if (mfaContext.twoFactorCompletedAt) {
|
|
283
|
+
requestBody.two_factor_completed_at = new Date(mfaContext.twoFactorCompletedAt).toISOString();
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
|
|
287
|
+
try {
|
|
288
|
+
const response = await fetch(`${idpUrl}/api/ExternalAuth/refresh`, {
|
|
289
|
+
method: 'POST',
|
|
290
|
+
headers: {
|
|
291
|
+
'Content-Type': 'application/json',
|
|
292
|
+
'X-Client-Id': clientId,
|
|
293
|
+
},
|
|
294
|
+
body: JSON.stringify(requestBody),
|
|
295
|
+
});
|
|
296
|
+
|
|
297
|
+
if (!response.ok) {
|
|
298
|
+
const errorText = await response.text().catch(() => 'Unknown error');
|
|
299
|
+
console.error('[IDP_CLIENT] Token refresh failed:', response.status, errorText);
|
|
300
|
+
return {
|
|
301
|
+
success: false,
|
|
302
|
+
error: {
|
|
303
|
+
code: `HTTP_${response.status}`,
|
|
304
|
+
message: response.status === 401 ? 'Refresh token expired' : 'Token refresh failed',
|
|
305
|
+
},
|
|
306
|
+
};
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
const data = await response.json();
|
|
310
|
+
|
|
311
|
+
if (data.success === false) {
|
|
312
|
+
return {
|
|
313
|
+
success: false,
|
|
314
|
+
error: data.error || { code: 'REFRESH_FAILED', message: 'Token refresh failed' },
|
|
315
|
+
};
|
|
316
|
+
}
|
|
317
|
+
|
|
318
|
+
const tokenData = data.data || data;
|
|
319
|
+
|
|
320
|
+
return {
|
|
321
|
+
success: true,
|
|
322
|
+
data: {
|
|
323
|
+
access_token: tokenData.access_token,
|
|
324
|
+
refresh_token: tokenData.refresh_token,
|
|
325
|
+
expires_in: tokenData.expires_in || 3600,
|
|
326
|
+
},
|
|
327
|
+
};
|
|
328
|
+
} catch (error) {
|
|
329
|
+
console.error('[IDP_CLIENT] Token refresh request failed:', error);
|
|
330
|
+
return {
|
|
331
|
+
success: false,
|
|
332
|
+
error: {
|
|
333
|
+
code: 'NETWORK_ERROR',
|
|
334
|
+
message: 'Failed to connect to authentication service',
|
|
335
|
+
},
|
|
336
|
+
};
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
// ============================================================================
|
|
341
|
+
// 2FA VERIFICATION
|
|
342
|
+
// ============================================================================
|
|
343
|
+
|
|
344
|
+
/**
|
|
345
|
+
* Verify 2FA code with IDP.
|
|
346
|
+
*
|
|
347
|
+
* @param sessionToken - Redis session ID
|
|
348
|
+
* @param code - The 2FA code entered by user
|
|
349
|
+
* @param method - The 2FA method ('email' | 'sms' | 'totp')
|
|
350
|
+
* @returns Success status and updated tokens
|
|
351
|
+
*/
|
|
352
|
+
export async function idpVerify2FA(
|
|
353
|
+
accessToken: string,
|
|
354
|
+
code: string,
|
|
355
|
+
method: 'email' | 'sms' | 'totp'
|
|
356
|
+
): Promise<{ success: boolean; error?: { code: string; message: string } }> {
|
|
357
|
+
const idpUrl = getIdpUrl();
|
|
358
|
+
const clientId = getClientId();
|
|
359
|
+
|
|
360
|
+
try {
|
|
361
|
+
const response = await fetch(`${idpUrl}/api/ExternalAuth/verify-2fa`, {
|
|
362
|
+
method: 'POST',
|
|
363
|
+
headers: {
|
|
364
|
+
'Content-Type': 'application/json',
|
|
365
|
+
'X-Client-Id': clientId,
|
|
366
|
+
Authorization: `Bearer ${accessToken}`,
|
|
367
|
+
},
|
|
368
|
+
body: JSON.stringify({
|
|
369
|
+
code,
|
|
370
|
+
method,
|
|
371
|
+
}),
|
|
372
|
+
});
|
|
373
|
+
|
|
374
|
+
if (!response.ok) {
|
|
375
|
+
const data = await response.json().catch(() => ({}));
|
|
376
|
+
return {
|
|
377
|
+
success: false,
|
|
378
|
+
error: data.error || {
|
|
379
|
+
code: `HTTP_${response.status}`,
|
|
380
|
+
message: response.status === 401 ? 'Invalid code' : '2FA verification failed',
|
|
381
|
+
},
|
|
382
|
+
};
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
return { success: true };
|
|
386
|
+
} catch (error) {
|
|
387
|
+
console.error('[IDP_CLIENT] 2FA verification failed:', error);
|
|
388
|
+
return {
|
|
389
|
+
success: false,
|
|
390
|
+
error: {
|
|
391
|
+
code: 'NETWORK_ERROR',
|
|
392
|
+
message: 'Failed to connect to authentication service',
|
|
393
|
+
},
|
|
394
|
+
};
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
|
|
398
|
+
/**
|
|
399
|
+
* Request a new 2FA code to be sent.
|
|
400
|
+
*
|
|
401
|
+
* @param accessToken - User's access token
|
|
402
|
+
* @param method - How to send the code ('email' | 'sms')
|
|
403
|
+
*/
|
|
404
|
+
export async function idpSend2FACode(
|
|
405
|
+
accessToken: string,
|
|
406
|
+
method: 'email' | 'sms'
|
|
407
|
+
): Promise<{ success: boolean; error?: { code: string; message: string } }> {
|
|
408
|
+
const idpUrl = getIdpUrl();
|
|
409
|
+
const clientId = getClientId();
|
|
410
|
+
|
|
411
|
+
try {
|
|
412
|
+
const response = await fetch(`${idpUrl}/api/ExternalAuth/send-2fa-code`, {
|
|
413
|
+
method: 'POST',
|
|
414
|
+
headers: {
|
|
415
|
+
'Content-Type': 'application/json',
|
|
416
|
+
'X-Client-Id': clientId,
|
|
417
|
+
Authorization: `Bearer ${accessToken}`,
|
|
418
|
+
},
|
|
419
|
+
body: JSON.stringify({ method }),
|
|
420
|
+
});
|
|
421
|
+
|
|
422
|
+
if (!response.ok) {
|
|
423
|
+
const data = await response.json().catch(() => ({}));
|
|
424
|
+
return {
|
|
425
|
+
success: false,
|
|
426
|
+
error: data.error || {
|
|
427
|
+
code: `HTTP_${response.status}`,
|
|
428
|
+
message: 'Failed to send 2FA code',
|
|
429
|
+
},
|
|
430
|
+
};
|
|
431
|
+
}
|
|
432
|
+
|
|
433
|
+
return { success: true };
|
|
434
|
+
} catch (error) {
|
|
435
|
+
console.error('[IDP_CLIENT] Send 2FA code failed:', error);
|
|
436
|
+
return {
|
|
437
|
+
success: false,
|
|
438
|
+
error: {
|
|
439
|
+
code: 'NETWORK_ERROR',
|
|
440
|
+
message: 'Failed to connect to authentication service',
|
|
441
|
+
},
|
|
442
|
+
};
|
|
443
|
+
}
|
|
444
|
+
}
|
|
@@ -0,0 +1,244 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Token Utilities
|
|
3
|
+
*
|
|
4
|
+
* JWT decoding and expiry checking utilities.
|
|
5
|
+
* Extracted from auth-options.ts for clarity.
|
|
6
|
+
*
|
|
7
|
+
* @version 1.0.0
|
|
8
|
+
* @since auth-refactor-2026-01
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import { jwtDecode, decodeJwtHeader, extractKidFromToken, JwtHeader } from '../../lib/jwt-decode';
|
|
12
|
+
import type { DecodedIdpAccessToken } from '../types/auth-types';
|
|
13
|
+
|
|
14
|
+
// Re-export header utilities for consumers
|
|
15
|
+
export { decodeJwtHeader, extractKidFromToken, type JwtHeader } from '../../lib/jwt-decode';
|
|
16
|
+
|
|
17
|
+
// ============================================================================
|
|
18
|
+
// TOKEN DECODING
|
|
19
|
+
// ============================================================================
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Decode an IDP access token and extract claims.
|
|
23
|
+
*
|
|
24
|
+
* @param token - The JWT access token from IDP
|
|
25
|
+
* @returns Decoded token claims, or null if decode fails
|
|
26
|
+
*/
|
|
27
|
+
export function decodeIdpAccessToken(token: string): DecodedIdpAccessToken | null {
|
|
28
|
+
try {
|
|
29
|
+
return jwtDecode<DecodedIdpAccessToken>(token);
|
|
30
|
+
} catch (error) {
|
|
31
|
+
console.error('[TOKEN_UTILS] Failed to decode access token:', error);
|
|
32
|
+
return null;
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
/**
|
|
37
|
+
* Decode both JWT header and payload from an IDP access token.
|
|
38
|
+
* Returns the signing key ID (kid) along with payload claims.
|
|
39
|
+
*
|
|
40
|
+
* @param token - The JWT access token from IDP
|
|
41
|
+
* @returns Object with header (including kid) and payload, or null if decode fails
|
|
42
|
+
*/
|
|
43
|
+
export function decodeIdpAccessTokenFull(token: string): {
|
|
44
|
+
header: JwtHeader;
|
|
45
|
+
payload: DecodedIdpAccessToken;
|
|
46
|
+
bearerKeyId: string | undefined;
|
|
47
|
+
} | null {
|
|
48
|
+
try {
|
|
49
|
+
const header = decodeJwtHeader(token);
|
|
50
|
+
const payload = jwtDecode<DecodedIdpAccessToken>(token);
|
|
51
|
+
|
|
52
|
+
if (!header || !payload) {
|
|
53
|
+
return null;
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
return {
|
|
57
|
+
header,
|
|
58
|
+
payload,
|
|
59
|
+
bearerKeyId: header.kid,
|
|
60
|
+
};
|
|
61
|
+
} catch (error) {
|
|
62
|
+
console.error('[TOKEN_UTILS] Failed to decode access token (full):', error);
|
|
63
|
+
return null;
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* Extract user email from decoded token.
|
|
69
|
+
* Handles multiple possible claim names used by IDP.
|
|
70
|
+
*/
|
|
71
|
+
export function extractEmailFromToken(decoded: DecodedIdpAccessToken): string {
|
|
72
|
+
return (
|
|
73
|
+
decoded.email ||
|
|
74
|
+
decoded['http://schemas.xmlsoap.org/ws/2005/05/identity/claims/name'] ||
|
|
75
|
+
''
|
|
76
|
+
);
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
/**
|
|
80
|
+
* Extract roles from decoded token.
|
|
81
|
+
* Handles both 'role' and 'roles' claims, and both string and array formats.
|
|
82
|
+
*/
|
|
83
|
+
export function extractRolesFromToken(decoded: DecodedIdpAccessToken): string[] {
|
|
84
|
+
const rolesClaim = decoded.role || decoded.roles;
|
|
85
|
+
|
|
86
|
+
if (!rolesClaim) {
|
|
87
|
+
return [];
|
|
88
|
+
}
|
|
89
|
+
|
|
90
|
+
if (Array.isArray(rolesClaim)) {
|
|
91
|
+
return rolesClaim;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (typeof rolesClaim === 'string') {
|
|
95
|
+
// Could be a single role or JSON array string
|
|
96
|
+
try {
|
|
97
|
+
const parsed = JSON.parse(rolesClaim);
|
|
98
|
+
return Array.isArray(parsed) ? parsed : [rolesClaim];
|
|
99
|
+
} catch {
|
|
100
|
+
return [rolesClaim];
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
return [];
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* Extract AMR (Authentication Methods References) from decoded token.
|
|
109
|
+
*/
|
|
110
|
+
export function extractAmrFromToken(decoded: DecodedIdpAccessToken): string[] {
|
|
111
|
+
const amr = decoded.amr;
|
|
112
|
+
|
|
113
|
+
if (!amr) {
|
|
114
|
+
return [];
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
if (Array.isArray(amr)) {
|
|
118
|
+
return amr;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if (typeof amr === 'string') {
|
|
122
|
+
try {
|
|
123
|
+
const parsed = JSON.parse(amr);
|
|
124
|
+
return Array.isArray(parsed) ? parsed : [amr];
|
|
125
|
+
} catch {
|
|
126
|
+
return [amr];
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
return [];
|
|
131
|
+
}
|
|
132
|
+
|
|
133
|
+
// ============================================================================
|
|
134
|
+
// EXPIRY CHECKING
|
|
135
|
+
// ============================================================================
|
|
136
|
+
|
|
137
|
+
/**
|
|
138
|
+
* Check if a token expiry timestamp indicates the token needs refresh.
|
|
139
|
+
*
|
|
140
|
+
* @param expiresAt - Token expiry timestamp (Unix milliseconds)
|
|
141
|
+
* @param bufferMs - How early to trigger refresh (default 5 minutes)
|
|
142
|
+
* @returns true if token is expired or will expire within buffer period
|
|
143
|
+
*/
|
|
144
|
+
export function tokenNeedsRefresh(
|
|
145
|
+
expiresAt: number | undefined,
|
|
146
|
+
bufferMs: number = 5 * 60 * 1000
|
|
147
|
+
): boolean {
|
|
148
|
+
if (!expiresAt) {
|
|
149
|
+
return true; // No expiry info = assume needs refresh
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
const timeUntilExpiry = expiresAt - Date.now();
|
|
153
|
+
return timeUntilExpiry <= bufferMs;
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
/**
|
|
157
|
+
* Check if a token is completely expired (past its exp time).
|
|
158
|
+
*
|
|
159
|
+
* @param expiresAt - Token expiry timestamp (Unix milliseconds)
|
|
160
|
+
* @returns true if token is expired
|
|
161
|
+
*/
|
|
162
|
+
export function tokenIsExpired(expiresAt: number | undefined): boolean {
|
|
163
|
+
if (!expiresAt) {
|
|
164
|
+
return true;
|
|
165
|
+
}
|
|
166
|
+
return Date.now() >= expiresAt;
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/**
|
|
170
|
+
* Calculate milliseconds until token expires.
|
|
171
|
+
*
|
|
172
|
+
* @param expiresAt - Token expiry timestamp (Unix milliseconds)
|
|
173
|
+
* @returns Milliseconds until expiry, or 0 if already expired
|
|
174
|
+
*/
|
|
175
|
+
export function msUntilExpiry(expiresAt: number | undefined): number {
|
|
176
|
+
if (!expiresAt) {
|
|
177
|
+
return 0;
|
|
178
|
+
}
|
|
179
|
+
return Math.max(0, expiresAt - Date.now());
|
|
180
|
+
}
|
|
181
|
+
|
|
182
|
+
/**
|
|
183
|
+
* Convert Unix seconds (from JWT exp claim) to milliseconds.
|
|
184
|
+
*/
|
|
185
|
+
export function expClaimToMs(exp: number): number {
|
|
186
|
+
// JWT exp is in seconds, we use milliseconds internally
|
|
187
|
+
return exp * 1000;
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
// ============================================================================
|
|
191
|
+
// TOKEN VALIDATION
|
|
192
|
+
// ============================================================================
|
|
193
|
+
|
|
194
|
+
/**
|
|
195
|
+
* Validate that an access token's actual JWT exp matches what we have cached.
|
|
196
|
+
* This catches cases where the token was refreshed but cache wasn't updated.
|
|
197
|
+
*
|
|
198
|
+
* @param accessToken - The JWT access token
|
|
199
|
+
* @param cachedExpiresAt - What we think the expiry is (Unix ms)
|
|
200
|
+
* @returns Object with validation result and actual expiry
|
|
201
|
+
*/
|
|
202
|
+
export function validateTokenExpiry(
|
|
203
|
+
accessToken: string,
|
|
204
|
+
cachedExpiresAt: number | undefined
|
|
205
|
+
): { valid: boolean; actualExpiresAt: number | null; mismatch: boolean } {
|
|
206
|
+
try {
|
|
207
|
+
const parts = accessToken.split('.');
|
|
208
|
+
if (parts.length !== 3) {
|
|
209
|
+
return { valid: false, actualExpiresAt: null, mismatch: false };
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
const payload = JSON.parse(Buffer.from(parts[1], 'base64url').toString());
|
|
213
|
+
const actualExpiresAt = payload.exp ? payload.exp * 1000 : null;
|
|
214
|
+
|
|
215
|
+
if (!actualExpiresAt) {
|
|
216
|
+
return { valid: false, actualExpiresAt: null, mismatch: false };
|
|
217
|
+
}
|
|
218
|
+
|
|
219
|
+
const now = Date.now();
|
|
220
|
+
const isExpired = actualExpiresAt < now;
|
|
221
|
+
|
|
222
|
+
// Check for mismatch between cached and actual expiry
|
|
223
|
+
const mismatch = cachedExpiresAt
|
|
224
|
+
? Math.abs(actualExpiresAt - cachedExpiresAt) > 1000 // Allow 1 second tolerance
|
|
225
|
+
: false;
|
|
226
|
+
|
|
227
|
+
if (mismatch) {
|
|
228
|
+
console.warn('[TOKEN_UTILS] Token expiry mismatch detected:', {
|
|
229
|
+
cached: cachedExpiresAt ? new Date(cachedExpiresAt).toISOString() : 'none',
|
|
230
|
+
actual: new Date(actualExpiresAt).toISOString(),
|
|
231
|
+
diff: cachedExpiresAt ? actualExpiresAt - cachedExpiresAt : 'N/A',
|
|
232
|
+
});
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
return {
|
|
236
|
+
valid: !isExpired,
|
|
237
|
+
actualExpiresAt,
|
|
238
|
+
mismatch,
|
|
239
|
+
};
|
|
240
|
+
} catch (error) {
|
|
241
|
+
console.error('[TOKEN_UTILS] Failed to validate token expiry:', error);
|
|
242
|
+
return { valid: false, actualExpiresAt: null, mismatch: false };
|
|
243
|
+
}
|
|
244
|
+
}
|