@payez/next-mvp 4.0.1 → 4.0.2
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/dist/api/auth-handler.d.ts +66 -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 +110 -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 +19 -0
- package/dist/api-handlers/admin/analytics.js +378 -0
- package/dist/api-handlers/admin/audit.d.ts +19 -0
- package/dist/api-handlers/admin/audit.js +213 -0
- package/dist/api-handlers/admin/index.d.ts +21 -0
- package/dist/api-handlers/admin/index.js +42 -0
- package/dist/api-handlers/admin/redis-sessions.d.ts +35 -0
- package/dist/api-handlers/admin/redis-sessions.js +203 -0
- package/dist/api-handlers/admin/sessions.d.ts +20 -0
- package/dist/api-handlers/admin/sessions.js +283 -0
- package/dist/api-handlers/admin/site-logs.d.ts +45 -0
- package/dist/api-handlers/admin/site-logs.js +317 -0
- package/dist/api-handlers/admin/stats.d.ts +20 -0
- package/dist/api-handlers/admin/stats.js +239 -0
- package/dist/api-handlers/admin/users.d.ts +19 -0
- package/dist/api-handlers/admin/users.js +221 -0
- package/dist/api-handlers/admin/vibe-data.d.ts +79 -0
- package/dist/api-handlers/admin/vibe-data.js +267 -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 +633 -0
- package/dist/api-handlers/auth/signout.d.ts +37 -0
- package/dist/api-handlers/auth/signout.js +186 -0
- package/dist/api-handlers/auth/status.d.ts +8 -0
- package/dist/api-handlers/auth/status.js +23 -0
- package/dist/api-handlers/auth/update-session.d.ts +37 -0
- package/dist/api-handlers/auth/update-session.js +93 -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 +90 -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 +114 -0
- package/dist/api-handlers/test/force-expire.d.ts +23 -0
- package/dist/api-handlers/test/force-expire.js +59 -0
- package/dist/auth/auth-decision.d.ts +39 -0
- package/dist/auth/auth-decision.js +182 -0
- package/dist/auth/better-auth.d.ts +79 -0
- package/dist/auth/better-auth.js +119 -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 +384 -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 +83 -0
- package/dist/auth/utils/token-utils.js +218 -0
- package/dist/client/AuthContext.d.ts +19 -0
- package/dist/client/AuthContext.js +115 -0
- package/dist/client/better-auth-client.d.ts +1020 -0
- package/dist/client/better-auth-client.js +68 -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 +121 -0
- package/dist/components/SignalRHealthCheck.d.ts +10 -0
- package/dist/components/SignalRHealthCheck.js +97 -0
- package/dist/components/account/MobileNavDrawer.d.ts +32 -0
- package/dist/components/account/MobileNavDrawer.js +81 -0
- package/dist/components/account/UserAvatarMenu.d.ts +20 -0
- package/dist/components/account/UserAvatarMenu.js +91 -0
- package/dist/components/account/index.d.ts +9 -0
- package/dist/components/account/index.js +13 -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 +71 -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 +81 -0
- package/dist/config/vibe-log-transport.js +212 -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 +83 -0
- package/dist/hooks/useAuthSettings.d.ts +59 -0
- package/dist/hooks/useAuthSettings.js +93 -0
- package/dist/hooks/useAvailableProviders.d.ts +43 -0
- package/dist/hooks/useAvailableProviders.js +112 -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 +56 -0
- package/dist/hooks/useSessionExpiration.js +72 -0
- package/dist/hooks/useViabilitySession.d.ts +75 -0
- package/dist/hooks/useViabilitySession.js +269 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.js +53 -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 +425 -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/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 +170 -0
- package/dist/lib/session-store.js +545 -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 +791 -0
- package/dist/lib/startup-init.d.ts +40 -0
- package/dist/lib/startup-init.js +257 -0
- package/dist/lib/test-aware-get-token.d.ts +2 -0
- package/dist/lib/test-aware-get-token.js +86 -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 +78 -0
- package/dist/lib/token-lifecycle.js +360 -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 +102 -0
- package/dist/middleware/create-middleware.js +469 -0
- package/dist/middleware/rbac-check.d.ts +51 -0
- package/dist/middleware/rbac-check.js +219 -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 +73 -0
- package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.d.ts +18 -0
- package/dist/pages/admin-page-permissions/PagePermissionsAdminPage.js +276 -0
- package/dist/pages/admin-page-permissions/index.d.ts +6 -0
- package/dist/pages/admin-page-permissions/index.js +13 -0
- package/dist/pages/admin-roles/RolesAdminPage.d.ts +16 -0
- package/dist/pages/admin-roles/RolesAdminPage.js +261 -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 +179 -0
- package/dist/pages/client-admin/index.d.ts +32 -0
- package/dist/pages/client-admin/index.js +37 -0
- package/dist/pages/coming-soon/page.d.ts +8 -0
- package/dist/pages/coming-soon/page.js +28 -0
- package/dist/pages/login/page.d.ts +22 -0
- package/dist/pages/login/page.js +230 -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/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/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 +142 -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 +99 -0
- package/dist/pages/test-env/JwtInspectPage.d.ts +14 -0
- package/dist/pages/test-env/JwtInspectPage.js +116 -0
- package/dist/pages/test-env/RefreshTokenPage.d.ts +15 -0
- package/dist/pages/test-env/RefreshTokenPage.js +93 -0
- package/dist/pages/test-env/TestEnvPage.d.ts +13 -0
- package/dist/pages/test-env/TestEnvPage.js +51 -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 +412 -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 +98 -0
- package/dist/routes/auth/nextauth.d.ts +22 -0
- package/dist/routes/auth/nextauth.js +40 -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 +43 -0
- package/dist/routes/auth/session.js +157 -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 +190 -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/server/auth-guard.d.ts +46 -0
- package/dist/server/auth-guard.js +128 -0
- package/dist/server/auth.d.ts +50 -0
- package/dist/server/auth.js +62 -0
- package/dist/server/decode-session.d.ts +30 -0
- package/dist/server/decode-session.js +78 -0
- package/dist/server/slim-middleware.d.ts +23 -0
- package/dist/server/slim-middleware.js +89 -0
- package/dist/server/with-auth.d.ts +33 -0
- package/dist/server/with-auth.js +59 -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 +1527 -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/enterprise-auth.d.ts +106 -0
- package/dist/vibe/enterprise-auth.js +173 -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 +25 -0
- package/dist/vibe/index.js +72 -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 +1 -1
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Better Auth Client (Phase 3)
|
|
4
|
+
*
|
|
5
|
+
* Drop-in replacement for next-auth/react hooks and functions.
|
|
6
|
+
* Import from '@payez/next-mvp/client/better-auth-client'.
|
|
7
|
+
*
|
|
8
|
+
* Includes useSessionCompat() — returns NextAuth-shaped { data, status }
|
|
9
|
+
* so existing components don't need destructure pattern changes.
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.signOut = exports.signIn = exports.useSession = exports.authClient = void 0;
|
|
13
|
+
exports.useSessionCompat = useSessionCompat;
|
|
14
|
+
exports.signOutCompat = signOutCompat;
|
|
15
|
+
const react_1 = require("better-auth/react");
|
|
16
|
+
const react_2 = require("react");
|
|
17
|
+
exports.authClient = (0, react_1.createAuthClient)({
|
|
18
|
+
// baseURL derived from BETTER_AUTH_URL or window.location.origin
|
|
19
|
+
});
|
|
20
|
+
// Convenience exports
|
|
21
|
+
exports.useSession = exports.authClient.useSession, exports.signIn = exports.authClient.signIn, exports.signOut = exports.authClient.signOut;
|
|
22
|
+
/**
|
|
23
|
+
* NextAuth-compatible useSession wrapper.
|
|
24
|
+
*
|
|
25
|
+
* Maps Better Auth's { data, error, isPending } to NextAuth's { data, status, update }.
|
|
26
|
+
* Drop-in replacement — no destructure changes needed in consuming components.
|
|
27
|
+
*/
|
|
28
|
+
function useSessionCompat() {
|
|
29
|
+
const baSession = exports.authClient.useSession();
|
|
30
|
+
const status = (0, react_2.useMemo)(() => {
|
|
31
|
+
if (baSession.isPending)
|
|
32
|
+
return 'loading';
|
|
33
|
+
if (baSession.data)
|
|
34
|
+
return 'authenticated';
|
|
35
|
+
return 'unauthenticated';
|
|
36
|
+
}, [baSession.isPending, baSession.data]);
|
|
37
|
+
// Map Better Auth session shape to NextAuth session shape
|
|
38
|
+
const data = (0, react_2.useMemo)(() => {
|
|
39
|
+
if (!baSession.data)
|
|
40
|
+
return null;
|
|
41
|
+
return {
|
|
42
|
+
...baSession.data,
|
|
43
|
+
user: baSession.data.user,
|
|
44
|
+
expires: '', // Better Auth handles expiry differently
|
|
45
|
+
};
|
|
46
|
+
}, [baSession.data]);
|
|
47
|
+
return {
|
|
48
|
+
data,
|
|
49
|
+
status,
|
|
50
|
+
update: async () => {
|
|
51
|
+
// Better Auth doesn't have a direct "refresh session" call.
|
|
52
|
+
// Force refetch by invalidating the query.
|
|
53
|
+
// TODO: Wire to proper session refresh when available.
|
|
54
|
+
return data;
|
|
55
|
+
},
|
|
56
|
+
};
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* NextAuth-compatible signOut wrapper.
|
|
60
|
+
*
|
|
61
|
+
* Maps NextAuth signOut({ redirect, callbackUrl }) to Better Auth.
|
|
62
|
+
*/
|
|
63
|
+
async function signOutCompat(options) {
|
|
64
|
+
await exports.authClient.signOut();
|
|
65
|
+
if (options?.redirect !== false && options?.callbackUrl) {
|
|
66
|
+
window.location.href = options.callbackUrl;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* A wrapper for the `fetch` API that automatically injects the session's
|
|
3
|
+
* accessToken into the Authorization header and handles 401 Unauthorized
|
|
4
|
+
* responses by redirecting the user to the login page.
|
|
5
|
+
*
|
|
6
|
+
* @param url The URL to fetch.
|
|
7
|
+
* @param options The standard `fetch` options.
|
|
8
|
+
* @returns A `Promise` that resolves to the `Response` object.
|
|
9
|
+
* @throws An 'UNAUTHORIZED_REDIRECT' error after initiating the redirect to halt further execution.
|
|
10
|
+
*/
|
|
11
|
+
export declare function fetchWithAuth(url: string, options?: RequestInit): Promise<Response>;
|
|
@@ -0,0 +1,44 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.fetchWithAuth = fetchWithAuth;
|
|
4
|
+
// src/client/fetch-with-auth.ts
|
|
5
|
+
const better_auth_client_1 = require("./better-auth-client");
|
|
6
|
+
/**
|
|
7
|
+
* A wrapper for the `fetch` API that automatically injects the session's
|
|
8
|
+
* accessToken into the Authorization header and handles 401 Unauthorized
|
|
9
|
+
* responses by redirecting the user to the login page.
|
|
10
|
+
*
|
|
11
|
+
* @param url The URL to fetch.
|
|
12
|
+
* @param options The standard `fetch` options.
|
|
13
|
+
* @returns A `Promise` that resolves to the `Response` object.
|
|
14
|
+
* @throws An 'UNAUTHORIZED_REDIRECT' error after initiating the redirect to halt further execution.
|
|
15
|
+
*/
|
|
16
|
+
async function fetchWithAuth(url, options = {}) {
|
|
17
|
+
// 1. Retrieve the client-side session to get the accessToken.
|
|
18
|
+
const { data: session } = await better_auth_client_1.authClient.getSession();
|
|
19
|
+
// 2. Inject the accessToken into the Authorization header.
|
|
20
|
+
const headers = new Headers(options.headers);
|
|
21
|
+
if (session?.accessToken) {
|
|
22
|
+
headers.set('Authorization', `Bearer ${session.accessToken}`);
|
|
23
|
+
}
|
|
24
|
+
options.headers = headers;
|
|
25
|
+
const response = await fetch(url, options);
|
|
26
|
+
// 3. Handle the 401 response intelligently.
|
|
27
|
+
if (response.status === 401) {
|
|
28
|
+
// If we have a valid session, this is likely a claim/permission error, not an auth error
|
|
29
|
+
if (session?.accessToken) {
|
|
30
|
+
console.warn('API returned 401 despite valid session. Likely insufficient claims or permissions.');
|
|
31
|
+
// Don't redirect - let the calling code handle the error gracefully
|
|
32
|
+
return response;
|
|
33
|
+
}
|
|
34
|
+
// No valid session - this is a real authentication failure
|
|
35
|
+
console.error('Unauthorized API call (no valid session). Redirecting to login.');
|
|
36
|
+
// SAFEGUARD: Never use auth pages as callback URLs to prevent redirect loops
|
|
37
|
+
const pathname = window.location.pathname;
|
|
38
|
+
const safeCallbackUrl = pathname.startsWith('/account-auth/') ? '/' : pathname;
|
|
39
|
+
window.location.href = `/account-auth/login?callbackUrl=${encodeURIComponent(safeCallbackUrl)}`;
|
|
40
|
+
// Throw a specific error to signal that a redirect has been initiated.
|
|
41
|
+
throw new Error('UNAUTHORIZED_REDIRECT');
|
|
42
|
+
}
|
|
43
|
+
return response;
|
|
44
|
+
}
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.fetchWithSession = fetchWithSession;
|
|
4
|
+
async function fetchWithSession(input, init = {}, opts = {}) {
|
|
5
|
+
const retry = opts.retry ?? 1;
|
|
6
|
+
const doFetch = async () => {
|
|
7
|
+
const res = await fetch(input, { ...init, credentials: 'include' });
|
|
8
|
+
if (res.ok)
|
|
9
|
+
return res;
|
|
10
|
+
if (res.status === 401 && retry > 0) {
|
|
11
|
+
const rf = await fetch('/api/auth/refresh', { method: 'POST', headers: { 'Accept': 'application/json' }, credentials: 'include' });
|
|
12
|
+
if (rf.ok)
|
|
13
|
+
return fetchWithSession(input, init, { retry: retry - 1 });
|
|
14
|
+
}
|
|
15
|
+
if ((res.status === 409 || res.status === 503) && retry > 0) {
|
|
16
|
+
const retryAfter = res.headers.get('Retry-After');
|
|
17
|
+
const waitMs = retryAfter ? parseInt(retryAfter) * 1000 : 1000;
|
|
18
|
+
await new Promise(r => setTimeout(r, Math.min(waitMs, 3000)));
|
|
19
|
+
return fetchWithSession(input, init, { retry: retry - 1 });
|
|
20
|
+
}
|
|
21
|
+
return res;
|
|
22
|
+
};
|
|
23
|
+
return doFetch();
|
|
24
|
+
}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Client-Side Exports
|
|
3
|
+
*
|
|
4
|
+
* This module exports only client-safe code for use in browser environments.
|
|
5
|
+
* Server-only utilities and Node.js dependencies are excluded.
|
|
6
|
+
*/
|
|
7
|
+
export { fetchWithAuth } from './fetch-with-auth';
|
|
8
|
+
export { AuthProvider, useAuthConfig, useAuthMode, useFederatedProviders, useFederatedAuthEnabled, useTraditionalAuthEnabled } from './AuthContext';
|
|
9
|
+
export type { AuthConfig } from '../types/auth';
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Client-Side Exports
|
|
4
|
+
*
|
|
5
|
+
* This module exports only client-safe code for use in browser environments.
|
|
6
|
+
* Server-only utilities and Node.js dependencies are excluded.
|
|
7
|
+
*/
|
|
8
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
9
|
+
exports.useTraditionalAuthEnabled = exports.useFederatedAuthEnabled = exports.useFederatedProviders = exports.useAuthMode = exports.useAuthConfig = exports.AuthProvider = exports.fetchWithAuth = void 0;
|
|
10
|
+
// Client-side fetch utility
|
|
11
|
+
var fetch_with_auth_1 = require("./fetch-with-auth");
|
|
12
|
+
Object.defineProperty(exports, "fetchWithAuth", { enumerable: true, get: function () { return fetch_with_auth_1.fetchWithAuth; } });
|
|
13
|
+
// Authentication context and hooks
|
|
14
|
+
var AuthContext_1 = require("./AuthContext");
|
|
15
|
+
Object.defineProperty(exports, "AuthProvider", { enumerable: true, get: function () { return AuthContext_1.AuthProvider; } });
|
|
16
|
+
Object.defineProperty(exports, "useAuthConfig", { enumerable: true, get: function () { return AuthContext_1.useAuthConfig; } });
|
|
17
|
+
Object.defineProperty(exports, "useAuthMode", { enumerable: true, get: function () { return AuthContext_1.useAuthMode; } });
|
|
18
|
+
Object.defineProperty(exports, "useFederatedProviders", { enumerable: true, get: function () { return AuthContext_1.useFederatedProviders; } });
|
|
19
|
+
Object.defineProperty(exports, "useFederatedAuthEnabled", { enumerable: true, get: function () { return AuthContext_1.useFederatedAuthEnabled; } });
|
|
20
|
+
Object.defineProperty(exports, "useTraditionalAuthEnabled", { enumerable: true, get: function () { return AuthContext_1.useTraditionalAuthEnabled; } });
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* useAnonSession - React hook for anonymous session management
|
|
3
|
+
*
|
|
4
|
+
* Provides access to anonymous session preferences stored in Redis.
|
|
5
|
+
* Works before user logs in, preferences persist across visits.
|
|
6
|
+
*/
|
|
7
|
+
export interface AnonPreferences {
|
|
8
|
+
theme?: string;
|
|
9
|
+
locale?: string;
|
|
10
|
+
[key: string]: any;
|
|
11
|
+
}
|
|
12
|
+
export interface AnonMetrics {
|
|
13
|
+
resumeGenerationCount?: number;
|
|
14
|
+
firstVisit?: number;
|
|
15
|
+
lastVisit?: number;
|
|
16
|
+
visitCount?: number;
|
|
17
|
+
[key: string]: any;
|
|
18
|
+
}
|
|
19
|
+
export interface AnonSession {
|
|
20
|
+
id: string;
|
|
21
|
+
preferences: AnonPreferences;
|
|
22
|
+
metrics: AnonMetrics;
|
|
23
|
+
}
|
|
24
|
+
export interface UseAnonSessionReturn {
|
|
25
|
+
session: AnonSession | null;
|
|
26
|
+
isLoading: boolean;
|
|
27
|
+
error: string | null;
|
|
28
|
+
updatePreferences: (preferences: Partial<AnonPreferences>) => Promise<void>;
|
|
29
|
+
setTheme: (theme: string) => Promise<void>;
|
|
30
|
+
refresh: () => Promise<void>;
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Hook to manage anonymous session state
|
|
34
|
+
*/
|
|
35
|
+
export declare function useAnonSession(): UseAnonSessionReturn;
|
|
36
|
+
export default useAnonSession;
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* useAnonSession - React hook for anonymous session management
|
|
4
|
+
*
|
|
5
|
+
* Provides access to anonymous session preferences stored in Redis.
|
|
6
|
+
* Works before user logs in, preferences persist across visits.
|
|
7
|
+
*/
|
|
8
|
+
'use client';
|
|
9
|
+
/**
|
|
10
|
+
* useAnonSession - React hook for anonymous session management
|
|
11
|
+
*
|
|
12
|
+
* Provides access to anonymous session preferences stored in Redis.
|
|
13
|
+
* Works before user logs in, preferences persist across visits.
|
|
14
|
+
*/
|
|
15
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
16
|
+
exports.useAnonSession = useAnonSession;
|
|
17
|
+
const react_1 = require("react");
|
|
18
|
+
/**
|
|
19
|
+
* Hook to manage anonymous session state
|
|
20
|
+
*/
|
|
21
|
+
function useAnonSession() {
|
|
22
|
+
const [session, setSession] = (0, react_1.useState)(null);
|
|
23
|
+
const [isLoading, setIsLoading] = (0, react_1.useState)(true);
|
|
24
|
+
const [error, setError] = (0, react_1.useState)(null);
|
|
25
|
+
// Fetch session on mount
|
|
26
|
+
const fetchSession = (0, react_1.useCallback)(async () => {
|
|
27
|
+
try {
|
|
28
|
+
setIsLoading(true);
|
|
29
|
+
setError(null);
|
|
30
|
+
const response = await fetch('/api/anon/preferences', {
|
|
31
|
+
method: 'GET',
|
|
32
|
+
credentials: 'include', // Important for cookies
|
|
33
|
+
});
|
|
34
|
+
if (!response.ok) {
|
|
35
|
+
throw new Error('Failed to fetch preferences');
|
|
36
|
+
}
|
|
37
|
+
const data = await response.json();
|
|
38
|
+
if (data.success && data.data) {
|
|
39
|
+
setSession({
|
|
40
|
+
id: data.data.id,
|
|
41
|
+
preferences: data.data.preferences || {},
|
|
42
|
+
metrics: data.data.metrics || {},
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
catch (err) {
|
|
47
|
+
console.error('[useAnonSession] Error fetching session:', err);
|
|
48
|
+
setError(err instanceof Error ? err.message : 'Unknown error');
|
|
49
|
+
}
|
|
50
|
+
finally {
|
|
51
|
+
setIsLoading(false);
|
|
52
|
+
}
|
|
53
|
+
}, []);
|
|
54
|
+
(0, react_1.useEffect)(() => {
|
|
55
|
+
fetchSession();
|
|
56
|
+
}, [fetchSession]);
|
|
57
|
+
// Update preferences
|
|
58
|
+
const updatePreferences = (0, react_1.useCallback)(async (preferences) => {
|
|
59
|
+
try {
|
|
60
|
+
setError(null);
|
|
61
|
+
const response = await fetch('/api/anon/preferences', {
|
|
62
|
+
method: 'POST',
|
|
63
|
+
credentials: 'include',
|
|
64
|
+
headers: {
|
|
65
|
+
'Content-Type': 'application/json',
|
|
66
|
+
},
|
|
67
|
+
body: JSON.stringify({ preferences }),
|
|
68
|
+
});
|
|
69
|
+
if (!response.ok) {
|
|
70
|
+
throw new Error('Failed to update preferences');
|
|
71
|
+
}
|
|
72
|
+
const data = await response.json();
|
|
73
|
+
if (data.success && data.data) {
|
|
74
|
+
setSession(prev => prev ? {
|
|
75
|
+
...prev,
|
|
76
|
+
preferences: data.data.preferences,
|
|
77
|
+
} : null);
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
catch (err) {
|
|
81
|
+
console.error('[useAnonSession] Error updating preferences:', err);
|
|
82
|
+
setError(err instanceof Error ? err.message : 'Unknown error');
|
|
83
|
+
throw err;
|
|
84
|
+
}
|
|
85
|
+
}, []);
|
|
86
|
+
// Convenience method to set theme
|
|
87
|
+
const setTheme = (0, react_1.useCallback)(async (theme) => {
|
|
88
|
+
await updatePreferences({ theme });
|
|
89
|
+
}, [updatePreferences]);
|
|
90
|
+
return {
|
|
91
|
+
session,
|
|
92
|
+
isLoading,
|
|
93
|
+
error,
|
|
94
|
+
updatePreferences,
|
|
95
|
+
setTheme,
|
|
96
|
+
refresh: fetchSession,
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
exports.default = useAnonSession;
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SessionSync - Bridges NextAuth session with Zustand auth store
|
|
3
|
+
*
|
|
4
|
+
* CRITICAL: This component enforces strict session validation. If NextAuth
|
|
5
|
+
* reports an authenticated status but the session data is invalid (empty user ID,
|
|
6
|
+
* empty email, or missing access token), it forces a sign-out to prevent
|
|
7
|
+
* contradictory state like "hasSession: true, userId: ''"
|
|
8
|
+
*
|
|
9
|
+
* This ensures the app NEVER shows authenticated UI with empty/invalid session data.
|
|
10
|
+
*/
|
|
11
|
+
export declare function SessionSync({ children }: {
|
|
12
|
+
children: React.ReactNode;
|
|
13
|
+
}): import("react/jsx-runtime").JSX.Element;
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
'use client';
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.SessionSync = SessionSync;
|
|
5
|
+
const jsx_runtime_1 = require("react/jsx-runtime");
|
|
6
|
+
const react_1 = require("react");
|
|
7
|
+
const better_auth_client_1 = require("../client/better-auth-client");
|
|
8
|
+
const authStore_1 = require("../stores/authStore");
|
|
9
|
+
const session_1 = require("../lib/session");
|
|
10
|
+
const app_slug_1 = require("../lib/app-slug");
|
|
11
|
+
/**
|
|
12
|
+
* Sanitize sensitive data for logging
|
|
13
|
+
* Never log full user IDs or emails in any environment
|
|
14
|
+
*/
|
|
15
|
+
function sanitizeForLog(value, type) {
|
|
16
|
+
if (!value)
|
|
17
|
+
return '(empty)';
|
|
18
|
+
if (type === 'email') {
|
|
19
|
+
return '***@***'; // Never log emails
|
|
20
|
+
}
|
|
21
|
+
if (type === 'userId') {
|
|
22
|
+
// Only show first 8 chars in development
|
|
23
|
+
if (process.env.NODE_ENV === 'development') {
|
|
24
|
+
return value.substring(0, 8) + '...';
|
|
25
|
+
}
|
|
26
|
+
return '***'; // Fully redact in production
|
|
27
|
+
}
|
|
28
|
+
return '***';
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* SessionSync - Bridges NextAuth session with Zustand auth store
|
|
32
|
+
*
|
|
33
|
+
* CRITICAL: This component enforces strict session validation. If NextAuth
|
|
34
|
+
* reports an authenticated status but the session data is invalid (empty user ID,
|
|
35
|
+
* empty email, or missing access token), it forces a sign-out to prevent
|
|
36
|
+
* contradictory state like "hasSession: true, userId: ''"
|
|
37
|
+
*
|
|
38
|
+
* This ensures the app NEVER shows authenticated UI with empty/invalid session data.
|
|
39
|
+
*/
|
|
40
|
+
function SessionSync({ children }) {
|
|
41
|
+
const { data: sessionData, isPending } = better_auth_client_1.authClient.useSession();
|
|
42
|
+
const session = sessionData;
|
|
43
|
+
const status = isPending ? 'loading' : session ? 'authenticated' : 'unauthenticated';
|
|
44
|
+
const { setSession, clearSession } = (0, authStore_1.useAuthStore)();
|
|
45
|
+
// Guard against duplicate sign-out calls
|
|
46
|
+
const isSigningOutRef = (0, react_1.useRef)(false);
|
|
47
|
+
(0, react_1.useEffect)(() => {
|
|
48
|
+
let isMounted = true;
|
|
49
|
+
// Only process when NextAuth has finished loading
|
|
50
|
+
if (status === 'loading') {
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
// Strict validation: Check if session is actually valid
|
|
54
|
+
const isValid = (0, session_1.isValidSession)(session);
|
|
55
|
+
// CRITICAL FIX: If NextAuth says "authenticated" but session is invalid,
|
|
56
|
+
// this is a broken state that must be fixed immediately
|
|
57
|
+
if (status === 'authenticated' && !isValid) {
|
|
58
|
+
// GUARD: Prevent duplicate sign-out
|
|
59
|
+
if (isSigningOutRef.current) {
|
|
60
|
+
console.warn('[SessionSync] Sign-out already in progress, skipping');
|
|
61
|
+
return;
|
|
62
|
+
}
|
|
63
|
+
isSigningOutRef.current = true;
|
|
64
|
+
console.error('[SessionSync] CRITICAL: Invalid session detected despite authenticated status!');
|
|
65
|
+
console.error('[SessionSync] This indicates a session with empty user data - forcing sign-out');
|
|
66
|
+
// Log diagnostic info with PII redaction
|
|
67
|
+
// Note: session is typed as Session | null from NextAuth, but may have invalid/partial data
|
|
68
|
+
const sessionData = session; // Explicit cast needed for error logging
|
|
69
|
+
console.error('[SessionSync] Session data:', {
|
|
70
|
+
hasSessionObject: !!sessionData,
|
|
71
|
+
hasUser: !!sessionData?.user,
|
|
72
|
+
userId: sanitizeForLog(sessionData?.user?.id, 'userId'),
|
|
73
|
+
userEmail: sanitizeForLog(sessionData?.user?.email, 'email'),
|
|
74
|
+
hasAccessToken: !!sessionData?.accessToken,
|
|
75
|
+
});
|
|
76
|
+
// Clear the auth store immediately
|
|
77
|
+
clearSession();
|
|
78
|
+
// FIX: Force clear all auth cookies including stale provisional tokens (app-slug prefixed)
|
|
79
|
+
// This prevents infinite loop when provisional token exists but session is invalid
|
|
80
|
+
// Root cause: OAuth creates provisional token before Redis session exists
|
|
81
|
+
try {
|
|
82
|
+
const secureSession = (0, app_slug_1.getSecureSessionCookieName)();
|
|
83
|
+
const secureCsrf = (0, app_slug_1.getSecureCsrfCookieName)();
|
|
84
|
+
const callbackUrl = (0, app_slug_1.getCallbackUrlCookieName)();
|
|
85
|
+
document.cookie = `${secureSession}=; path=/; expires=Thu, 01 Jan 1970 00:00:00 UTC; Secure; SameSite=Lax`;
|
|
86
|
+
document.cookie = `${secureCsrf}=; path=/; expires=Thu, 01 Jan 1970 00:00:00 UTC; SameSite=Lax`;
|
|
87
|
+
document.cookie = `__Secure-${callbackUrl}=; path=/; expires=Thu, 01 Jan 1970 00:00:00 UTC; Secure; SameSite=Lax`;
|
|
88
|
+
}
|
|
89
|
+
catch (e) {
|
|
90
|
+
// Cookie clearing failed - non-critical, continue with signout
|
|
91
|
+
}
|
|
92
|
+
// Force Better Auth to sign out (this will clear cookies and trigger redirect)
|
|
93
|
+
better_auth_client_1.authClient.signOut()
|
|
94
|
+
.then(() => {
|
|
95
|
+
if (isMounted) {
|
|
96
|
+
// Use generic error code instead of implementation details
|
|
97
|
+
window.location.href = '/account-auth/login?error=SessionExpired&code=1001';
|
|
98
|
+
}
|
|
99
|
+
})
|
|
100
|
+
.catch((err) => {
|
|
101
|
+
console.error('[SessionSync] Error during forced signout:', err);
|
|
102
|
+
if (isMounted) {
|
|
103
|
+
window.location.href = '/account-auth/login?error=SessionExpired&code=1001';
|
|
104
|
+
}
|
|
105
|
+
});
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
// Normal flow: Update store based on valid session status
|
|
109
|
+
if (status === 'authenticated' && isValid) {
|
|
110
|
+
setSession(session);
|
|
111
|
+
}
|
|
112
|
+
else if (status === 'unauthenticated') {
|
|
113
|
+
clearSession();
|
|
114
|
+
}
|
|
115
|
+
// Cleanup function to prevent post-unmount updates
|
|
116
|
+
return () => {
|
|
117
|
+
isMounted = false;
|
|
118
|
+
};
|
|
119
|
+
}, [session, status, setSession, clearSession]);
|
|
120
|
+
return (0, jsx_runtime_1.jsx)(jsx_runtime_1.Fragment, { children: children });
|
|
121
|
+
}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
interface SignalRHealthCheckProps {
|
|
2
|
+
className?: string;
|
|
3
|
+
idpBaseUrl: string;
|
|
4
|
+
}
|
|
5
|
+
/**
|
|
6
|
+
* Simple health check component using SignalR connection state
|
|
7
|
+
* Following Occam's Razor: Connection alive = Service healthy, Connection dead = Service unhealthy
|
|
8
|
+
*/
|
|
9
|
+
export default function SignalRHealthCheck({ className, idpBaseUrl }: SignalRHealthCheckProps): import("react/jsx-runtime").JSX.Element | null;
|
|
10
|
+
export {};
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
'use client';
|
|
3
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
4
|
+
exports.default = SignalRHealthCheck;
|
|
5
|
+
const jsx_runtime_1 = require("react/jsx-runtime");
|
|
6
|
+
const react_1 = require("react");
|
|
7
|
+
const signalrActivityService_1 = require("../services/signalrActivityService");
|
|
8
|
+
/**
|
|
9
|
+
* Simple health check component using SignalR connection state
|
|
10
|
+
* Following Occam's Razor: Connection alive = Service healthy, Connection dead = Service unhealthy
|
|
11
|
+
*/
|
|
12
|
+
function SignalRHealthCheck({ className = '', idpBaseUrl }) {
|
|
13
|
+
// Allow disabling via env for noisy environments (e.g., production demos)
|
|
14
|
+
if (process.env.NEXT_PUBLIC_DISABLE_HEALTH_MONITOR === 'true') {
|
|
15
|
+
return null;
|
|
16
|
+
}
|
|
17
|
+
const [healthStatus, setHealthStatus] = (0, react_1.useState)({
|
|
18
|
+
isHealthy: false,
|
|
19
|
+
message: 'Initializing...',
|
|
20
|
+
lastHeartbeat: null,
|
|
21
|
+
connectionId: null
|
|
22
|
+
});
|
|
23
|
+
const [isInitializing, setIsInitializing] = (0, react_1.useState)(true);
|
|
24
|
+
(0, react_1.useEffect)(() => {
|
|
25
|
+
let unsubscribe = null;
|
|
26
|
+
const initializeHealthService = async () => {
|
|
27
|
+
try {
|
|
28
|
+
// Subscribe to health status changes
|
|
29
|
+
unsubscribe = signalrActivityService_1.signalRActivityService.subscribe((status) => {
|
|
30
|
+
setHealthStatus(status);
|
|
31
|
+
setIsInitializing(false);
|
|
32
|
+
});
|
|
33
|
+
// Start the health monitoring
|
|
34
|
+
await signalrActivityService_1.signalRActivityService.start(idpBaseUrl);
|
|
35
|
+
}
|
|
36
|
+
catch (error) {
|
|
37
|
+
// Handle service unavailable gracefully without console spam
|
|
38
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
39
|
+
const isServiceDown = errorMessage.includes('ERR_CONNECTION_REFUSED') ||
|
|
40
|
+
errorMessage.includes('Failed to fetch');
|
|
41
|
+
if (isServiceDown) {
|
|
42
|
+
console.info('Health monitoring: Backend service unavailable');
|
|
43
|
+
setHealthStatus({
|
|
44
|
+
isHealthy: false,
|
|
45
|
+
message: 'Backend service offline',
|
|
46
|
+
lastHeartbeat: null,
|
|
47
|
+
connectionId: null
|
|
48
|
+
});
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
console.error('Failed to initialize SignalR health service:', error);
|
|
52
|
+
setHealthStatus({
|
|
53
|
+
isHealthy: false,
|
|
54
|
+
message: 'Failed to initialize health monitoring',
|
|
55
|
+
lastHeartbeat: null,
|
|
56
|
+
connectionId: null
|
|
57
|
+
});
|
|
58
|
+
}
|
|
59
|
+
setIsInitializing(false);
|
|
60
|
+
}
|
|
61
|
+
};
|
|
62
|
+
initializeHealthService();
|
|
63
|
+
// Cleanup on unmount
|
|
64
|
+
return () => {
|
|
65
|
+
if (unsubscribe) {
|
|
66
|
+
unsubscribe();
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
}, [idpBaseUrl]);
|
|
70
|
+
const getStatusColor = () => {
|
|
71
|
+
if (isInitializing)
|
|
72
|
+
return 'text-yellow-600';
|
|
73
|
+
return healthStatus.isHealthy ? 'text-green-600' : 'text-orange-600'; // Orange instead of alarming red
|
|
74
|
+
};
|
|
75
|
+
const getStatusIcon = () => {
|
|
76
|
+
if (isInitializing)
|
|
77
|
+
return '🔄';
|
|
78
|
+
return healthStatus.isHealthy ? '✅' : '🔶'; // Orange diamond instead of scary red X
|
|
79
|
+
};
|
|
80
|
+
const getStatusMessage = () => {
|
|
81
|
+
if (isInitializing)
|
|
82
|
+
return 'Connecting to service...';
|
|
83
|
+
return healthStatus.message;
|
|
84
|
+
};
|
|
85
|
+
const formatLastHeartbeat = () => {
|
|
86
|
+
if (!healthStatus.lastHeartbeat)
|
|
87
|
+
return null;
|
|
88
|
+
const now = new Date();
|
|
89
|
+
const diff = Math.floor((now.getTime() - healthStatus.lastHeartbeat.getTime()) / 1000);
|
|
90
|
+
if (diff < 60)
|
|
91
|
+
return `${diff}s ago`;
|
|
92
|
+
if (diff < 3600)
|
|
93
|
+
return `${Math.floor(diff / 60)}m ago`;
|
|
94
|
+
return `${Math.floor(diff / 3600)}h ago`;
|
|
95
|
+
};
|
|
96
|
+
return ((0, jsx_runtime_1.jsxs)("div", { className: `flex items-center space-x-2 ${className}`, children: [(0, jsx_runtime_1.jsx)("span", { className: "text-lg", role: "img", "aria-label": "status", children: getStatusIcon() }), (0, jsx_runtime_1.jsxs)("div", { className: "flex flex-col", children: [(0, jsx_runtime_1.jsx)("span", { className: `text-sm font-medium ${getStatusColor()}`, children: getStatusMessage() }), healthStatus.lastHeartbeat && ((0, jsx_runtime_1.jsxs)("span", { className: "text-xs text-gray-500", children: ["Last heartbeat: ", formatLastHeartbeat()] }))] })] }));
|
|
97
|
+
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
export interface NavItem {
|
|
2
|
+
href: string;
|
|
3
|
+
label: string;
|
|
4
|
+
icon?: React.ReactNode;
|
|
5
|
+
}
|
|
6
|
+
export interface NavSection {
|
|
7
|
+
title?: string;
|
|
8
|
+
items: Array<{
|
|
9
|
+
label: string;
|
|
10
|
+
icon?: React.ReactNode;
|
|
11
|
+
href?: string;
|
|
12
|
+
onClick?: () => void;
|
|
13
|
+
}>;
|
|
14
|
+
}
|
|
15
|
+
export interface MobileNavDrawerProps {
|
|
16
|
+
isOpen: boolean;
|
|
17
|
+
onClose: () => void;
|
|
18
|
+
navItems: NavItem[];
|
|
19
|
+
/** Extra sections like Admin, rendered after nav items with optional title */
|
|
20
|
+
customSections?: NavSection[];
|
|
21
|
+
/** Base path for account link (default: '/account') */
|
|
22
|
+
basePath?: string;
|
|
23
|
+
/** Custom sign-in handler (default: next-auth signIn) */
|
|
24
|
+
onSignIn?: () => void;
|
|
25
|
+
/** Callback URL after sign in (default: '/dashboard') */
|
|
26
|
+
signInCallbackUrl?: string;
|
|
27
|
+
/** Custom unauthenticated actions (replaces default Login + Start Free buttons) */
|
|
28
|
+
unauthActions?: React.ReactNode;
|
|
29
|
+
/** Custom authenticated footer (replaces default "Account Settings" link) */
|
|
30
|
+
authFooter?: React.ReactNode;
|
|
31
|
+
}
|
|
32
|
+
export declare function MobileNavDrawer({ isOpen, onClose, navItems, customSections, basePath, onSignIn, signInCallbackUrl, unauthActions, authFooter, }: MobileNavDrawerProps): import("react/jsx-runtime").JSX.Element;
|