@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,212 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Authentication Signout API Handler
|
|
3
|
+
*
|
|
4
|
+
* Handles user session termination and cookie cleanup.
|
|
5
|
+
* This handler is designed to be imported and used by Next.js applications
|
|
6
|
+
* using the @payez/next-mvp package.
|
|
7
|
+
*
|
|
8
|
+
* @version 2.0
|
|
9
|
+
* @requires No authentication (public endpoint)
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
13
|
+
import { cookies } from 'next/headers';
|
|
14
|
+
import { deleteSession } from '../../lib/session-store';
|
|
15
|
+
import { getToken } from 'next-auth/jwt';
|
|
16
|
+
import {
|
|
17
|
+
getSessionCookieName,
|
|
18
|
+
getSecureSessionCookieName,
|
|
19
|
+
getCsrfCookieName,
|
|
20
|
+
getSecureCsrfCookieName,
|
|
21
|
+
getCallbackUrlCookieName,
|
|
22
|
+
getJwtCookieName
|
|
23
|
+
} from '../../lib/app-slug';
|
|
24
|
+
|
|
25
|
+
// JWT decode helper - simple base64 decode without verification
|
|
26
|
+
function jwtDecode(token: string): any {
|
|
27
|
+
try {
|
|
28
|
+
const parts = token.split('.');
|
|
29
|
+
if (parts.length !== 3) return null;
|
|
30
|
+
const payload = parts[1];
|
|
31
|
+
const decoded = Buffer.from(payload, 'base64').toString('utf-8');
|
|
32
|
+
return JSON.parse(decoded);
|
|
33
|
+
} catch (e) {
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Add protection headers to all responses
|
|
39
|
+
function addSecurityHeaders(response: NextResponse) {
|
|
40
|
+
response.headers.set('Content-Security-Policy', "default-src 'self'; script-src 'self' 'unsafe-inline'; style-src 'self' 'unsafe-inline'; img-src 'self' data: https:; connect-src 'self'; frame-ancestors 'none'; base-uri 'self'; form-action 'self';");
|
|
41
|
+
response.headers.set('X-Frame-Options', 'DENY');
|
|
42
|
+
response.headers.set('X-Content-Type-Options', 'nosniff');
|
|
43
|
+
response.headers.set('X-XSS-Protection', '1; mode=block');
|
|
44
|
+
response.headers.set('Strict-Transport-Security', 'max-age=31536000; includeSubDomains; preload');
|
|
45
|
+
response.headers.set('Referrer-Policy', 'strict-origin-when-cross-origin');
|
|
46
|
+
response.headers.set('Permissions-Policy', 'geolocation=(), microphone=(), camera=(), payment=(), usb=(), magnetometer=(), gyroscope=(), accelerometer=()');
|
|
47
|
+
response.headers.set('Cache-Control', 'no-store, no-cache, must-revalidate, proxy-revalidate');
|
|
48
|
+
response.headers.set('Pragma', 'no-cache');
|
|
49
|
+
response.headers.set('Expires', '0');
|
|
50
|
+
return response;
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
interface SignoutResponse {
|
|
54
|
+
success: boolean;
|
|
55
|
+
message: string;
|
|
56
|
+
sessionDeleted?: boolean;
|
|
57
|
+
chunkCookiesDeleted?: number;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
interface SignoutConfig {
|
|
61
|
+
nextAuthSecret: string;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
/**
|
|
65
|
+
* Creates a signout handler for Next.js API routes
|
|
66
|
+
*
|
|
67
|
+
* @param config Configuration for NextAuth
|
|
68
|
+
* @returns Next.js POST handler function
|
|
69
|
+
*
|
|
70
|
+
* @example
|
|
71
|
+
* ```typescript
|
|
72
|
+
* // In your app's /app/api/auth/signout/route.ts
|
|
73
|
+
* import { createSignoutHandler } from '@payez/next-mvp/api-handlers/auth/signout';
|
|
74
|
+
*
|
|
75
|
+
* export const POST = createSignoutHandler({
|
|
76
|
+
* nextAuthSecret: process.env.NEXTAUTH_SECRET!
|
|
77
|
+
* });
|
|
78
|
+
* ```
|
|
79
|
+
*/
|
|
80
|
+
export function createSignoutHandler(config: SignoutConfig) {
|
|
81
|
+
const { nextAuthSecret } = config;
|
|
82
|
+
|
|
83
|
+
return async function POST(req: NextRequest) {
|
|
84
|
+
const cookieStore = await cookies();
|
|
85
|
+
|
|
86
|
+
// Get app-slug prefixed cookie names
|
|
87
|
+
const sessionCookieName = getSessionCookieName();
|
|
88
|
+
const secureSessionCookieName = getSecureSessionCookieName();
|
|
89
|
+
|
|
90
|
+
// Handle chunked session tokens (for large JWTs)
|
|
91
|
+
let sessionToken: string | undefined;
|
|
92
|
+
const sessionTokenCookie = cookieStore.get(sessionCookieName);
|
|
93
|
+
|
|
94
|
+
if (sessionTokenCookie?.value) {
|
|
95
|
+
// Single cookie case
|
|
96
|
+
sessionToken = sessionTokenCookie.value;
|
|
97
|
+
} else {
|
|
98
|
+
// Chunked cookie case - reconstruct from parts
|
|
99
|
+
const chunkCookies = cookieStore.getAll()
|
|
100
|
+
.filter(cookie => cookie.name.startsWith(`${sessionCookieName}.`))
|
|
101
|
+
.sort((a, b) => {
|
|
102
|
+
const aIndex = parseInt(a.name.split('.').pop() || '0');
|
|
103
|
+
const bIndex = parseInt(b.name.split('.').pop() || '0');
|
|
104
|
+
return aIndex - bIndex;
|
|
105
|
+
});
|
|
106
|
+
|
|
107
|
+
if (chunkCookies.length > 0) {
|
|
108
|
+
sessionToken = chunkCookies.map(cookie => cookie.value).join('');
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// Get chunk cookies for cleanup count
|
|
113
|
+
const chunkCookies = cookieStore.getAll()
|
|
114
|
+
.filter(cookie => cookie.name.startsWith(`${sessionCookieName}.`));
|
|
115
|
+
|
|
116
|
+
// Decode NextAuth JWT to extract the Redis session UUID before deletion
|
|
117
|
+
let redisSessionToken: string | null = null;
|
|
118
|
+
|
|
119
|
+
// First attempt: NextAuth getToken (verified + robust in most cases)
|
|
120
|
+
// Support both field names: sessionToken (auth.ts JWT) and redisSessionId (legacy)
|
|
121
|
+
try {
|
|
122
|
+
const token = await getToken({ req, secret: nextAuthSecret, cookieName: getJwtCookieName() });
|
|
123
|
+
redisSessionToken = (token as any)?.sessionToken || (token as any)?.redisSessionId || null;
|
|
124
|
+
} catch (e) {
|
|
125
|
+
console.warn('[SIGNOUT] getToken() failed to extract session token (will try manual decode)');
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
// Second attempt: manual decode of the session cookie JWT (no verification)
|
|
129
|
+
if (!redisSessionToken && (sessionToken || cookieStore.getAll().some(c => c.name.startsWith(`${sessionCookieName}.`)))) {
|
|
130
|
+
try {
|
|
131
|
+
// Reconstruct raw JWT from chunked cookies if necessary
|
|
132
|
+
let rawJwt: string | null = null;
|
|
133
|
+
const direct = cookieStore.get(sessionCookieName);
|
|
134
|
+
if (direct?.value) {
|
|
135
|
+
rawJwt = direct.value;
|
|
136
|
+
} else {
|
|
137
|
+
const chunks = cookieStore.getAll()
|
|
138
|
+
.filter(c => c.name.startsWith(`${sessionCookieName}.`))
|
|
139
|
+
.sort((a, b) => {
|
|
140
|
+
const ai = parseInt(a.name.split('.').pop() || '0');
|
|
141
|
+
const bi = parseInt(b.name.split('.').pop() || '0');
|
|
142
|
+
return ai - bi;
|
|
143
|
+
})
|
|
144
|
+
.map(c => c.value);
|
|
145
|
+
if (chunks.length > 0) rawJwt = chunks.join('');
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
if (rawJwt) {
|
|
149
|
+
const decoded: any = jwtDecode(rawJwt);
|
|
150
|
+
if (decoded && typeof decoded === 'object' && typeof decoded.sessionToken === 'string') {
|
|
151
|
+
redisSessionToken = decoded.sessionToken;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
} catch (e) {
|
|
155
|
+
console.warn('[SIGNOUT] Manual JWT decode failed to extract session token');
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
// Delete Redis session if UUID was extracted
|
|
160
|
+
if (redisSessionToken) {
|
|
161
|
+
try {
|
|
162
|
+
await deleteSession(redisSessionToken);
|
|
163
|
+
console.info('[SIGNOUT] Redis session cleanup successful', {
|
|
164
|
+
sessionToken: redisSessionToken.substring(0, 8) + '...'
|
|
165
|
+
});
|
|
166
|
+
} catch (sessionError) {
|
|
167
|
+
// Log error but don't fail the signout process
|
|
168
|
+
console.error('[SIGNOUT] Redis session cleanup failed', { error: sessionError });
|
|
169
|
+
}
|
|
170
|
+
} else {
|
|
171
|
+
console.warn('[SIGNOUT] No Redis session token (UUID) extracted from cookie; skipping Redis deletion');
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
// Build response
|
|
175
|
+
const responseData: SignoutResponse = {
|
|
176
|
+
success: true,
|
|
177
|
+
message: sessionToken ? 'Session deleted successfully' : 'No active session found',
|
|
178
|
+
sessionDeleted: !!sessionToken,
|
|
179
|
+
chunkCookiesDeleted: chunkCookies.length
|
|
180
|
+
};
|
|
181
|
+
|
|
182
|
+
const response = NextResponse.json(responseData);
|
|
183
|
+
|
|
184
|
+
// Always clear cookies, regardless of success/failure (using app-slug prefixed names)
|
|
185
|
+
try {
|
|
186
|
+
response.cookies.delete(sessionCookieName);
|
|
187
|
+
response.cookies.delete(secureSessionCookieName);
|
|
188
|
+
response.cookies.delete(getCsrfCookieName());
|
|
189
|
+
response.cookies.delete(getSecureCsrfCookieName());
|
|
190
|
+
response.cookies.delete(getCallbackUrlCookieName());
|
|
191
|
+
response.cookies.delete(`__Secure-${getCallbackUrlCookieName()}`);
|
|
192
|
+
response.cookies.delete('twoFactorSessionVerified');
|
|
193
|
+
|
|
194
|
+
// Delete chunked session cookies
|
|
195
|
+
chunkCookies.forEach(cookie => {
|
|
196
|
+
response.cookies.delete(cookie.name);
|
|
197
|
+
});
|
|
198
|
+
} catch (cookieError) {
|
|
199
|
+
console.error('[SIGNOUT] Cookie cleanup failed:', cookieError);
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
return addSecurityHeaders(response);
|
|
203
|
+
};
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
/**
|
|
207
|
+
* Default export for backward compatibility
|
|
208
|
+
* Requires environment variable: NEXTAUTH_SECRET
|
|
209
|
+
*/
|
|
210
|
+
export const POST = createSignoutHandler({
|
|
211
|
+
nextAuthSecret: process.env.NEXTAUTH_SECRET || ''
|
|
212
|
+
});
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { getToken } from 'next-auth/jwt';
|
|
3
|
+
import { resolveNextAuthSecret } from '../../lib/nextauth-secret';
|
|
4
|
+
import { getSession as getRedisSession } from '../../lib/session-store';
|
|
5
|
+
import { getJwtCookieName } from '../../lib/app-slug';
|
|
6
|
+
|
|
7
|
+
export async function GET(req: NextRequest) {
|
|
8
|
+
try {
|
|
9
|
+
let token: any = await getToken({ req, secret: await resolveNextAuthSecret(), cookieName: getJwtCookieName() });
|
|
10
|
+
// Support both field names: sessionToken (auth.ts JWT) and redisSessionId (legacy)
|
|
11
|
+
const sessionToken = (token as any)?.sessionToken || (token as any)?.redisSessionId;
|
|
12
|
+
if (!sessionToken) {
|
|
13
|
+
return NextResponse.json({ success: false, error: 'No session token' }, { status: 401 });
|
|
14
|
+
}
|
|
15
|
+
const sessionModel = await getRedisSession(sessionToken);
|
|
16
|
+
if (!sessionModel) {
|
|
17
|
+
return NextResponse.json({ success: false, error: 'Session missing in Redis' }, { status: 401 });
|
|
18
|
+
}
|
|
19
|
+
return NextResponse.json({ success: true, userId: (sessionModel as any).userId || null });
|
|
20
|
+
} catch (err) {
|
|
21
|
+
return NextResponse.json({ success: false, error: err instanceof Error ? err.message : 'Unknown error' }, { status: 500 });
|
|
22
|
+
}
|
|
23
|
+
}
|
|
@@ -0,0 +1,125 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Authentication Update Session API Handler
|
|
3
|
+
*
|
|
4
|
+
* Handles session updates, particularly for 2FA status changes.
|
|
5
|
+
* This handler is designed to be imported and used by Next.js applications
|
|
6
|
+
* using the @payez/next-mvp package.
|
|
7
|
+
*
|
|
8
|
+
* @version 2.0
|
|
9
|
+
* @requires Authentication (authenticated endpoint)
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
13
|
+
import { getToken } from 'next-auth/jwt';
|
|
14
|
+
import { getJwtCookieName } from '../../lib/app-slug';
|
|
15
|
+
|
|
16
|
+
// Add protection headers to all responses
|
|
17
|
+
function addSecurityHeaders(response: NextResponse) {
|
|
18
|
+
response.headers.set('Content-Security-Policy', "default-src 'self'; script-src 'self'; style-src 'self'; img-src 'self'; connect-src 'self'; frame-ancestors 'none'; base-uri 'self'; form-action 'self';");
|
|
19
|
+
response.headers.set('X-Frame-Options', 'DENY');
|
|
20
|
+
response.headers.set('X-Content-Type-Options', 'nosniff');
|
|
21
|
+
response.headers.set('X-XSS-Protection', '1; mode=block');
|
|
22
|
+
response.headers.set('Strict-Transport-Security', 'max-age=31536000; includeSubDomains; preload');
|
|
23
|
+
response.headers.set('Referrer-Policy', 'strict-origin-when-cross-origin');
|
|
24
|
+
response.headers.set('Permissions-Policy', 'geolocation=(), microphone=(), camera=(), payment=(), usb=(), magnetometer=(), gyroscope=(), accelerometer=()');
|
|
25
|
+
response.headers.set('Cache-Control', 'no-store, no-cache, must-revalidate, proxy-revalidate');
|
|
26
|
+
response.headers.set('Pragma', 'no-cache');
|
|
27
|
+
response.headers.set('Expires', '0');
|
|
28
|
+
return response;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
interface UpdateSessionRequest {
|
|
32
|
+
twoFactorSessionVerified?: boolean;
|
|
33
|
+
twoFactorMethod?: string;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
interface UpdateSessionResponse {
|
|
37
|
+
success: boolean;
|
|
38
|
+
twoFactorSessionVerified: boolean;
|
|
39
|
+
twoFactorMethod?: string;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
interface UpdateSessionConfig {
|
|
43
|
+
nextAuthSecret: string;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* Creates an update-session handler for Next.js API routes
|
|
48
|
+
*
|
|
49
|
+
* @param config Configuration for NextAuth
|
|
50
|
+
* @returns Next.js POST handler function
|
|
51
|
+
*
|
|
52
|
+
* @example
|
|
53
|
+
* ```typescript
|
|
54
|
+
* // In your app's /app/api/auth/update-session/route.ts
|
|
55
|
+
* import { createUpdateSessionHandler } from '@payez/next-mvp/api-handlers/auth/update-session';
|
|
56
|
+
*
|
|
57
|
+
* export const POST = createUpdateSessionHandler({
|
|
58
|
+
* nextAuthSecret: process.env.NEXTAUTH_SECRET!
|
|
59
|
+
* });
|
|
60
|
+
* ```
|
|
61
|
+
*/
|
|
62
|
+
export function createUpdateSessionHandler(config: UpdateSessionConfig) {
|
|
63
|
+
const { nextAuthSecret } = config;
|
|
64
|
+
|
|
65
|
+
return async function POST(req: NextRequest) {
|
|
66
|
+
try {
|
|
67
|
+
let body: UpdateSessionRequest;
|
|
68
|
+
|
|
69
|
+
try {
|
|
70
|
+
body = await req.json();
|
|
71
|
+
} catch (parseError) {
|
|
72
|
+
return addSecurityHeaders(NextResponse.json(
|
|
73
|
+
{ error: 'Invalid JSON format' },
|
|
74
|
+
{ status: 400 }
|
|
75
|
+
));
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const { twoFactorSessionVerified, twoFactorMethod } = body;
|
|
79
|
+
|
|
80
|
+
// Get the current token
|
|
81
|
+
const token = await getToken({ req, secret: nextAuthSecret, cookieName: getJwtCookieName() });
|
|
82
|
+
if (!token) {
|
|
83
|
+
return addSecurityHeaders(NextResponse.json(
|
|
84
|
+
{ error: 'No session token available' },
|
|
85
|
+
{ status: 401 }
|
|
86
|
+
));
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Update the token with 2FA challenge completion status
|
|
90
|
+
const updatedToken = {
|
|
91
|
+
...token,
|
|
92
|
+
twoFactorSessionVerified: !!twoFactorSessionVerified,
|
|
93
|
+
twoFactorMethod: twoFactorMethod || token.twoFactorMethod
|
|
94
|
+
};
|
|
95
|
+
|
|
96
|
+
console.info('[UPDATE-SESSION] Session updated successfully', {
|
|
97
|
+
userId: token.sub,
|
|
98
|
+
twoFactorSessionVerified: updatedToken.twoFactorSessionVerified,
|
|
99
|
+
twoFactorMethod: updatedToken.twoFactorMethod
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
const responseData: UpdateSessionResponse = {
|
|
103
|
+
success: true,
|
|
104
|
+
twoFactorSessionVerified: updatedToken.twoFactorSessionVerified as boolean,
|
|
105
|
+
twoFactorMethod: updatedToken.twoFactorMethod as string | undefined
|
|
106
|
+
};
|
|
107
|
+
|
|
108
|
+
return addSecurityHeaders(NextResponse.json(responseData));
|
|
109
|
+
} catch (error) {
|
|
110
|
+
console.error('[UPDATE-SESSION] Error updating session', { error });
|
|
111
|
+
return addSecurityHeaders(NextResponse.json(
|
|
112
|
+
{ error: 'Failed to update session' },
|
|
113
|
+
{ status: 500 }
|
|
114
|
+
));
|
|
115
|
+
}
|
|
116
|
+
};
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Default export for backward compatibility
|
|
121
|
+
* Requires environment variable: NEXTAUTH_SECRET
|
|
122
|
+
*/
|
|
123
|
+
export const POST = createUpdateSessionHandler({
|
|
124
|
+
nextAuthSecret: process.env.NEXTAUTH_SECRET || ''
|
|
125
|
+
});
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { idpFetchJSON } from '../../lib/idp-fetch';
|
|
3
|
+
import { ENV_CONFIG, API_ENDPOINTS } from '../../config/env';
|
|
4
|
+
import { getTokenTestAware } from '../../lib/test-aware-get-token';
|
|
5
|
+
import { getSession } from '../../lib/session-store';
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* Validates current access token with the IDP and returns normalized info.
|
|
9
|
+
* Sources access token from Authorization header or Redis session via cookie.
|
|
10
|
+
*/
|
|
11
|
+
export async function GET(req: NextRequest) {
|
|
12
|
+
try {
|
|
13
|
+
// 1) Prefer Authorization header if present
|
|
14
|
+
let bearer: string | undefined = undefined;
|
|
15
|
+
const authHeader = req.headers.get('authorization') || req.headers.get('Authorization');
|
|
16
|
+
if (authHeader && /^Bearer\s+/i.test(authHeader)) bearer = authHeader.replace(/^Bearer\s+/i, '').trim();
|
|
17
|
+
|
|
18
|
+
// 2) If not, resolve from session
|
|
19
|
+
if (!bearer) {
|
|
20
|
+
const tok = await getTokenTestAware(req);
|
|
21
|
+
const sessionToken = tok?.sessionToken as string | undefined;
|
|
22
|
+
if (sessionToken) {
|
|
23
|
+
const sess = await getSession(sessionToken);
|
|
24
|
+
bearer = sess?.accessToken;
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
if (!bearer) {
|
|
29
|
+
return NextResponse.json({ success: false, error: { code: 'UNAUTHORIZED', message: 'No access token available' } }, { status: 401 });
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
const url = `${ENV_CONFIG.IDP_URL}${API_ENDPOINTS.externalAuth.validate}`;
|
|
33
|
+
const result = await idpFetchJSON(req, url, { method: 'GET', headers: { Authorization: `Bearer ${bearer}` } });
|
|
34
|
+
|
|
35
|
+
if (!result.ok) {
|
|
36
|
+
return NextResponse.json({ success: false, error: { code: 'UPSTREAM_SERVICE_ERROR', status: result.status }, data: result.json }, { status: result.status });
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
// Passthrough normalized
|
|
40
|
+
return NextResponse.json(result.json ?? { success: true }, { status: 200 });
|
|
41
|
+
} catch (e: any) {
|
|
42
|
+
return NextResponse.json({ success: false, error: { code: 'INTERNAL_SERVER_ERROR', message: e?.message || 'validate error' } }, { status: 500 });
|
|
43
|
+
}
|
|
44
|
+
}
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Authentication Verify Code / Complete 2FA API Handler
|
|
3
|
+
*
|
|
4
|
+
* Handles 2FA verification and token updates after successful verification.
|
|
5
|
+
* This handler is designed to be imported and used by Next.js applications
|
|
6
|
+
* using the @payez/next-mvp package.
|
|
7
|
+
*
|
|
8
|
+
* @version 2.0
|
|
9
|
+
* @requires Authentication (authenticated endpoint)
|
|
10
|
+
*/
|
|
11
|
+
|
|
12
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
13
|
+
import { getToken } from 'next-auth/jwt';
|
|
14
|
+
import { updateTokens, mark2FAComplete } from '../../lib/session-store';
|
|
15
|
+
import { getJwtCookieName } from '../../lib/app-slug';
|
|
16
|
+
|
|
17
|
+
interface VerifyCodeRequest {
|
|
18
|
+
accessToken: string;
|
|
19
|
+
refreshToken: string;
|
|
20
|
+
accessTokenExpires: number;
|
|
21
|
+
refreshTokenExpires?: number;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
interface VerifyCodeConfig {
|
|
25
|
+
nextAuthSecret: string;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Creates a verify-code/complete-2FA handler for Next.js API routes
|
|
30
|
+
*
|
|
31
|
+
* @param config Configuration for NextAuth
|
|
32
|
+
* @returns Next.js POST handler function
|
|
33
|
+
*
|
|
34
|
+
* @example
|
|
35
|
+
* ```typescript
|
|
36
|
+
* // In your app's /app/api/auth/verify-code/route.ts
|
|
37
|
+
* import { createVerifyCodeHandler } from '@payez/next-mvp/api-handlers/auth/verify-code';
|
|
38
|
+
*
|
|
39
|
+
* export const POST = createVerifyCodeHandler({
|
|
40
|
+
* nextAuthSecret: process.env.NEXTAUTH_SECRET!
|
|
41
|
+
* });
|
|
42
|
+
* ```
|
|
43
|
+
*/
|
|
44
|
+
export function createVerifyCodeHandler(config: VerifyCodeConfig) {
|
|
45
|
+
const { nextAuthSecret } = config;
|
|
46
|
+
|
|
47
|
+
return async function POST(req: NextRequest) {
|
|
48
|
+
try {
|
|
49
|
+
let body: VerifyCodeRequest;
|
|
50
|
+
|
|
51
|
+
try {
|
|
52
|
+
body = await req.json();
|
|
53
|
+
} catch (parseError) {
|
|
54
|
+
return NextResponse.json(
|
|
55
|
+
{ success: false, message: 'Invalid JSON format' },
|
|
56
|
+
{ status: 400 }
|
|
57
|
+
);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
const { accessToken, refreshToken, accessTokenExpires, refreshTokenExpires } = body;
|
|
61
|
+
|
|
62
|
+
// Get current session token from JWT
|
|
63
|
+
let token: any = await getToken({ req, secret: nextAuthSecret, cookieName: getJwtCookieName() });
|
|
64
|
+
|
|
65
|
+
// The sessionToken is stored in the JWT token object
|
|
66
|
+
// Support both field names: sessionToken (auth.ts JWT) and redisSessionId (legacy)
|
|
67
|
+
const sessionToken = (token?.sessionToken || token?.redisSessionId) as string;
|
|
68
|
+
|
|
69
|
+
if (!sessionToken) {
|
|
70
|
+
console.error('[VERIFY-CODE] No session token found in JWT', {
|
|
71
|
+
hasToken: !!token,
|
|
72
|
+
tokenKeys: token ? Object.keys(token) : []
|
|
73
|
+
});
|
|
74
|
+
return NextResponse.json(
|
|
75
|
+
{ success: false, message: 'No session found' },
|
|
76
|
+
{ status: 401 }
|
|
77
|
+
);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
console.info('[VERIFY-CODE] Updating session with new tokens after 2FA', {
|
|
81
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
82
|
+
userId: token?.sub,
|
|
83
|
+
hasAccessToken: !!accessToken,
|
|
84
|
+
hasRefreshToken: !!refreshToken,
|
|
85
|
+
accessTokenLength: accessToken?.length,
|
|
86
|
+
refreshTokenLength: refreshToken?.length
|
|
87
|
+
});
|
|
88
|
+
|
|
89
|
+
// Update tokens in Redis
|
|
90
|
+
await updateTokens(
|
|
91
|
+
sessionToken,
|
|
92
|
+
accessToken,
|
|
93
|
+
refreshToken,
|
|
94
|
+
accessTokenExpires,
|
|
95
|
+
refreshTokenExpires
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
// Mark 2FA as complete
|
|
99
|
+
await mark2FAComplete(sessionToken);
|
|
100
|
+
|
|
101
|
+
console.info('[VERIFY-CODE] 2FA completion successful', {
|
|
102
|
+
sessionToken: sessionToken.substring(0, 8) + '...',
|
|
103
|
+
userId: token?.sub
|
|
104
|
+
});
|
|
105
|
+
|
|
106
|
+
return NextResponse.json({
|
|
107
|
+
success: true,
|
|
108
|
+
message: '2FA verification complete'
|
|
109
|
+
});
|
|
110
|
+
} catch (error) {
|
|
111
|
+
console.error('[VERIFY-CODE] Failed to complete 2FA', {
|
|
112
|
+
error: error instanceof Error ? error.message : String(error)
|
|
113
|
+
});
|
|
114
|
+
|
|
115
|
+
return NextResponse.json(
|
|
116
|
+
{ success: false, message: 'Failed to complete 2FA' },
|
|
117
|
+
{ status: 500 }
|
|
118
|
+
);
|
|
119
|
+
}
|
|
120
|
+
};
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
/**
|
|
124
|
+
* Default export for backward compatibility
|
|
125
|
+
* Requires environment variable: NEXTAUTH_SECRET
|
|
126
|
+
*/
|
|
127
|
+
export const POST = createVerifyCodeHandler({
|
|
128
|
+
nextAuthSecret: process.env.NEXTAUTH_SECRET || ''
|
|
129
|
+
});
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { NextRequest, NextResponse } from 'next/server';
|
|
2
|
+
import { getSession, isRefreshInProgress } from '../../lib/session-store';
|
|
3
|
+
import { checkRefreshViability } from '../../lib/refresh-token-validator';
|
|
4
|
+
import { getTokenTestAware } from '../../lib/test-aware-get-token';
|
|
5
|
+
import { logger } from '../../config/logger';
|
|
6
|
+
|
|
7
|
+
export async function GET(req: NextRequest) {
|
|
8
|
+
try {
|
|
9
|
+
const sessionToken = req.nextUrl.searchParams.get('token');
|
|
10
|
+
let finalSessionToken = sessionToken;
|
|
11
|
+
if (!finalSessionToken) {
|
|
12
|
+
const token = await getTokenTestAware(req);
|
|
13
|
+
// Support both field names: sessionToken (auth.ts JWT) and redisSessionId (legacy)
|
|
14
|
+
finalSessionToken = (token?.sessionToken || token?.redisSessionId) as string;
|
|
15
|
+
}
|
|
16
|
+
if (!finalSessionToken) {
|
|
17
|
+
return NextResponse.json({ canRefresh: false, reason: 'not_logged_in', sessionToken: null }, { status: 200 });
|
|
18
|
+
}
|
|
19
|
+
const refreshInProgress = await isRefreshInProgress(finalSessionToken);
|
|
20
|
+
if (refreshInProgress) {
|
|
21
|
+
// Still need to get session data for twoFactorComplete even when refresh is in progress
|
|
22
|
+
const sessionData = await getSession(finalSessionToken);
|
|
23
|
+
logger.info('[REFRESH-VIABILITY] Refresh already in progress, telling middleware to wait', { sessionToken: finalSessionToken.substring(0, 8) + '...' });
|
|
24
|
+
return NextResponse.json({ canRefresh: true, reason: 'refresh_in_progress', refreshInProgress: true, sessionToken: finalSessionToken, twoFactorComplete: sessionData?.mfaVerified ?? (sessionData as any)?.twoFactorComplete ?? false }, { status: 200 });
|
|
25
|
+
}
|
|
26
|
+
const sessionData = await getSession(finalSessionToken);
|
|
27
|
+
if (!sessionData) {
|
|
28
|
+
return NextResponse.json({ canRefresh: false, reason: 'session_not_found', sessionToken: finalSessionToken }, { status: 200 });
|
|
29
|
+
}
|
|
30
|
+
const viabilityCheck = checkRefreshViability(sessionData);
|
|
31
|
+
return NextResponse.json({ canRefresh: viabilityCheck.canRefresh, reason: viabilityCheck.reason, timeRemaining: viabilityCheck.timeRemaining, expiresAt: viabilityCheck.expiresAt, accessTokenExpired: viabilityCheck.accessTokenExpired, accessTokenTimeRemaining: viabilityCheck.accessTokenTimeRemaining, sessionToken: finalSessionToken, twoFactorComplete: sessionData.mfaVerified ?? (sessionData as any).twoFactorComplete ?? false, userId: sessionData.userId, refreshInProgress: false });
|
|
32
|
+
} catch (error) {
|
|
33
|
+
logger.error('[REFRESH-VIABILITY] Error checking refresh viability', { error: error instanceof Error ? error.message : String(error) });
|
|
34
|
+
return NextResponse.json({ canRefresh: false, reason: 'check_error', error: error instanceof Error ? error.message : String(error) }, { status: 500 });
|
|
35
|
+
}
|
|
36
|
+
}
|