@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,171 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Centralized internal API helper for the app to call ITSELF.
|
|
3
|
+
*
|
|
4
|
+
* IMPORTANT: All calls from the Next.js server to its own API routes MUST use
|
|
5
|
+
* these functions. Never use req.url, req.nextUrl.origin, or construct URLs
|
|
6
|
+
* from the incoming request.
|
|
7
|
+
*
|
|
8
|
+
* WHY HTTP IS REQUIRED (not optional):
|
|
9
|
+
* - This is the app calling its OWN backend within the same pod/container
|
|
10
|
+
* - NextAuth cookies are encrypted based on request protocol
|
|
11
|
+
* - TLS is terminated at ingress, so the pod receives HTTP internally
|
|
12
|
+
* - Using HTTPS here causes cookie decryption failures and 403 errors
|
|
13
|
+
* - This is NOT about "K8s traffic doesn't need TLS" - it's about
|
|
14
|
+
* protocol consistency for cookie/session encryption
|
|
15
|
+
*
|
|
16
|
+
* Environment:
|
|
17
|
+
* - INTERNAL_API_URL: Required in production (e.g., http://service.namespace.svc.cluster.local:80)
|
|
18
|
+
* - Falls back to http://localhost:3200 in development only
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Get the internal API base URL for the app to call itself.
|
|
23
|
+
*
|
|
24
|
+
* @throws Error in production if INTERNAL_API_URL is not set
|
|
25
|
+
* @returns The base URL (no trailing slash)
|
|
26
|
+
*/
|
|
27
|
+
export function getInternalApiUrl(): string {
|
|
28
|
+
const url = process.env.INTERNAL_API_URL;
|
|
29
|
+
if (url) return url.replace(/\/$/, ''); // strip trailing slash
|
|
30
|
+
|
|
31
|
+
if (process.env.NODE_ENV !== 'production') {
|
|
32
|
+
return 'http://localhost:3200';
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
throw new Error(
|
|
36
|
+
'[INTERNAL_API_URL] FATAL: INTERNAL_API_URL environment variable is REQUIRED in production. ' +
|
|
37
|
+
'This is for the app to call ITSELF. MUST be HTTP (not HTTPS) due to cookie encryption. ' +
|
|
38
|
+
'Set to http://service.namespace.svc.cluster.local:80'
|
|
39
|
+
);
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
/**
|
|
43
|
+
* Options for internal fetch calls
|
|
44
|
+
*/
|
|
45
|
+
export interface InternalFetchOptions extends Omit<RequestInit, 'headers'> {
|
|
46
|
+
/** Additional headers to include */
|
|
47
|
+
headers?: Record<string, string>;
|
|
48
|
+
/** Cookie header to forward (typically from req.headers.get('cookie')) */
|
|
49
|
+
cookie?: string;
|
|
50
|
+
/** Session token for x-session-token header */
|
|
51
|
+
sessionToken?: string;
|
|
52
|
+
/** Request ID for tracing */
|
|
53
|
+
requestId?: string;
|
|
54
|
+
/** Parse response as JSON (default: true) */
|
|
55
|
+
parseJson?: boolean;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
/**
|
|
59
|
+
* Result of an internal fetch call
|
|
60
|
+
*/
|
|
61
|
+
export interface InternalFetchResult<T = unknown> {
|
|
62
|
+
ok: boolean;
|
|
63
|
+
status: number;
|
|
64
|
+
statusText: string;
|
|
65
|
+
data: T | null;
|
|
66
|
+
response: Response;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
/**
|
|
70
|
+
* Make a fetch call to an internal API route (app calling itself).
|
|
71
|
+
*
|
|
72
|
+
* @param path - The API path (e.g., '/api/auth/refresh')
|
|
73
|
+
* @param options - Fetch options
|
|
74
|
+
* @returns The fetch result with parsed data
|
|
75
|
+
*
|
|
76
|
+
* @example
|
|
77
|
+
* ```ts
|
|
78
|
+
* // Simple GET
|
|
79
|
+
* const result = await internalFetch('/api/health');
|
|
80
|
+
*
|
|
81
|
+
* // POST with session
|
|
82
|
+
* const result = await internalFetch('/api/auth/refresh', {
|
|
83
|
+
* method: 'POST',
|
|
84
|
+
* cookie: req.headers.get('cookie') || '',
|
|
85
|
+
* sessionToken: token.redisSessionId,
|
|
86
|
+
* body: JSON.stringify({ refresh_token: refreshToken }),
|
|
87
|
+
* });
|
|
88
|
+
* ```
|
|
89
|
+
*/
|
|
90
|
+
export async function internalFetch<T = unknown>(
|
|
91
|
+
path: string,
|
|
92
|
+
options: InternalFetchOptions = {}
|
|
93
|
+
): Promise<InternalFetchResult<T>> {
|
|
94
|
+
const {
|
|
95
|
+
headers: extraHeaders = {},
|
|
96
|
+
cookie,
|
|
97
|
+
sessionToken,
|
|
98
|
+
requestId,
|
|
99
|
+
parseJson = true,
|
|
100
|
+
...fetchOptions
|
|
101
|
+
} = options;
|
|
102
|
+
|
|
103
|
+
const baseUrl = getInternalApiUrl();
|
|
104
|
+
const url = `${baseUrl}${path.startsWith('/') ? path : `/${path}`}`;
|
|
105
|
+
|
|
106
|
+
// Build headers
|
|
107
|
+
const headers: Record<string, string> = {
|
|
108
|
+
'Accept': 'application/json',
|
|
109
|
+
'Content-Type': 'application/json',
|
|
110
|
+
...extraHeaders,
|
|
111
|
+
};
|
|
112
|
+
|
|
113
|
+
if (cookie) {
|
|
114
|
+
headers['Cookie'] = cookie;
|
|
115
|
+
}
|
|
116
|
+
if (sessionToken) {
|
|
117
|
+
headers['X-Session-Token'] = sessionToken;
|
|
118
|
+
}
|
|
119
|
+
if (requestId) {
|
|
120
|
+
headers['X-Request-Id'] = requestId;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
const response = await fetch(url, {
|
|
124
|
+
...fetchOptions,
|
|
125
|
+
headers,
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
let data: T | null = null;
|
|
129
|
+
if (parseJson) {
|
|
130
|
+
try {
|
|
131
|
+
data = await response.json() as T;
|
|
132
|
+
} catch {
|
|
133
|
+
data = null;
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
return {
|
|
138
|
+
ok: response.ok,
|
|
139
|
+
status: response.status,
|
|
140
|
+
statusText: response.statusText,
|
|
141
|
+
data,
|
|
142
|
+
response,
|
|
143
|
+
};
|
|
144
|
+
}
|
|
145
|
+
|
|
146
|
+
/**
|
|
147
|
+
* Trigger a token refresh via the internal API.
|
|
148
|
+
* This is a convenience wrapper for the common refresh pattern.
|
|
149
|
+
*
|
|
150
|
+
* @param cookie - The cookie header from the incoming request
|
|
151
|
+
* @param sessionToken - The session token
|
|
152
|
+
* @param refreshToken - Optional refresh token to include in body
|
|
153
|
+
* @param requestId - Optional request ID for tracing
|
|
154
|
+
* @returns Whether the refresh was successful
|
|
155
|
+
*/
|
|
156
|
+
export async function internalRefresh(
|
|
157
|
+
cookie: string,
|
|
158
|
+
sessionToken: string,
|
|
159
|
+
refreshToken?: string,
|
|
160
|
+
requestId?: string
|
|
161
|
+
): Promise<{ ok: boolean; status: number }> {
|
|
162
|
+
const result = await internalFetch('/api/auth/refresh', {
|
|
163
|
+
method: 'POST',
|
|
164
|
+
cookie,
|
|
165
|
+
sessionToken,
|
|
166
|
+
requestId,
|
|
167
|
+
body: refreshToken ? JSON.stringify({ refresh_token: refreshToken }) : undefined,
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
return { ok: result.ok, status: result.status };
|
|
171
|
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Client-safe JWT decode (no Node.js dependencies)
|
|
3
|
+
* This is a lightweight version for browser usage
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
// Decode base64url
|
|
7
|
+
function base64urlDecode(base64url: string): string {
|
|
8
|
+
try {
|
|
9
|
+
// Convert base64url to base64
|
|
10
|
+
let base64 = base64url.replace(/-/g, '+').replace(/_/g, '/');
|
|
11
|
+
// Add padding if needed
|
|
12
|
+
while (base64.length % 4) {
|
|
13
|
+
base64 += '=';
|
|
14
|
+
}
|
|
15
|
+
return atob(base64);
|
|
16
|
+
} catch (e) {
|
|
17
|
+
throw new Error('Invalid base64url encoding');
|
|
18
|
+
}
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* Simple JWT decode for client-side use (no signature verification)
|
|
23
|
+
* @param token - JWT token string
|
|
24
|
+
* @returns Decoded payload or null if invalid
|
|
25
|
+
*/
|
|
26
|
+
export function jwtDecode<T = any>(token: string): T | null {
|
|
27
|
+
if (!token) return null;
|
|
28
|
+
|
|
29
|
+
try {
|
|
30
|
+
const parts = token.split('.');
|
|
31
|
+
if (parts.length < 2) {
|
|
32
|
+
console.error('[JWT] Invalid token format');
|
|
33
|
+
return null;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
const payload = parts[1];
|
|
37
|
+
const decoded = base64urlDecode(payload);
|
|
38
|
+
const parsedPayload = JSON.parse(decoded);
|
|
39
|
+
|
|
40
|
+
return parsedPayload;
|
|
41
|
+
} catch (e) {
|
|
42
|
+
console.error('[JWT] Decode failed:', e instanceof Error ? e.message : 'Unknown error');
|
|
43
|
+
return null;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { jwtDecode as originalJwtDecode } from 'jwt-decode';
|
|
2
|
+
|
|
3
|
+
// Define JwtPayload interface since jwt-decode v4 doesn't export it
|
|
4
|
+
export interface JwtPayload {
|
|
5
|
+
iss?: string;
|
|
6
|
+
sub?: string;
|
|
7
|
+
aud?: string[] | string;
|
|
8
|
+
exp?: number;
|
|
9
|
+
nbf?: number;
|
|
10
|
+
iat?: number;
|
|
11
|
+
jti?: string;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
/**
|
|
15
|
+
* JWT Header structure.
|
|
16
|
+
* Contains metadata about the token including the signing key ID.
|
|
17
|
+
*/
|
|
18
|
+
export interface JwtHeader {
|
|
19
|
+
/** Algorithm used to sign the token (e.g., 'RS256', 'HS256') */
|
|
20
|
+
alg: string;
|
|
21
|
+
/** Token type (typically 'JWT') */
|
|
22
|
+
typ?: string;
|
|
23
|
+
/** Key ID - identifies which key was used to sign this token */
|
|
24
|
+
kid?: string;
|
|
25
|
+
/** Content type */
|
|
26
|
+
cty?: string;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Decode JWT payload (standard claims).
|
|
31
|
+
* This is a thin wrapper around jwt-decode library.
|
|
32
|
+
*/
|
|
33
|
+
export function jwtDecode<T = JwtPayload>(token: string): T {
|
|
34
|
+
return originalJwtDecode<T>(token);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Decode JWT header to extract kid, alg, and other header claims.
|
|
39
|
+
*
|
|
40
|
+
* The JWT header contains critical information:
|
|
41
|
+
* - kid: Key ID used to sign the token (needed for key governance)
|
|
42
|
+
* - alg: Algorithm used for signing
|
|
43
|
+
* - typ: Token type
|
|
44
|
+
*
|
|
45
|
+
* @param token - The JWT token string
|
|
46
|
+
* @returns Decoded header or null if decoding fails
|
|
47
|
+
*/
|
|
48
|
+
export function decodeJwtHeader(token: string): JwtHeader | null {
|
|
49
|
+
try {
|
|
50
|
+
if (!token || typeof token !== 'string') {
|
|
51
|
+
return null;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const parts = token.split('.');
|
|
55
|
+
if (parts.length !== 3) {
|
|
56
|
+
console.warn('[JWT_DECODE] Invalid JWT structure - expected 3 parts, got', parts.length);
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Decode base64url header (part 0)
|
|
61
|
+
const headerB64 = parts[0].replace(/-/g, '+').replace(/_/g, '/');
|
|
62
|
+
const headerJson = typeof atob !== 'undefined'
|
|
63
|
+
? atob(headerB64)
|
|
64
|
+
: Buffer.from(headerB64, 'base64').toString('utf-8');
|
|
65
|
+
|
|
66
|
+
return JSON.parse(headerJson) as JwtHeader;
|
|
67
|
+
} catch (error) {
|
|
68
|
+
console.error('[JWT_DECODE] Failed to decode JWT header:', error);
|
|
69
|
+
return null;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
/**
|
|
74
|
+
* Extract just the kid (Key ID) from a JWT token.
|
|
75
|
+
* Convenience function for when you only need the key ID.
|
|
76
|
+
*
|
|
77
|
+
* @param token - The JWT token string
|
|
78
|
+
* @returns The kid value or undefined if not present/decodable
|
|
79
|
+
*/
|
|
80
|
+
export function extractKidFromToken(token: string): string | undefined {
|
|
81
|
+
const header = decodeJwtHeader(token);
|
|
82
|
+
return header?.kid;
|
|
83
|
+
}
|
|
@@ -0,0 +1,126 @@
|
|
|
1
|
+
import 'server-only';
|
|
2
|
+
import { validateNextAuthSecret } from './secret-validation';
|
|
3
|
+
import { randomUUID } from 'crypto';
|
|
4
|
+
|
|
5
|
+
let cachedSecret: string | null = null;
|
|
6
|
+
let lastFetchedAt = 0;
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Resolve the NextAuth secret (server-only).
|
|
10
|
+
*
|
|
11
|
+
* Priority:
|
|
12
|
+
* 1) Use process.env.NEXTAUTH_SECRET if present (allows overrides/production)
|
|
13
|
+
* 2) Fetch from IDP broker endpoint - IDP handles all Key Vault/signing
|
|
14
|
+
* 3) Cache result in-memory and set process.env.NEXTAUTH_SECRET for subsequent calls
|
|
15
|
+
*/
|
|
16
|
+
export async function resolveNextAuthSecret(): Promise<string> {
|
|
17
|
+
// Check if already in environment
|
|
18
|
+
if (process.env.NEXTAUTH_SECRET && process.env.NEXTAUTH_SECRET.trim() !== '') {
|
|
19
|
+
// Silent - already configured
|
|
20
|
+
return process.env.NEXTAUTH_SECRET;
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// Check if cached and fresh (within 5 minutes)
|
|
24
|
+
if (cachedSecret && Date.now() - lastFetchedAt < 5 * 60 * 1000) {
|
|
25
|
+
return cachedSecret;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// Broker mode: fetch from IDP (IDP handles all Key Vault/signing)
|
|
29
|
+
const base = process.env.IDP_URL;
|
|
30
|
+
if (!base) throw new Error('IDP_URL environment variable is required');
|
|
31
|
+
|
|
32
|
+
const clientIdStr = process.env.CLIENT_ID;
|
|
33
|
+
if (!clientIdStr || clientIdStr.trim() === '') throw new Error('CLIENT_ID is required (e.g., "ideal_resume_website")');
|
|
34
|
+
|
|
35
|
+
// Determine if clientId is numeric or string
|
|
36
|
+
const isNumeric = /^[0-9]+$/.test(clientIdStr);
|
|
37
|
+
const clientId = isNumeric ? parseInt(clientIdStr, 10) : clientIdStr;
|
|
38
|
+
|
|
39
|
+
// Step 1: Request IDP to sign a client assertion (IDP has the keys, not us)
|
|
40
|
+
|
|
41
|
+
const signingUrl = new URL(`${base.replace(/\/$/, '')}/api/ExternalAuth/sign-client-assertion`);
|
|
42
|
+
// Client ID passed via X-Client-Id header, not query string
|
|
43
|
+
|
|
44
|
+
const signingPayload = {
|
|
45
|
+
issuer: clientIdStr,
|
|
46
|
+
subject: clientIdStr,
|
|
47
|
+
audience: 'urn:payez:externalauth:nextauthsecret',
|
|
48
|
+
expires_in: 60
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
const signingResp = await fetch(signingUrl.toString(), {
|
|
52
|
+
method: 'POST',
|
|
53
|
+
headers: {
|
|
54
|
+
'Accept': 'application/json',
|
|
55
|
+
'Content-Type': 'application/json',
|
|
56
|
+
'X-Client-Id': clientIdStr,
|
|
57
|
+
'X-Correlation-Id': randomUUID().replace(/-/g, ''),
|
|
58
|
+
},
|
|
59
|
+
body: JSON.stringify(signingPayload),
|
|
60
|
+
cache: 'no-store'
|
|
61
|
+
} as RequestInit);
|
|
62
|
+
|
|
63
|
+
if (!signingResp.ok) {
|
|
64
|
+
const txt = await signingResp.text().catch(() => 'Unknown error');
|
|
65
|
+
throw new Error(`Failed to sign client assertion: ${signingResp.status} ${signingResp.statusText} - ${txt}`);
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
const signingBody: any = await signingResp.json().catch(() => ({}));
|
|
69
|
+
const client_assertion = (
|
|
70
|
+
signingBody?.data?.client_assertion ??
|
|
71
|
+
signingBody?.data?.clientAssertion ??
|
|
72
|
+
signingBody?.client_assertion ??
|
|
73
|
+
signingBody?.clientAssertion ??
|
|
74
|
+
signingBody?.data?.ClientAssertion ??
|
|
75
|
+
signingBody?.ClientAssertion
|
|
76
|
+
) as string | undefined;
|
|
77
|
+
|
|
78
|
+
if (!client_assertion || typeof client_assertion !== 'string') {
|
|
79
|
+
throw new Error('IDP did not return a valid signed client assertion');
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Step 2: Use the signed assertion to fetch the NextAuth secret
|
|
83
|
+
|
|
84
|
+
const proxyUrl = new URL(`${base.replace(/\/$/, '')}/api/ExternalAuth/next-auth/secret`);
|
|
85
|
+
|
|
86
|
+
const proxyResp = await fetch(proxyUrl.toString(), {
|
|
87
|
+
method: 'POST',
|
|
88
|
+
headers: {
|
|
89
|
+
'Accept': 'application/json',
|
|
90
|
+
'Content-Type': 'application/json',
|
|
91
|
+
'X-Client-Id': clientIdStr,
|
|
92
|
+
'X-Correlation-Id': randomUUID().replace(/-/g, ''),
|
|
93
|
+
},
|
|
94
|
+
body: JSON.stringify({ client_assertion }),
|
|
95
|
+
cache: 'no-store'
|
|
96
|
+
} as RequestInit);
|
|
97
|
+
|
|
98
|
+
if (!proxyResp.ok) {
|
|
99
|
+
const txt = await proxyResp.text().catch(() => 'Unknown error');
|
|
100
|
+
throw new Error(`Proxy error: ${proxyResp.status} ${proxyResp.statusText} - ${txt}`);
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
const proxyBody: any = await proxyResp.json().catch(() => ({}));
|
|
104
|
+
|
|
105
|
+
const secret = (proxyBody?.data?.secret ?? proxyBody?.secret) as string | undefined;
|
|
106
|
+
const configuration = (proxyBody?.data?.configuration ?? proxyBody?.configuration) as any | undefined;
|
|
107
|
+
|
|
108
|
+
// Configuration is available but we don't log it verbosely
|
|
109
|
+
|
|
110
|
+
if (!secret || typeof secret !== 'string') {
|
|
111
|
+
throw new Error('Proxy did not return a valid NextAuth secret');
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
const validation = validateNextAuthSecret(secret);
|
|
115
|
+
if (!validation.valid) {
|
|
116
|
+
throw new Error(`Fetched NextAuth secret failed validation: ${validation.reason}`);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
cachedSecret = secret;
|
|
120
|
+
lastFetchedAt = Date.now();
|
|
121
|
+
process.env.NEXTAUTH_SECRET = secret;
|
|
122
|
+
|
|
123
|
+
console.log('[NEXTAUTH-SECRET] Resolved from IDP (length:', secret.length + ')');
|
|
124
|
+
|
|
125
|
+
return secret;
|
|
126
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import redis from '../lib/redis';
|
|
2
|
+
import { logger } from '../config/logger';
|
|
3
|
+
|
|
4
|
+
export interface RateLimitRule { endpoint: string; period: string; limit: number; }
|
|
5
|
+
export interface RateLimitResult { isAllowed: boolean; requestCount: number; limit: number; retryAfterSeconds?: number; failedAttempts?: number; }
|
|
6
|
+
|
|
7
|
+
export function createPayEzRateLimitResponse(retryAfterSeconds: number, remainingAttempts: number = 0) {
|
|
8
|
+
return { success: false, message: 'Too many failed attempts', user_info: null, errors: [ { code: 'RateLimitExceeded', message: 'Too many failed authentication attempts', resolution: `Please try again in ${retryAfterSeconds} seconds`, remainingAttempts } ] };
|
|
9
|
+
}
|
package/src/lib/redis.ts
ADDED
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
// E:\Repos\PayEz-Next-MVP\packages\next-mvp\src\lib\redis.ts
|
|
2
|
+
import Redis, { RedisOptions } from 'ioredis';
|
|
3
|
+
|
|
4
|
+
let client: Redis | null = null;
|
|
5
|
+
|
|
6
|
+
function createClient(): Redis {
|
|
7
|
+
const url = process.env.REDIS_URL;
|
|
8
|
+
|
|
9
|
+
if (url && url.trim() !== '') {
|
|
10
|
+
// Use a standard configuration for better Docker compatibility
|
|
11
|
+
return new Redis(url);
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
// No REDIS_URL set, create a client that will fail fast.
|
|
15
|
+
return new Redis({ lazyConnect: true });
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export function getRedis(): Redis {
|
|
19
|
+
if (!client) {
|
|
20
|
+
client = createClient();
|
|
21
|
+
}
|
|
22
|
+
return client;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
const redis = getRedis();
|
|
26
|
+
export { redis };
|
|
27
|
+
export default redis;
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { jwtDecode } from './jwt-decode';
|
|
2
|
+
import { tokenRefreshLogger } from '../config/logger';
|
|
3
|
+
|
|
4
|
+
interface RefreshTokenPayload { exp: number; iat: number; jti: string; user_id: string; token_type: string; binding: string; }
|
|
5
|
+
|
|
6
|
+
export function isRefreshTokenValid(token: string): boolean {
|
|
7
|
+
if (!token) return false;
|
|
8
|
+
try { const decoded = jwtDecode<RefreshTokenPayload>(token); if (!decoded) return false; const now = Math.floor(Date.now() / 1000); if (decoded.exp < now) return false; if (decoded.token_type !== 'refresh_token') return false; return true; } catch { return false; }
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
export function isRefreshTokenExpiring(token: string, bufferMinutes: number = 60): boolean {
|
|
12
|
+
if (!token) return true;
|
|
13
|
+
try { const decoded = jwtDecode<RefreshTokenPayload>(token); if (!decoded?.exp) return true; const now = Math.floor(Date.now() / 1000); const buffer = bufferMinutes * 60; return decoded.exp <= (now + buffer); } catch { return true; }
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
export function getRefreshTokenExpiration(token: string): number | null {
|
|
17
|
+
if (!token) return null; try { const decoded = jwtDecode<RefreshTokenPayload>(token); if (!decoded?.exp) return null; return decoded.exp * 1000; } catch { return null; }
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
export function getRefreshTokenTimeRemaining(token: string): number | null {
|
|
21
|
+
if (!token) return null; try { const decoded = jwtDecode<RefreshTokenPayload>(token); if (!decoded?.exp) return null; const now = Math.floor(Date.now() / 1000); const timeRemaining = decoded.exp - now; return timeRemaining > 0 ? timeRemaining : null; } catch { return null; }
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
export interface RefreshViabilityCheck {
|
|
25
|
+
canRefresh: boolean;
|
|
26
|
+
reason: 'valid_refresh_token' | 'no_refresh_token' | 'refresh_token_expired' | 'session_missing';
|
|
27
|
+
timeRemaining?: number;
|
|
28
|
+
expiresAt?: string;
|
|
29
|
+
accessTokenExpired?: boolean;
|
|
30
|
+
accessTokenTimeRemaining?: number;
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export function checkRefreshViability(sessionData: any): RefreshViabilityCheck {
|
|
34
|
+
if (!sessionData) return { canRefresh: false, reason: 'session_missing' };
|
|
35
|
+
let accessTokenExpired = false; let accessTokenTimeRemaining: number | undefined;
|
|
36
|
+
if (sessionData.idpAccessTokenExpires) {
|
|
37
|
+
const now = Date.now(); let expiresAtMs = sessionData.idpAccessTokenExpires;
|
|
38
|
+
if (typeof expiresAtMs === 'string') expiresAtMs = parseInt(expiresAtMs, 10);
|
|
39
|
+
if (expiresAtMs < 1000000000000) expiresAtMs = expiresAtMs * 1000;
|
|
40
|
+
accessTokenTimeRemaining = Math.floor((expiresAtMs - now) / 1000);
|
|
41
|
+
const bufferSec = 5 * 60; // 5 minutes pre-expiry buffer
|
|
42
|
+
accessTokenExpired = accessTokenTimeRemaining <= bufferSec;
|
|
43
|
+
tokenRefreshLogger.debug('[REFRESH_VIABILITY] Access token expiration check', { now, expiresAtMs, accessTokenTimeRemaining, bufferSec, accessTokenExpired });
|
|
44
|
+
}
|
|
45
|
+
if (!sessionData.idpRefreshToken) return { canRefresh: false, reason: 'no_refresh_token', accessTokenExpired, accessTokenTimeRemaining };
|
|
46
|
+
if (sessionData.idpRefreshTokenExpires) {
|
|
47
|
+
let refreshExpMs = sessionData.idpRefreshTokenExpires;
|
|
48
|
+
if (typeof refreshExpMs === 'string') refreshExpMs = parseInt(refreshExpMs, 10);
|
|
49
|
+
if (refreshExpMs < 1000000000000) refreshExpMs = refreshExpMs * 1000;
|
|
50
|
+
const nowMs = Date.now(); const timeRemainingSec = Math.floor((refreshExpMs - nowMs) / 1000);
|
|
51
|
+
if (timeRemainingSec <= 0) return { canRefresh: false, reason: 'refresh_token_expired', accessTokenExpired, accessTokenTimeRemaining };
|
|
52
|
+
return { canRefresh: true, reason: 'valid_refresh_token', timeRemaining: timeRemainingSec, expiresAt: new Date(refreshExpMs).toISOString(), accessTokenExpired, accessTokenTimeRemaining };
|
|
53
|
+
}
|
|
54
|
+
try {
|
|
55
|
+
const decoded = jwtDecode<RefreshTokenPayload>(sessionData.idpRefreshToken); const nowSec = Math.floor(Date.now() / 1000);
|
|
56
|
+
if (!decoded?.exp || decoded.token_type !== 'refresh_token') return { canRefresh: false, reason: 'refresh_token_expired', accessTokenExpired, accessTokenTimeRemaining };
|
|
57
|
+
const timeRemaining = decoded.exp - nowSec; if (timeRemaining <= 0) return { canRefresh: false, reason: 'refresh_token_expired', accessTokenExpired, accessTokenTimeRemaining };
|
|
58
|
+
const expiresAtIso = new Date(decoded.exp * 1000).toISOString();
|
|
59
|
+
return { canRefresh: true, reason: 'valid_refresh_token', timeRemaining, expiresAt: expiresAtIso, accessTokenExpired, accessTokenTimeRemaining };
|
|
60
|
+
} catch (error) {
|
|
61
|
+
tokenRefreshLogger.debug('[REFRESH_VIABILITY] Failed to decode refresh token for viability', { error: error instanceof Error ? error.message : String(error) });
|
|
62
|
+
return { canRefresh: false, reason: 'refresh_token_expired', accessTokenExpired, accessTokenTimeRemaining };
|
|
63
|
+
}
|
|
64
|
+
}
|