@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,550 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Enhanced Auth Handler with Coordinated Token Refresh
|
|
3
|
+
*
|
|
4
|
+
* Provides a middleware wrapper that automatically handles token lifecycle:
|
|
5
|
+
* - Checks token expiry before each request
|
|
6
|
+
* - Automatically refreshes expired or near-expired tokens
|
|
7
|
+
* - Uses Redis locks for coordinated refresh (prevents race conditions)
|
|
8
|
+
* - Retries requests on 401 responses with fresh tokens
|
|
9
|
+
*
|
|
10
|
+
* Pattern ported from website-membership simple-api-handler.ts
|
|
11
|
+
*
|
|
12
|
+
* @version 2.1.0
|
|
13
|
+
* @since auth-ready-v2
|
|
14
|
+
*/
|
|
15
|
+
|
|
16
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
17
|
+
import { getToken, JWT } from 'next-auth/jwt';
|
|
18
|
+
import { nanoid } from 'nanoid';
|
|
19
|
+
import {
|
|
20
|
+
getSession,
|
|
21
|
+
updateSession,
|
|
22
|
+
acquireRefreshLock,
|
|
23
|
+
releaseRefreshLock,
|
|
24
|
+
checkRefreshLock,
|
|
25
|
+
type SessionData
|
|
26
|
+
} from '../lib/session-store';
|
|
27
|
+
import { getJwtCookieName } from '../lib/app-slug';
|
|
28
|
+
|
|
29
|
+
export interface AuthContext {
|
|
30
|
+
token: JWT;
|
|
31
|
+
accessToken: string;
|
|
32
|
+
userId: string;
|
|
33
|
+
sessionToken: string;
|
|
34
|
+
refreshToken?: string;
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
export interface AuthHandlerOptions {
|
|
38
|
+
/** Whether authentication is required for this route (default: true) */
|
|
39
|
+
requireAuth?: boolean;
|
|
40
|
+
|
|
41
|
+
/** Automatically refresh expired or near-expired tokens (default: true) */
|
|
42
|
+
autoRefresh?: boolean;
|
|
43
|
+
|
|
44
|
+
/** Buffer time in seconds before token expiry to trigger refresh (default: 300 = 5 minutes) */
|
|
45
|
+
refreshBuffer?: number;
|
|
46
|
+
|
|
47
|
+
/** Retry request on 401 response after refreshing token (default: true) */
|
|
48
|
+
retryOn401?: boolean;
|
|
49
|
+
|
|
50
|
+
/** Maximum number of retry attempts on 401 (default: 1) */
|
|
51
|
+
maxRetries?: number;
|
|
52
|
+
|
|
53
|
+
/** NextAuth secret for JWT decoding */
|
|
54
|
+
nextAuthSecret?: string;
|
|
55
|
+
|
|
56
|
+
/** IDP base URL for refresh requests */
|
|
57
|
+
idpBaseUrl?: string;
|
|
58
|
+
|
|
59
|
+
/** OAuth client ID */
|
|
60
|
+
clientId?: string;
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
export type HandlerFunction = (
|
|
64
|
+
req: NextRequest,
|
|
65
|
+
context: any,
|
|
66
|
+
auth: AuthContext
|
|
67
|
+
) => Promise<NextResponse | Response>;
|
|
68
|
+
|
|
69
|
+
interface RefreshResult {
|
|
70
|
+
success: boolean;
|
|
71
|
+
accessToken?: string;
|
|
72
|
+
refreshToken?: string;
|
|
73
|
+
expiresIn?: number;
|
|
74
|
+
error?: string;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Creates an auth-aware handler with automatic token refresh
|
|
79
|
+
*
|
|
80
|
+
* @example
|
|
81
|
+
* ```typescript
|
|
82
|
+
* import { createAuthHandler } from '@payez/next-mvp/api';
|
|
83
|
+
*
|
|
84
|
+
* const handler = createAuthHandler({ requireAuth: true });
|
|
85
|
+
*
|
|
86
|
+
* export const GET = handler.handle(async (req, context, auth) => {
|
|
87
|
+
* // auth.accessToken is guaranteed to be fresh
|
|
88
|
+
* const response = await fetch('https://api.example.com/data', {
|
|
89
|
+
* headers: { 'Authorization': `Bearer ${auth.accessToken}` }
|
|
90
|
+
* });
|
|
91
|
+
* return NextResponse.json(await response.json());
|
|
92
|
+
* });
|
|
93
|
+
* ```
|
|
94
|
+
*/
|
|
95
|
+
export function createAuthHandler(options: AuthHandlerOptions = {}) {
|
|
96
|
+
const {
|
|
97
|
+
requireAuth = true,
|
|
98
|
+
autoRefresh = true,
|
|
99
|
+
refreshBuffer = 60, // 60 seconds - matches website-membership proven threshold
|
|
100
|
+
retryOn401 = true,
|
|
101
|
+
maxRetries = 1,
|
|
102
|
+
nextAuthSecret = process.env.NEXTAUTH_SECRET,
|
|
103
|
+
idpBaseUrl = process.env.IDP_URL,
|
|
104
|
+
clientId = process.env.CLIENT_ID || process.env.NEXT_PUBLIC_IDP_CLIENT_ID
|
|
105
|
+
} = options;
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* Performs coordinated token refresh with Redis locking
|
|
109
|
+
* This prevents multiple concurrent requests from all trying to refresh simultaneously
|
|
110
|
+
*/
|
|
111
|
+
async function performCoordinatedRefresh(
|
|
112
|
+
sessionToken: string,
|
|
113
|
+
requestId: string
|
|
114
|
+
): Promise<RefreshResult> {
|
|
115
|
+
// Check if refresh is already in progress
|
|
116
|
+
const existingLock = await checkRefreshLock(sessionToken);
|
|
117
|
+
|
|
118
|
+
if (existingLock) {
|
|
119
|
+
console.info('[AUTH_HANDLER] Refresh already in progress, waiting...', {
|
|
120
|
+
requestId,
|
|
121
|
+
lockOwner: existingLock.acquiredBy,
|
|
122
|
+
lockAge: Date.now() - existingLock.acquiredAt
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
// Wait for the refresh to complete
|
|
126
|
+
const waitResult = await waitForRefreshCompletion(sessionToken, requestId, 10000);
|
|
127
|
+
if (waitResult.success) {
|
|
128
|
+
// Get the fresh token from session
|
|
129
|
+
const freshSession = await getSession(sessionToken);
|
|
130
|
+
if (freshSession?.accessToken) {
|
|
131
|
+
return {
|
|
132
|
+
success: true,
|
|
133
|
+
accessToken: freshSession.accessToken,
|
|
134
|
+
refreshToken: freshSession.refreshToken,
|
|
135
|
+
expiresIn: freshSession.accessTokenExpires
|
|
136
|
+
? Math.floor((freshSession.accessTokenExpires - Date.now()) / 1000)
|
|
137
|
+
: undefined
|
|
138
|
+
};
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
return { success: false, error: waitResult.reason || 'Wait for refresh failed' };
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// Try to acquire the refresh lock
|
|
145
|
+
const lockAcquired = await acquireRefreshLock(sessionToken, requestId, 5000);
|
|
146
|
+
|
|
147
|
+
if (!lockAcquired) {
|
|
148
|
+
// Another request grabbed the lock, wait for it
|
|
149
|
+
console.info('[AUTH_HANDLER] Failed to acquire lock, waiting for other request', { requestId });
|
|
150
|
+
const waitResult = await waitForRefreshCompletion(sessionToken, requestId, 10000);
|
|
151
|
+
if (waitResult.success) {
|
|
152
|
+
const freshSession = await getSession(sessionToken);
|
|
153
|
+
if (freshSession?.accessToken) {
|
|
154
|
+
return {
|
|
155
|
+
success: true,
|
|
156
|
+
accessToken: freshSession.accessToken,
|
|
157
|
+
refreshToken: freshSession.refreshToken
|
|
158
|
+
};
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
return { success: false, error: waitResult.reason || 'Wait for refresh failed' };
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
try {
|
|
165
|
+
// Double-check if tokens are still stale after acquiring lock
|
|
166
|
+
const latestSession = await getSession(sessionToken);
|
|
167
|
+
if (latestSession && !tokenNeedsRefresh(latestSession, refreshBuffer)) {
|
|
168
|
+
console.info('[AUTH_HANDLER] Tokens already fresh after acquiring lock, skipping refresh', { requestId });
|
|
169
|
+
return {
|
|
170
|
+
success: true,
|
|
171
|
+
accessToken: latestSession.accessToken,
|
|
172
|
+
refreshToken: latestSession.refreshToken
|
|
173
|
+
};
|
|
174
|
+
}
|
|
175
|
+
|
|
176
|
+
// Actually perform the refresh
|
|
177
|
+
return await executeRefresh(sessionToken, latestSession);
|
|
178
|
+
|
|
179
|
+
} finally {
|
|
180
|
+
// Always release the lock
|
|
181
|
+
await releaseRefreshLock(sessionToken, requestId);
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
|
|
185
|
+
/**
|
|
186
|
+
* Wait for an in-progress refresh to complete
|
|
187
|
+
*/
|
|
188
|
+
async function waitForRefreshCompletion(
|
|
189
|
+
sessionToken: string,
|
|
190
|
+
requestId: string,
|
|
191
|
+
maxWaitMs: number
|
|
192
|
+
): Promise<{ success: boolean; reason?: string }> {
|
|
193
|
+
const startTime = Date.now();
|
|
194
|
+
const pollInterval = 100;
|
|
195
|
+
|
|
196
|
+
while (Date.now() - startTime < maxWaitMs) {
|
|
197
|
+
const lockExists = await checkRefreshLock(sessionToken);
|
|
198
|
+
|
|
199
|
+
if (!lockExists) {
|
|
200
|
+
// Lock released, check if tokens are fresh
|
|
201
|
+
const session = await getSession(sessionToken);
|
|
202
|
+
if (session?.accessToken && !tokenNeedsRefresh(session, refreshBuffer)) {
|
|
203
|
+
return { success: true };
|
|
204
|
+
} else {
|
|
205
|
+
return { success: false, reason: 'Lock released but tokens not fresh' };
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
await new Promise(resolve => setTimeout(resolve, pollInterval));
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
return { success: false, reason: `Timeout waiting for refresh (${maxWaitMs}ms)` };
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
/**
|
|
216
|
+
* Check if token needs refresh based on expiry and buffer
|
|
217
|
+
*/
|
|
218
|
+
function tokenNeedsRefresh(session: SessionData, bufferSeconds: number): boolean {
|
|
219
|
+
if (!session.accessToken) return true;
|
|
220
|
+
|
|
221
|
+
const expires = session.accessTokenExpires || 0;
|
|
222
|
+
const bufferMs = bufferSeconds * 1000;
|
|
223
|
+
const timeUntilExpiry = expires - Date.now();
|
|
224
|
+
|
|
225
|
+
return timeUntilExpiry <= bufferMs;
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
/**
|
|
229
|
+
* Execute the actual token refresh against IDP
|
|
230
|
+
*/
|
|
231
|
+
async function executeRefresh(
|
|
232
|
+
sessionToken: string,
|
|
233
|
+
currentSession: SessionData | null
|
|
234
|
+
): Promise<RefreshResult> {
|
|
235
|
+
try {
|
|
236
|
+
if (!idpBaseUrl || !clientId) {
|
|
237
|
+
console.error('[AUTH_HANDLER] Missing IDP configuration for refresh');
|
|
238
|
+
return { success: false, error: 'Missing IDP configuration' };
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
if (!currentSession) {
|
|
242
|
+
console.error('[AUTH_HANDLER] No session found for refresh');
|
|
243
|
+
return { success: false, error: 'No session found' };
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
if (!currentSession.refreshToken) {
|
|
247
|
+
console.error('[AUTH_HANDLER] No refresh token available');
|
|
248
|
+
return { success: false, error: 'No refresh token' };
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// Extract authentication methods from session
|
|
252
|
+
const authMethods = (currentSession as any).authMethods ||
|
|
253
|
+
((currentSession as any).token?.amr ? JSON.parse((currentSession as any).token.amr) : ['pwd', 'mfa']);
|
|
254
|
+
const authLevel = String((currentSession as any).authenticationLevel || (currentSession as any).token?.acr || '2');
|
|
255
|
+
const twoFactorMethod = (currentSession as any).twoFactorMethod || 'authenticator';
|
|
256
|
+
|
|
257
|
+
// Build refresh request body
|
|
258
|
+
const refreshRequestBody: any = {
|
|
259
|
+
refresh_token: currentSession.refreshToken,
|
|
260
|
+
amr: authMethods,
|
|
261
|
+
acr: authLevel
|
|
262
|
+
};
|
|
263
|
+
|
|
264
|
+
if ((currentSession as any).twoFactorComplete) {
|
|
265
|
+
refreshRequestBody.two_factor_verified = true;
|
|
266
|
+
}
|
|
267
|
+
if (twoFactorMethod) {
|
|
268
|
+
refreshRequestBody.two_factor_method = twoFactorMethod;
|
|
269
|
+
}
|
|
270
|
+
if ((currentSession as any).mfaCompletedAt) {
|
|
271
|
+
refreshRequestBody.two_factor_completed_at = new Date((currentSession as any).mfaCompletedAt).toISOString();
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
console.info('[AUTH_HANDLER] Executing refresh against IDP', {
|
|
275
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
276
|
+
hasRefreshToken: true
|
|
277
|
+
});
|
|
278
|
+
|
|
279
|
+
const response = await fetch(`${idpBaseUrl}/api/ExternalAuth/refresh`, {
|
|
280
|
+
method: 'POST',
|
|
281
|
+
headers: {
|
|
282
|
+
'Content-Type': 'application/json',
|
|
283
|
+
'X-Client-Id': clientId,
|
|
284
|
+
},
|
|
285
|
+
body: JSON.stringify(refreshRequestBody),
|
|
286
|
+
});
|
|
287
|
+
|
|
288
|
+
if (!response.ok) {
|
|
289
|
+
const errorText = await response.text().catch(() => 'Unknown error');
|
|
290
|
+
console.error('[AUTH_HANDLER] Refresh failed:', response.status, errorText);
|
|
291
|
+
return { success: false, error: `Refresh failed: ${response.status}` };
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
const data = await response.json();
|
|
295
|
+
|
|
296
|
+
if (data.success === false) {
|
|
297
|
+
return { success: false, error: data.error?.message || data.message || 'Refresh failed' };
|
|
298
|
+
}
|
|
299
|
+
|
|
300
|
+
const tokenData = data.data || data;
|
|
301
|
+
|
|
302
|
+
if (!tokenData.access_token) {
|
|
303
|
+
console.error('[AUTH_HANDLER] No access token in refresh response');
|
|
304
|
+
return { success: false, error: 'No access token received' };
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
// Update session with new tokens
|
|
308
|
+
const updatedSession = {
|
|
309
|
+
...currentSession,
|
|
310
|
+
accessToken: tokenData.access_token,
|
|
311
|
+
refreshToken: tokenData.refresh_token || currentSession.refreshToken,
|
|
312
|
+
accessTokenExpires: tokenData.expires_in
|
|
313
|
+
? Date.now() + (tokenData.expires_in * 1000)
|
|
314
|
+
: Date.now() + (3600 * 1000),
|
|
315
|
+
};
|
|
316
|
+
|
|
317
|
+
await updateSession(sessionToken, updatedSession);
|
|
318
|
+
|
|
319
|
+
console.info('[AUTH_HANDLER] Token refresh successful', {
|
|
320
|
+
newExpiry: new Date(updatedSession.accessTokenExpires).toISOString()
|
|
321
|
+
});
|
|
322
|
+
|
|
323
|
+
return {
|
|
324
|
+
success: true,
|
|
325
|
+
accessToken: tokenData.access_token,
|
|
326
|
+
refreshToken: tokenData.refresh_token,
|
|
327
|
+
expiresIn: tokenData.expires_in,
|
|
328
|
+
};
|
|
329
|
+
} catch (error) {
|
|
330
|
+
console.error('[AUTH_HANDLER] Refresh exception:', error);
|
|
331
|
+
return { success: false, error: error instanceof Error ? error.message : 'Refresh failed' };
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
|
|
335
|
+
/**
|
|
336
|
+
* Checks if auth context token needs refresh based on expiry and buffer
|
|
337
|
+
*/
|
|
338
|
+
function needsRefresh(auth: AuthContext): boolean {
|
|
339
|
+
if (!autoRefresh) return false;
|
|
340
|
+
|
|
341
|
+
// Check if we have token expiry information
|
|
342
|
+
const token = auth.token as any;
|
|
343
|
+
const expiresAt = token.accessTokenExpires || token.exp;
|
|
344
|
+
|
|
345
|
+
if (!expiresAt) {
|
|
346
|
+
// No expiry info, can't determine if refresh needed
|
|
347
|
+
return false;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
const now = Math.floor(Date.now() / 1000);
|
|
351
|
+
const expiryTime = typeof expiresAt === 'number' && expiresAt > 1000000000000
|
|
352
|
+
? Math.floor(expiresAt / 1000) // Convert milliseconds to seconds
|
|
353
|
+
: expiresAt;
|
|
354
|
+
|
|
355
|
+
// Check if token expires within the buffer period
|
|
356
|
+
return (expiryTime - now) <= refreshBuffer;
|
|
357
|
+
}
|
|
358
|
+
|
|
359
|
+
/**
|
|
360
|
+
* Generate a unique request ID for coordinated refresh
|
|
361
|
+
*/
|
|
362
|
+
function generateRequestId(): string {
|
|
363
|
+
return nanoid();
|
|
364
|
+
}
|
|
365
|
+
|
|
366
|
+
/**
|
|
367
|
+
* Main handler wrapper
|
|
368
|
+
*/
|
|
369
|
+
return {
|
|
370
|
+
handle: (handler: HandlerFunction) => {
|
|
371
|
+
return async (req: NextRequest, context: any = {}) => {
|
|
372
|
+
// Extract token from NextAuth
|
|
373
|
+
const token = await getToken({ req, secret: nextAuthSecret, cookieName: getJwtCookieName() });
|
|
374
|
+
|
|
375
|
+
// Check if auth is required
|
|
376
|
+
if (requireAuth && !token) {
|
|
377
|
+
return NextResponse.json(
|
|
378
|
+
{ error: 'Authentication required', code: 'UNAUTHORIZED' },
|
|
379
|
+
{ status: 401 }
|
|
380
|
+
);
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
// If no token and auth not required, call handler without auth context
|
|
384
|
+
if (!token) {
|
|
385
|
+
return handler(req, context, null as any);
|
|
386
|
+
}
|
|
387
|
+
|
|
388
|
+
// Validate client_slug (token confusion attack prevention)
|
|
389
|
+
// SECURITY: Fail closed - require configuration to be explicitly set
|
|
390
|
+
const expectedClientSlug = process.env.NEXT_PUBLIC_EXPECTED_CLIENT_SLUG;
|
|
391
|
+
|
|
392
|
+
if (!expectedClientSlug) {
|
|
393
|
+
console.error('[AUTH_HANDLER] SECURITY MISCONFIGURATION: NEXT_PUBLIC_EXPECTED_CLIENT_SLUG not set');
|
|
394
|
+
return NextResponse.json(
|
|
395
|
+
{
|
|
396
|
+
error: 'Server configuration error',
|
|
397
|
+
code: 'SECURITY_CONFIGURATION_MISSING'
|
|
398
|
+
},
|
|
399
|
+
{ status: 500 }
|
|
400
|
+
);
|
|
401
|
+
}
|
|
402
|
+
|
|
403
|
+
// Extract client_slug from token (normalize property name)
|
|
404
|
+
const tokenClientSlug = (token as any).client_slug || (token as any).clientSlug;
|
|
405
|
+
|
|
406
|
+
// SECURITY: Require client_slug claim in all tokens (no backward compat)
|
|
407
|
+
if (!tokenClientSlug) {
|
|
408
|
+
console.warn('[AUTH_HANDLER] Token missing required client_slug claim');
|
|
409
|
+
return NextResponse.json(
|
|
410
|
+
{
|
|
411
|
+
error: 'Token missing required claim',
|
|
412
|
+
code: 'TOKEN_MISSING_CLIENT_SLUG'
|
|
413
|
+
},
|
|
414
|
+
{ status: 401 }
|
|
415
|
+
);
|
|
416
|
+
}
|
|
417
|
+
|
|
418
|
+
// SECURITY: Case-insensitive comparison to avoid casing attacks
|
|
419
|
+
if (tokenClientSlug.toLowerCase() !== expectedClientSlug.toLowerCase()) {
|
|
420
|
+
// Log without exposing sensitive details
|
|
421
|
+
console.warn('[AUTH_HANDLER] Token client mismatch detected');
|
|
422
|
+
return NextResponse.json(
|
|
423
|
+
{
|
|
424
|
+
error: 'Token issued for different client',
|
|
425
|
+
code: 'TOKEN_CLIENT_MISMATCH'
|
|
426
|
+
},
|
|
427
|
+
{ status: 401 }
|
|
428
|
+
);
|
|
429
|
+
}
|
|
430
|
+
|
|
431
|
+
// Build initial auth context
|
|
432
|
+
let authContext: AuthContext = {
|
|
433
|
+
token,
|
|
434
|
+
accessToken: (token as any).accessToken || '',
|
|
435
|
+
userId: token.sub || (token as any).userId || '',
|
|
436
|
+
sessionToken: (token as any).redisSessionId || '',
|
|
437
|
+
refreshToken: (token as any).refreshToken,
|
|
438
|
+
};
|
|
439
|
+
|
|
440
|
+
// Check if token needs refresh
|
|
441
|
+
if (needsRefresh(authContext) && authContext.refreshToken) {
|
|
442
|
+
const requestId = generateRequestId();
|
|
443
|
+
console.info('[AUTH_HANDLER] Token near expiry, initiating coordinated refresh', {
|
|
444
|
+
requestId,
|
|
445
|
+
sessionToken: authContext.sessionToken.substring(0, 8) + '...'
|
|
446
|
+
});
|
|
447
|
+
|
|
448
|
+
const refreshResult = await performCoordinatedRefresh(
|
|
449
|
+
authContext.sessionToken,
|
|
450
|
+
requestId
|
|
451
|
+
);
|
|
452
|
+
|
|
453
|
+
if (refreshResult.success && refreshResult.accessToken) {
|
|
454
|
+
// Update auth context with fresh token
|
|
455
|
+
authContext.accessToken = refreshResult.accessToken;
|
|
456
|
+
if (refreshResult.refreshToken) {
|
|
457
|
+
authContext.refreshToken = refreshResult.refreshToken;
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
// Update token object for future checks
|
|
461
|
+
(authContext.token as any).accessToken = refreshResult.accessToken;
|
|
462
|
+
if (refreshResult.expiresIn) {
|
|
463
|
+
(authContext.token as any).accessTokenExpires = Date.now() + (refreshResult.expiresIn * 1000);
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
console.info('[AUTH_HANDLER] Coordinated refresh successful', { requestId });
|
|
467
|
+
} else {
|
|
468
|
+
console.warn('[AUTH_HANDLER] Failed to refresh token:', refreshResult.error);
|
|
469
|
+
// Continue with potentially expired token - handler may still succeed
|
|
470
|
+
}
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
// Attach auth context to request for downstream use (following existing pattern)
|
|
474
|
+
// IMPORTANT: Set this ONCE before the retry loop to avoid overwriting with stale data
|
|
475
|
+
(req as any).__authContext = {
|
|
476
|
+
accessToken: authContext.accessToken,
|
|
477
|
+
userId: authContext.userId,
|
|
478
|
+
sessionToken: authContext.sessionToken,
|
|
479
|
+
};
|
|
480
|
+
|
|
481
|
+
// Call the actual handler
|
|
482
|
+
let response: NextResponse | Response;
|
|
483
|
+
let retryCount = 0;
|
|
484
|
+
|
|
485
|
+
while (retryCount <= maxRetries) {
|
|
486
|
+
try {
|
|
487
|
+
response = await handler(req, context, authContext);
|
|
488
|
+
|
|
489
|
+
// Check if we got a 401 and should retry
|
|
490
|
+
if (
|
|
491
|
+
response.status === 401 &&
|
|
492
|
+
retryOn401 &&
|
|
493
|
+
retryCount < maxRetries &&
|
|
494
|
+
authContext.refreshToken
|
|
495
|
+
) {
|
|
496
|
+
const retryRequestId = generateRequestId();
|
|
497
|
+
console.info('[AUTH_HANDLER] Got 401, attempting coordinated refresh and retry', { retryRequestId });
|
|
498
|
+
|
|
499
|
+
const refreshResult = await performCoordinatedRefresh(
|
|
500
|
+
authContext.sessionToken,
|
|
501
|
+
retryRequestId
|
|
502
|
+
);
|
|
503
|
+
|
|
504
|
+
if (refreshResult.success && refreshResult.accessToken) {
|
|
505
|
+
console.info('[AUTH_HANDLER] Refresh succeeded, updating tokens', { retryRequestId });
|
|
506
|
+
|
|
507
|
+
// Update auth context with fresh token
|
|
508
|
+
authContext.accessToken = refreshResult.accessToken;
|
|
509
|
+
if (refreshResult.refreshToken) {
|
|
510
|
+
authContext.refreshToken = refreshResult.refreshToken;
|
|
511
|
+
}
|
|
512
|
+
|
|
513
|
+
// Update request context
|
|
514
|
+
(req as any).__authContext.accessToken = refreshResult.accessToken;
|
|
515
|
+
|
|
516
|
+
console.info('[AUTH_HANDLER] Updated req.__authContext with new token, retrying request', { retryRequestId });
|
|
517
|
+
|
|
518
|
+
retryCount++;
|
|
519
|
+
continue; // Retry the request
|
|
520
|
+
} else {
|
|
521
|
+
console.warn('[AUTH_HANDLER] Refresh failed on 401 retry:', refreshResult.error);
|
|
522
|
+
break; // Don't retry if refresh failed
|
|
523
|
+
}
|
|
524
|
+
}
|
|
525
|
+
|
|
526
|
+
// Success or non-401 error - return response
|
|
527
|
+
break;
|
|
528
|
+
} catch (error) {
|
|
529
|
+
// Handler threw an error
|
|
530
|
+
console.error('[AUTH_HANDLER] Handler error:', error);
|
|
531
|
+
return NextResponse.json(
|
|
532
|
+
{
|
|
533
|
+
error: 'Internal server error',
|
|
534
|
+
details: error instanceof Error ? error.message : 'Unknown error'
|
|
535
|
+
},
|
|
536
|
+
{ status: 500 }
|
|
537
|
+
);
|
|
538
|
+
}
|
|
539
|
+
}
|
|
540
|
+
|
|
541
|
+
return response!;
|
|
542
|
+
};
|
|
543
|
+
}
|
|
544
|
+
};
|
|
545
|
+
}
|
|
546
|
+
|
|
547
|
+
/**
|
|
548
|
+
* Default export for convenience
|
|
549
|
+
*/
|
|
550
|
+
export default createAuthHandler;
|
package/src/api/index.ts
ADDED
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @payez/next-mvp API Module Exports
|
|
3
|
+
*
|
|
4
|
+
* Provides enhanced API route handlers with automatic token management
|
|
5
|
+
*
|
|
6
|
+
* @version 2.0.0
|
|
7
|
+
* @since auth-ready-v2
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
export {
|
|
11
|
+
createAuthHandler,
|
|
12
|
+
type AuthContext,
|
|
13
|
+
type AuthHandlerOptions,
|
|
14
|
+
type HandlerFunction
|
|
15
|
+
} from './auth-handler';
|
|
16
|
+
|
|
17
|
+
// Default export for convenience
|
|
18
|
+
export { default } from './auth-handler';
|