@atproto/oauth-provider 0.6.6 → 0.7.1
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/CHANGELOG.md +49 -0
- package/dist/access-token/access-token-mode.d.ts +5 -0
- package/dist/access-token/access-token-mode.d.ts.map +1 -0
- package/dist/access-token/access-token-mode.js +9 -0
- package/dist/access-token/access-token-mode.js.map +1 -0
- package/dist/account/account-manager.d.ts +13 -7
- package/dist/account/account-manager.d.ts.map +1 -1
- package/dist/account/account-manager.js +69 -52
- package/dist/account/account-manager.js.map +1 -1
- package/dist/account/account-store.d.ts +88 -77
- package/dist/account/account-store.d.ts.map +1 -1
- package/dist/account/account-store.js +24 -73
- package/dist/account/account-store.js.map +1 -1
- package/dist/account/sign-in-data.d.ts +4 -13
- package/dist/account/sign-in-data.d.ts.map +1 -1
- package/dist/account/sign-in-data.js +9 -9
- package/dist/account/sign-in-data.js.map +1 -1
- package/dist/account/sign-up-input.d.ts +4 -5
- package/dist/account/sign-up-input.d.ts.map +1 -1
- package/dist/account/sign-up-input.js +13 -3
- package/dist/account/sign-up-input.js.map +1 -1
- package/dist/client/client-manager.d.ts +4 -1
- package/dist/client/client-manager.d.ts.map +1 -1
- package/dist/client/client-manager.js +13 -1
- package/dist/client/client-manager.js.map +1 -1
- package/dist/client/client-store.d.ts +1 -1
- package/dist/client/client-store.d.ts.map +1 -1
- package/dist/constants.d.ts +5 -1
- package/dist/constants.d.ts.map +1 -1
- package/dist/constants.js +6 -2
- package/dist/constants.js.map +1 -1
- package/dist/customization/branding.d.ts +54 -0
- package/dist/customization/branding.d.ts.map +1 -0
- package/dist/customization/branding.js +13 -0
- package/dist/customization/branding.js.map +1 -0
- package/dist/customization/build-customization-css.d.ts +3 -0
- package/dist/customization/build-customization-css.d.ts.map +1 -0
- package/dist/customization/build-customization-css.js +27 -0
- package/dist/customization/build-customization-css.js.map +1 -0
- package/dist/customization/build-customization-data.d.ts +4 -0
- package/dist/customization/build-customization-data.d.ts.map +1 -0
- package/dist/customization/build-customization-data.js +18 -0
- package/dist/customization/build-customization-data.js.map +1 -0
- package/dist/customization/colors.d.ts +7 -0
- package/dist/customization/colors.d.ts.map +1 -0
- package/dist/customization/colors.js +27 -0
- package/dist/customization/colors.js.map +1 -0
- package/dist/customization/customization.d.ts +129 -0
- package/dist/customization/customization.d.ts.map +1 -0
- package/dist/customization/customization.js +26 -0
- package/dist/customization/customization.js.map +1 -0
- package/dist/customization/links.d.ts +26 -0
- package/dist/customization/links.d.ts.map +1 -0
- package/dist/customization/links.js +12 -0
- package/dist/customization/links.js.map +1 -0
- package/dist/device/device-id.d.ts +1 -0
- package/dist/device/device-id.d.ts.map +1 -1
- package/dist/device/device-id.js +4 -0
- package/dist/device/device-id.js.map +1 -1
- package/dist/device/device-manager.d.ts +6 -36
- package/dist/device/device-manager.d.ts.map +1 -1
- package/dist/device/device-manager.js +49 -43
- package/dist/device/device-manager.js.map +1 -1
- package/dist/device/device-store.d.ts +1 -0
- package/dist/device/device-store.d.ts.map +1 -1
- package/dist/device/device-store.js.map +1 -1
- package/dist/dpop/dpop-manager.d.ts +3 -3
- package/dist/dpop/dpop-nonce.d.ts +3 -3
- package/dist/dpop/dpop-nonce.d.ts.map +1 -1
- package/dist/errors/access-denied-error.d.ts +4 -3
- package/dist/errors/access-denied-error.d.ts.map +1 -1
- package/dist/errors/access-denied-error.js +5 -6
- package/dist/errors/access-denied-error.js.map +1 -1
- package/dist/{output/build-error-payload.d.ts → errors/error-parser.d.ts} +1 -1
- package/dist/errors/error-parser.d.ts.map +1 -0
- package/dist/{output/build-error-payload.js → errors/error-parser.js} +2 -2
- package/dist/errors/error-parser.js.map +1 -0
- package/dist/errors/invalid-grant-error.d.ts +1 -0
- package/dist/errors/invalid-grant-error.d.ts.map +1 -1
- package/dist/errors/invalid-grant-error.js +5 -0
- package/dist/errors/invalid-grant-error.js.map +1 -1
- package/dist/errors/login-required-error.d.ts +1 -0
- package/dist/errors/login-required-error.d.ts.map +1 -1
- package/dist/errors/login-required-error.js +5 -0
- package/dist/errors/login-required-error.js.map +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +1 -0
- package/dist/index.js.map +1 -1
- package/dist/lib/html/build-document.d.ts +2 -2
- package/dist/lib/html/build-document.d.ts.map +1 -1
- package/dist/lib/html/build-document.js +4 -0
- package/dist/lib/html/build-document.js.map +1 -1
- package/dist/lib/html/hydration-data.d.ts +4 -0
- package/dist/lib/html/hydration-data.d.ts.map +1 -0
- package/dist/{output/backend-data.js → lib/html/hydration-data.js} +8 -8
- package/dist/lib/html/hydration-data.js.map +1 -0
- package/dist/lib/html/tags.d.ts +1 -1
- package/dist/lib/html/tags.d.ts.map +1 -1
- package/dist/lib/html/tags.js +1 -1
- package/dist/lib/html/tags.js.map +1 -1
- package/dist/lib/http/accept.d.ts +2 -2
- package/dist/lib/http/accept.d.ts.map +1 -1
- package/dist/lib/http/accept.js +1 -1
- package/dist/lib/http/accept.js.map +1 -1
- package/dist/lib/http/context.d.ts +2 -4
- package/dist/lib/http/context.d.ts.map +1 -1
- package/dist/lib/http/context.js +29 -4
- package/dist/lib/http/context.js.map +1 -1
- package/dist/lib/http/headers.d.ts +3 -0
- package/dist/lib/http/headers.d.ts.map +1 -0
- package/dist/lib/http/headers.js +14 -0
- package/dist/lib/http/headers.js.map +1 -0
- package/dist/lib/http/index.d.ts +1 -0
- package/dist/lib/http/index.d.ts.map +1 -1
- package/dist/lib/http/index.js +1 -0
- package/dist/lib/http/index.js.map +1 -1
- package/dist/lib/http/middleware.d.ts +1 -1
- package/dist/lib/http/middleware.d.ts.map +1 -1
- package/dist/lib/http/middleware.js +8 -24
- package/dist/lib/http/middleware.js.map +1 -1
- package/dist/lib/http/parser.d.ts +3 -3
- package/dist/lib/http/parser.d.ts.map +1 -1
- package/dist/lib/http/request.d.ts +13 -9
- package/dist/lib/http/request.d.ts.map +1 -1
- package/dist/lib/http/request.js +27 -49
- package/dist/lib/http/request.js.map +1 -1
- package/dist/lib/http/response.d.ts +6 -2
- package/dist/lib/http/response.d.ts.map +1 -1
- package/dist/lib/http/response.js +31 -11
- package/dist/lib/http/response.js.map +1 -1
- package/dist/lib/http/route.d.ts +3 -3
- package/dist/lib/http/route.d.ts.map +1 -1
- package/dist/lib/http/route.js +1 -1
- package/dist/lib/http/route.js.map +1 -1
- package/dist/lib/http/router.d.ts +12 -11
- package/dist/lib/http/router.d.ts.map +1 -1
- package/dist/lib/http/router.js +26 -34
- package/dist/lib/http/router.js.map +1 -1
- package/dist/lib/http/security-headers.js +1 -1
- package/dist/lib/http/security-headers.js.map +1 -1
- package/dist/lib/http/stream.d.ts +3 -3
- package/dist/lib/http/stream.d.ts.map +1 -1
- package/dist/lib/http/types.d.ts +1 -1
- package/dist/lib/http/types.d.ts.map +1 -1
- package/dist/lib/send-web-page.d.ts +8 -0
- package/dist/lib/send-web-page.d.ts.map +1 -0
- package/dist/{output → lib}/send-web-page.js +9 -7
- package/dist/lib/send-web-page.js.map +1 -0
- package/dist/lib/util/authorization-header.d.ts.map +1 -1
- package/dist/lib/util/color.d.ts +32 -0
- package/dist/lib/util/color.d.ts.map +1 -0
- package/dist/lib/util/color.js +116 -0
- package/dist/lib/util/color.js.map +1 -0
- package/dist/lib/util/crypto.d.ts +1 -0
- package/dist/lib/util/crypto.d.ts.map +1 -1
- package/dist/lib/util/crypto.js +8 -3
- package/dist/lib/util/crypto.js.map +1 -1
- package/dist/lib/util/function.d.ts +1 -0
- package/dist/lib/util/function.d.ts.map +1 -1
- package/dist/lib/util/function.js +12 -0
- package/dist/lib/util/function.js.map +1 -1
- package/dist/lib/util/locale.d.ts +20 -0
- package/dist/lib/util/locale.d.ts.map +1 -0
- package/dist/lib/util/locale.js +14 -0
- package/dist/lib/util/locale.js.map +1 -0
- package/dist/lib/util/time.d.ts +1 -1
- package/dist/lib/util/time.d.ts.map +1 -1
- package/dist/lib/util/time.js +1 -1
- package/dist/lib/util/time.js.map +1 -1
- package/dist/lib/util/type.d.ts +22 -0
- package/dist/lib/util/type.d.ts.map +1 -1
- package/dist/lib/util/type.js.map +1 -1
- package/dist/lib/util/ui8.d.ts +4 -0
- package/dist/lib/util/ui8.d.ts.map +1 -0
- package/dist/lib/util/ui8.js +17 -0
- package/dist/lib/util/ui8.js.map +1 -0
- package/dist/lib/util/zod-error.d.ts +2 -0
- package/dist/lib/util/zod-error.d.ts.map +1 -0
- package/dist/lib/util/zod-error.js +16 -0
- package/dist/lib/util/zod-error.js.map +1 -0
- package/dist/oauth-errors.d.ts +22 -22
- package/dist/oauth-errors.d.ts.map +1 -1
- package/dist/oauth-errors.js +37 -45
- package/dist/oauth-errors.js.map +1 -1
- package/dist/oauth-hooks.d.ts +11 -23
- package/dist/oauth-hooks.d.ts.map +1 -1
- package/dist/oauth-hooks.js.map +1 -1
- package/dist/oauth-middleware.d.ts +12 -0
- package/dist/oauth-middleware.d.ts.map +1 -0
- package/dist/oauth-middleware.js +32 -0
- package/dist/oauth-middleware.js.map +1 -0
- package/dist/oauth-provider.d.ts +109 -113
- package/dist/oauth-provider.d.ts.map +1 -1
- package/dist/oauth-provider.js +124 -542
- package/dist/oauth-provider.js.map +1 -1
- package/dist/oauth-verifier.d.ts +7 -26
- package/dist/oauth-verifier.d.ts.map +1 -1
- package/dist/oauth-verifier.js +6 -16
- package/dist/oauth-verifier.js.map +1 -1
- package/dist/request/code.d.ts.map +1 -1
- package/dist/request/request-data.d.ts +2 -4
- package/dist/request/request-data.d.ts.map +1 -1
- package/dist/request/request-data.js.map +1 -1
- package/dist/request/request-manager.d.ts +4 -2
- package/dist/request/request-manager.d.ts.map +1 -1
- package/dist/request/request-manager.js +9 -8
- package/dist/request/request-manager.js.map +1 -1
- package/dist/request/request-store.d.ts +6 -0
- package/dist/request/request-store.d.ts.map +1 -1
- package/dist/request/request-store.js +3 -1
- package/dist/request/request-store.js.map +1 -1
- package/dist/result/authorization-redirect-parameters.d.ts +18 -0
- package/dist/result/authorization-redirect-parameters.d.ts.map +1 -0
- package/dist/result/authorization-redirect-parameters.js +3 -0
- package/dist/result/authorization-redirect-parameters.js.map +1 -0
- package/dist/result/authorization-result-authorize-page.d.ts +13 -0
- package/dist/result/authorization-result-authorize-page.d.ts.map +1 -0
- package/dist/result/authorization-result-authorize-page.js +3 -0
- package/dist/result/authorization-result-authorize-page.js.map +1 -0
- package/dist/result/authorization-result-redirect.d.ts +8 -0
- package/dist/result/authorization-result-redirect.d.ts.map +1 -0
- package/dist/result/authorization-result-redirect.js +3 -0
- package/dist/result/authorization-result-redirect.js.map +1 -0
- package/dist/router/assets/assets-manifest.d.ts +10 -0
- package/dist/router/assets/assets-manifest.d.ts.map +1 -0
- package/dist/router/assets/assets-manifest.js +77 -0
- package/dist/router/assets/assets-manifest.js.map +1 -0
- package/dist/router/assets/assets.d.ts +16 -0
- package/dist/router/assets/assets.d.ts.map +1 -0
- package/dist/router/assets/assets.js +43 -0
- package/dist/router/assets/assets.js.map +1 -0
- package/dist/router/assets/csrf.d.ts +4 -0
- package/dist/router/assets/csrf.d.ts.map +1 -0
- package/dist/router/assets/csrf.js +51 -0
- package/dist/router/assets/csrf.js.map +1 -0
- package/dist/router/assets/send-account-page.d.ts +7 -0
- package/dist/router/assets/send-account-page.d.ts.map +1 -0
- package/dist/router/assets/send-account-page.js +34 -0
- package/dist/router/assets/send-account-page.js.map +1 -0
- package/dist/router/assets/send-authorization-page.d.ts +5 -0
- package/dist/router/assets/send-authorization-page.d.ts.map +1 -0
- package/dist/router/assets/send-authorization-page.js +49 -0
- package/dist/router/assets/send-authorization-page.js.map +1 -0
- package/dist/router/assets/send-error-page.d.ts +4 -0
- package/dist/router/assets/send-error-page.d.ts.map +1 -0
- package/dist/router/assets/send-error-page.js +34 -0
- package/dist/router/assets/send-error-page.js.map +1 -0
- package/dist/router/create-account-page-middleware.d.ts +6 -0
- package/dist/router/create-account-page-middleware.d.ts.map +1 -0
- package/dist/router/create-account-page-middleware.js +39 -0
- package/dist/router/create-account-page-middleware.js.map +1 -0
- package/dist/router/create-api-middleware.d.ts +8 -0
- package/dist/router/create-api-middleware.d.ts.map +1 -0
- package/dist/router/create-api-middleware.js +501 -0
- package/dist/router/create-api-middleware.js.map +1 -0
- package/dist/router/create-authorization-page-middleware.d.ts +6 -0
- package/dist/router/create-authorization-page-middleware.d.ts.map +1 -0
- package/dist/router/create-authorization-page-middleware.js +104 -0
- package/dist/router/create-authorization-page-middleware.js.map +1 -0
- package/dist/router/create-oauth-middleware.d.ts +6 -0
- package/dist/router/create-oauth-middleware.d.ts.map +1 -0
- package/dist/router/create-oauth-middleware.js +142 -0
- package/dist/router/create-oauth-middleware.js.map +1 -0
- package/dist/router/error-handler.d.ts +3 -0
- package/dist/router/error-handler.d.ts.map +1 -0
- package/dist/{account/account.js → router/error-handler.js} +1 -1
- package/dist/router/error-handler.js.map +1 -0
- package/dist/router/middleware-options.d.ts +6 -0
- package/dist/router/middleware-options.d.ts.map +1 -0
- package/dist/router/middleware-options.js +3 -0
- package/dist/router/middleware-options.js.map +1 -0
- package/dist/router/send-redirect.d.ts +16 -0
- package/dist/router/send-redirect.d.ts.map +1 -0
- package/dist/{output/send-authorize-redirect.js → router/send-redirect.js} +40 -24
- package/dist/router/send-redirect.js.map +1 -0
- package/dist/{token/token-claims.d.ts → signer/api-token-payload.d.ts} +237 -232
- package/dist/signer/api-token-payload.d.ts.map +1 -0
- package/dist/signer/api-token-payload.js +17 -0
- package/dist/signer/api-token-payload.js.map +1 -0
- package/dist/signer/signed-token-payload.d.ts +164 -159
- package/dist/signer/signed-token-payload.d.ts.map +1 -1
- package/dist/signer/signed-token-payload.js +10 -16
- package/dist/signer/signed-token-payload.js.map +1 -1
- package/dist/signer/signer.d.ts +42 -11246
- package/dist/signer/signer.d.ts.map +1 -1
- package/dist/signer/signer.js +30 -15
- package/dist/signer/signer.js.map +1 -1
- package/dist/token/refresh-token.d.ts.map +1 -1
- package/dist/token/token-data.d.ts +1 -1
- package/dist/token/token-data.d.ts.map +1 -1
- package/dist/token/token-id.d.ts.map +1 -1
- package/dist/token/token-manager.d.ts +28 -26
- package/dist/token/token-manager.d.ts.map +1 -1
- package/dist/token/token-manager.js +138 -196
- package/dist/token/token-manager.js.map +1 -1
- package/dist/token/token-store.d.ts +4 -4
- package/dist/token/token-store.d.ts.map +1 -1
- package/dist/token/token-store.js +1 -0
- package/dist/token/token-store.js.map +1 -1
- package/dist/token/verify-token-claims.d.ts +3 -3
- package/dist/token/verify-token-claims.d.ts.map +1 -1
- package/dist/token/verify-token-claims.js +1 -1
- package/dist/token/verify-token-claims.js.map +1 -1
- package/dist/types/email-otp.d.ts +3 -0
- package/dist/types/email-otp.d.ts.map +1 -0
- package/dist/types/email-otp.js +6 -0
- package/dist/types/email-otp.js.map +1 -0
- package/dist/types/email.d.ts +3 -0
- package/dist/types/email.d.ts.map +1 -0
- package/dist/types/email.js +29 -0
- package/dist/types/email.js.map +1 -0
- package/dist/types/handle.d.ts +3 -0
- package/dist/types/handle.d.ts.map +1 -0
- package/dist/types/handle.js +22 -0
- package/dist/types/handle.js.map +1 -0
- package/dist/types/invite-code.d.ts +4 -0
- package/dist/types/invite-code.d.ts.map +1 -0
- package/dist/types/invite-code.js +6 -0
- package/dist/types/invite-code.js.map +1 -0
- package/dist/types/password.d.ts +4 -0
- package/dist/types/password.d.ts.map +1 -0
- package/dist/types/password.js +7 -0
- package/dist/types/password.js.map +1 -0
- package/package.json +11 -14
- package/src/access-token/access-token-mode.ts +4 -0
- package/src/account/account-manager.ts +105 -75
- package/src/account/account-store.ts +118 -114
- package/src/account/sign-in-data.ts +10 -10
- package/src/account/sign-up-input.ts +13 -4
- package/src/client/client-manager.ts +34 -2
- package/src/client/client-store.ts +1 -1
- package/src/constants.ts +6 -1
- package/src/customization/branding.ts +12 -0
- package/src/customization/build-customization-css.ts +30 -0
- package/src/customization/build-customization-data.ts +22 -0
- package/src/customization/colors.ts +30 -0
- package/src/customization/customization.ts +25 -0
- package/src/customization/links.ts +10 -0
- package/src/device/device-id.ts +5 -0
- package/src/device/device-manager.ts +76 -66
- package/src/device/device-store.ts +2 -0
- package/src/errors/access-denied-error.ts +24 -17
- package/src/{output/build-error-payload.ts → errors/error-parser.ts} +1 -1
- package/src/errors/invalid-grant-error.ts +5 -0
- package/src/errors/login-required-error.ts +10 -0
- package/src/index.ts +1 -0
- package/src/lib/html/build-document.ts +6 -4
- package/src/{output/backend-data.ts → lib/html/hydration-data.ts} +7 -5
- package/src/lib/html/tags.ts +2 -2
- package/src/lib/http/accept.ts +3 -3
- package/src/lib/http/context.ts +41 -10
- package/src/lib/http/headers.ts +15 -0
- package/src/lib/http/index.ts +1 -0
- package/src/lib/http/middleware.ts +8 -23
- package/src/lib/http/request.ts +40 -75
- package/src/lib/http/response.ts +39 -15
- package/src/lib/http/route.ts +8 -5
- package/src/lib/http/router.ts +40 -46
- package/src/lib/http/security-headers.ts +1 -1
- package/src/lib/http/types.ts +1 -6
- package/src/{output → lib}/send-web-page.ts +10 -9
- package/src/lib/util/color.ts +132 -0
- package/src/lib/util/crypto.ts +9 -4
- package/src/lib/util/function.ts +14 -0
- package/src/lib/util/locale.ts +18 -0
- package/src/lib/util/time.ts +3 -4
- package/src/lib/util/type.ts +24 -0
- package/src/lib/util/ui8.ts +14 -0
- package/src/lib/util/zod-error.ts +14 -0
- package/src/oauth-errors.ts +22 -22
- package/src/oauth-hooks.ts +11 -24
- package/src/oauth-middleware.ts +53 -0
- package/src/oauth-provider.ts +290 -1061
- package/src/oauth-verifier.ts +9 -55
- package/src/request/request-data.ts +5 -4
- package/src/request/request-manager.ts +11 -11
- package/src/request/request-store.ts +7 -0
- package/src/result/authorization-redirect-parameters.ts +24 -0
- package/src/result/authorization-result-authorize-page.ts +14 -0
- package/src/result/authorization-result-redirect.ts +8 -0
- package/src/router/assets/assets-manifest.ts +115 -0
- package/src/router/assets/assets.ts +54 -0
- package/src/router/assets/csrf.ts +63 -0
- package/src/router/assets/send-account-page.ts +43 -0
- package/src/router/assets/send-authorization-page.ts +62 -0
- package/src/router/assets/send-error-page.ts +42 -0
- package/src/router/create-account-page-middleware.ts +69 -0
- package/src/router/create-api-middleware.ts +814 -0
- package/src/router/create-authorization-page-middleware.ts +173 -0
- package/src/router/create-oauth-middleware.ts +247 -0
- package/src/router/error-handler.ts +6 -0
- package/src/router/middleware-options.ts +9 -0
- package/src/router/send-redirect.ts +142 -0
- package/src/signer/api-token-payload.ts +18 -0
- package/src/signer/signed-token-payload.ts +18 -28
- package/src/signer/signer.ts +49 -34
- package/src/token/token-data.ts +1 -1
- package/src/token/token-manager.ts +190 -239
- package/src/token/token-store.ts +6 -4
- package/src/token/verify-token-claims.ts +4 -4
- package/src/types/email-otp.ts +3 -0
- package/src/types/email.ts +26 -0
- package/src/types/handle.ts +18 -0
- package/src/types/invite-code.ts +4 -0
- package/src/types/password.ts +4 -0
- package/tsconfig.build.tsbuildinfo +1 -0
- package/tsconfig.json +1 -1
- package/dist/access-token/access-token-type.d.ts +0 -6
- package/dist/access-token/access-token-type.d.ts.map +0 -1
- package/dist/access-token/access-token-type.js +0 -10
- package/dist/access-token/access-token-type.js.map +0 -1
- package/dist/account/account.d.ts +0 -2
- package/dist/account/account.d.ts.map +0 -1
- package/dist/account/account.js.map +0 -1
- package/dist/assets/assets-middleware.d.ts +0 -5
- package/dist/assets/assets-middleware.d.ts.map +0 -1
- package/dist/assets/assets-middleware.js +0 -41
- package/dist/assets/assets-middleware.js.map +0 -1
- package/dist/lib/locale.d.ts +0 -15
- package/dist/lib/locale.d.ts.map +0 -1
- package/dist/lib/locale.js +0 -17
- package/dist/lib/locale.js.map +0 -1
- package/dist/output/backend-data.d.ts +0 -4
- package/dist/output/backend-data.d.ts.map +0 -1
- package/dist/output/backend-data.js.map +0 -1
- package/dist/output/build-authorize-data.d.ts +0 -29
- package/dist/output/build-authorize-data.d.ts.map +0 -1
- package/dist/output/build-authorize-data.js +0 -21
- package/dist/output/build-authorize-data.js.map +0 -1
- package/dist/output/build-customization-data.d.ts +0 -234
- package/dist/output/build-customization-data.d.ts.map +0 -1
- package/dist/output/build-customization-data.js +0 -174
- package/dist/output/build-customization-data.js.map +0 -1
- package/dist/output/build-error-data.d.ts +0 -3
- package/dist/output/build-error-data.d.ts.map +0 -1
- package/dist/output/build-error-data.js +0 -10
- package/dist/output/build-error-data.js.map +0 -1
- package/dist/output/build-error-payload.d.ts.map +0 -1
- package/dist/output/build-error-payload.js.map +0 -1
- package/dist/output/output-manager.d.ts +0 -28
- package/dist/output/output-manager.d.ts.map +0 -1
- package/dist/output/output-manager.js +0 -134
- package/dist/output/output-manager.js.map +0 -1
- package/dist/output/send-authorize-redirect.d.ts +0 -25
- package/dist/output/send-authorize-redirect.d.ts.map +0 -1
- package/dist/output/send-authorize-redirect.js.map +0 -1
- package/dist/output/send-web-page.d.ts +0 -8
- package/dist/output/send-web-page.d.ts.map +0 -1
- package/dist/output/send-web-page.js.map +0 -1
- package/dist/token/token-claims.d.ts.map +0 -1
- package/dist/token/token-claims.js +0 -27
- package/dist/token/token-claims.js.map +0 -1
- package/src/access-token/access-token-type.ts +0 -5
- package/src/account/account.ts +0 -1
- package/src/assets/assets-middleware.ts +0 -44
- package/src/lib/locale.ts +0 -21
- package/src/output/build-authorize-data.ts +0 -53
- package/src/output/build-customization-data.ts +0 -217
- package/src/output/build-error-data.ts +0 -8
- package/src/output/output-manager.ts +0 -188
- package/src/output/send-authorize-redirect.ts +0 -137
- package/src/token/token-claims.ts +0 -30
- package/tsconfig.backend.tsbuildinfo +0 -1
- /package/{tsconfig.backend.json → tsconfig.build.json} +0 -0
@@ -0,0 +1,814 @@
|
|
1
|
+
import type { IncomingMessage, ServerResponse } from 'node:http'
|
2
|
+
import createHttpError from 'http-errors'
|
3
|
+
import { z } from 'zod'
|
4
|
+
import { signedJwtSchema } from '@atproto/jwk'
|
5
|
+
import {
|
6
|
+
API_ENDPOINT_PREFIX,
|
7
|
+
ActiveAccountSession,
|
8
|
+
ActiveDeviceSession,
|
9
|
+
ActiveOAuthSession,
|
10
|
+
ApiEndpoints,
|
11
|
+
ISODateString,
|
12
|
+
} from '@atproto/oauth-provider-api'
|
13
|
+
import {
|
14
|
+
OAuthAuthorizationRequestParameters,
|
15
|
+
OAuthRedirectUri,
|
16
|
+
OAuthResponseMode,
|
17
|
+
oauthRedirectUriSchema,
|
18
|
+
oauthResponseModeSchema,
|
19
|
+
} from '@atproto/oauth-types'
|
20
|
+
import { signInDataSchema } from '../account/sign-in-data.js'
|
21
|
+
import { signUpInputSchema } from '../account/sign-up-input.js'
|
22
|
+
import { DeviceId, deviceIdSchema } from '../device/device-id.js'
|
23
|
+
import { AccessDeniedError } from '../errors/access-denied-error.js'
|
24
|
+
import { buildErrorPayload, buildErrorStatus } from '../errors/error-parser.js'
|
25
|
+
import { InvalidRequestError } from '../errors/invalid-request-error.js'
|
26
|
+
import { WWWAuthenticateError } from '../errors/www-authenticate-error.js'
|
27
|
+
import {
|
28
|
+
Middleware,
|
29
|
+
RequestMetadata,
|
30
|
+
Router,
|
31
|
+
RouterCtx,
|
32
|
+
SubCtx,
|
33
|
+
jsonHandler,
|
34
|
+
parseHttpRequest,
|
35
|
+
subCtx,
|
36
|
+
validateFetchMode,
|
37
|
+
validateFetchSite,
|
38
|
+
validateOrigin,
|
39
|
+
validateReferrer,
|
40
|
+
} from '../lib/http/index.js'
|
41
|
+
import { RouteCtx, createRoute } from '../lib/http/route.js'
|
42
|
+
import { asArray } from '../lib/util/cast.js'
|
43
|
+
import { localeSchema } from '../lib/util/locale.js'
|
44
|
+
import type { Awaitable } from '../lib/util/type.js'
|
45
|
+
import type { OAuthProvider } from '../oauth-provider.js'
|
46
|
+
import { Sub, subSchema } from '../oidc/sub.js'
|
47
|
+
import { RequestUri, requestUriSchema } from '../request/request-uri.js'
|
48
|
+
import { AuthorizationRedirectParameters } from '../result/authorization-redirect-parameters.js'
|
49
|
+
import { tokenIdSchema } from '../token/token-id.js'
|
50
|
+
import { emailOtpSchema } from '../types/email-otp.js'
|
51
|
+
import { emailSchema } from '../types/email.js'
|
52
|
+
import { handleSchema } from '../types/handle.js'
|
53
|
+
import { newPasswordSchema } from '../types/password.js'
|
54
|
+
import { validateCsrfToken } from './assets/csrf.js'
|
55
|
+
import type { MiddlewareOptions } from './middleware-options.js'
|
56
|
+
import {
|
57
|
+
ERROR_REDIRECT_KEYS,
|
58
|
+
OAuthRedirectOptions,
|
59
|
+
OAuthRedirectQueryParameter,
|
60
|
+
SUCCESS_REDIRECT_KEYS,
|
61
|
+
buildRedirectMode,
|
62
|
+
buildRedirectParams,
|
63
|
+
buildRedirectUri,
|
64
|
+
} from './send-redirect.js'
|
65
|
+
|
66
|
+
const verifyHandleSchema = z.object({ handle: handleSchema }).strict()
|
67
|
+
|
68
|
+
export function createApiMiddleware<
|
69
|
+
Ctx extends object | void = void,
|
70
|
+
Req extends IncomingMessage = IncomingMessage,
|
71
|
+
Res extends ServerResponse = ServerResponse,
|
72
|
+
>(
|
73
|
+
server: OAuthProvider,
|
74
|
+
{ onError }: MiddlewareOptions<Req, Res>,
|
75
|
+
): Middleware<Ctx, Req, Res> {
|
76
|
+
const issuerUrl = new URL(server.issuer)
|
77
|
+
const issuerOrigin = issuerUrl.origin
|
78
|
+
const router = new Router<Ctx, Req, Res>(issuerUrl)
|
79
|
+
|
80
|
+
router.use(
|
81
|
+
apiRoute({
|
82
|
+
method: 'POST',
|
83
|
+
endpoint: '/verify-handle-availability',
|
84
|
+
schema: verifyHandleSchema,
|
85
|
+
async handler() {
|
86
|
+
await server.accountManager.verifyHandleAvailability(this.input.handle)
|
87
|
+
return { available: true }
|
88
|
+
},
|
89
|
+
}),
|
90
|
+
)
|
91
|
+
|
92
|
+
router.use(
|
93
|
+
apiRoute({
|
94
|
+
method: 'POST',
|
95
|
+
endpoint: '/sign-up',
|
96
|
+
schema: signUpInputSchema,
|
97
|
+
rotateDeviceCookies: true,
|
98
|
+
async handler() {
|
99
|
+
const { deviceId, deviceMetadata, input, requestUri } = this
|
100
|
+
|
101
|
+
const account = await server.accountManager.createAccount(
|
102
|
+
deviceId,
|
103
|
+
deviceMetadata,
|
104
|
+
input,
|
105
|
+
)
|
106
|
+
|
107
|
+
// Remember when not in the context of a request by default
|
108
|
+
const remember = requestUri == null
|
109
|
+
|
110
|
+
// Only "remember" the newly created account if it was not created during an
|
111
|
+
// OAuth flow.
|
112
|
+
if (remember) {
|
113
|
+
await server.accountManager.upsertDeviceAccount(deviceId, account.sub)
|
114
|
+
}
|
115
|
+
|
116
|
+
const ephemeralToken = remember
|
117
|
+
? undefined
|
118
|
+
: await server.signer.createEphemeralToken({
|
119
|
+
sub: account.sub,
|
120
|
+
deviceId,
|
121
|
+
requestUri: this.requestUri,
|
122
|
+
})
|
123
|
+
|
124
|
+
return { account, ephemeralToken }
|
125
|
+
},
|
126
|
+
}),
|
127
|
+
)
|
128
|
+
|
129
|
+
router.use(
|
130
|
+
apiRoute({
|
131
|
+
method: 'POST',
|
132
|
+
endpoint: '/sign-in',
|
133
|
+
schema: signInDataSchema.extend({ remember: z.boolean().optional() }),
|
134
|
+
rotateDeviceCookies: true,
|
135
|
+
async handler() {
|
136
|
+
const { deviceId, deviceMetadata, requestUri } = this
|
137
|
+
|
138
|
+
// Remember when not in the context of a request by default
|
139
|
+
const { remember = requestUri == null, ...input } = this.input
|
140
|
+
|
141
|
+
const account = await server.accountManager.authenticateAccount(
|
142
|
+
deviceId,
|
143
|
+
deviceMetadata,
|
144
|
+
input,
|
145
|
+
)
|
146
|
+
|
147
|
+
if (remember) {
|
148
|
+
await server.accountManager.upsertDeviceAccount(deviceId, account.sub)
|
149
|
+
} else {
|
150
|
+
// In case the user was already signed in, and signed in again, this
|
151
|
+
// time without "remember me", let's sign them off of the device.
|
152
|
+
await server.accountManager.removeDeviceAccount(deviceId, account.sub)
|
153
|
+
}
|
154
|
+
|
155
|
+
const ephemeralToken = remember
|
156
|
+
? undefined
|
157
|
+
: await server.signer.createEphemeralToken({
|
158
|
+
sub: account.sub,
|
159
|
+
deviceId,
|
160
|
+
requestUri,
|
161
|
+
})
|
162
|
+
|
163
|
+
if (requestUri) {
|
164
|
+
// Check if a consent is required for the client, but only if this
|
165
|
+
// call is made within the context of an oauth request.
|
166
|
+
|
167
|
+
const { clientId, parameters } = await server.requestManager.get(
|
168
|
+
requestUri,
|
169
|
+
deviceId,
|
170
|
+
)
|
171
|
+
|
172
|
+
const { authorizedClients } = await server.accountManager.getAccount(
|
173
|
+
account.sub,
|
174
|
+
)
|
175
|
+
|
176
|
+
return {
|
177
|
+
account,
|
178
|
+
ephemeralToken,
|
179
|
+
consentRequired: server.checkConsentRequired(
|
180
|
+
parameters,
|
181
|
+
authorizedClients.get(clientId),
|
182
|
+
),
|
183
|
+
}
|
184
|
+
}
|
185
|
+
|
186
|
+
return { account, ephemeralToken }
|
187
|
+
},
|
188
|
+
}),
|
189
|
+
)
|
190
|
+
|
191
|
+
router.use(
|
192
|
+
apiRoute({
|
193
|
+
method: 'POST',
|
194
|
+
endpoint: '/sign-out',
|
195
|
+
schema: z
|
196
|
+
.object({
|
197
|
+
sub: z.union([subSchema, z.array(subSchema)]),
|
198
|
+
})
|
199
|
+
.strict(),
|
200
|
+
rotateDeviceCookies: true,
|
201
|
+
async handler() {
|
202
|
+
const uniqueSubs = new Set(asArray(this.input.sub))
|
203
|
+
|
204
|
+
for (const sub of uniqueSubs) {
|
205
|
+
await server.accountManager.removeDeviceAccount(this.deviceId, sub)
|
206
|
+
}
|
207
|
+
|
208
|
+
return { success: true as const }
|
209
|
+
},
|
210
|
+
}),
|
211
|
+
)
|
212
|
+
|
213
|
+
router.use(
|
214
|
+
apiRoute({
|
215
|
+
method: 'POST',
|
216
|
+
endpoint: '/reset-password-request',
|
217
|
+
schema: z
|
218
|
+
.object({
|
219
|
+
locale: localeSchema,
|
220
|
+
email: emailSchema,
|
221
|
+
})
|
222
|
+
.strict(),
|
223
|
+
async handler() {
|
224
|
+
await server.accountManager.resetPasswordRequest(this.input)
|
225
|
+
return { success: true }
|
226
|
+
},
|
227
|
+
}),
|
228
|
+
)
|
229
|
+
|
230
|
+
router.use(
|
231
|
+
apiRoute({
|
232
|
+
method: 'POST',
|
233
|
+
endpoint: '/reset-password-confirm',
|
234
|
+
schema: z
|
235
|
+
.object({
|
236
|
+
token: emailOtpSchema,
|
237
|
+
password: newPasswordSchema,
|
238
|
+
})
|
239
|
+
.strict(),
|
240
|
+
async handler() {
|
241
|
+
await server.accountManager.resetPasswordConfirm(this.input)
|
242
|
+
return { success: true }
|
243
|
+
},
|
244
|
+
}),
|
245
|
+
)
|
246
|
+
|
247
|
+
router.use(
|
248
|
+
apiRoute({
|
249
|
+
method: 'GET',
|
250
|
+
endpoint: '/device-sessions',
|
251
|
+
schema: undefined,
|
252
|
+
async handler() {
|
253
|
+
const deviceAccounts = await server.accountManager.listDeviceAccounts(
|
254
|
+
this.deviceId,
|
255
|
+
)
|
256
|
+
|
257
|
+
return deviceAccounts.map(
|
258
|
+
(deviceAccount): ActiveDeviceSession => ({
|
259
|
+
account: deviceAccount.account,
|
260
|
+
loginRequired: server.checkLoginRequired(deviceAccount),
|
261
|
+
}),
|
262
|
+
)
|
263
|
+
},
|
264
|
+
}),
|
265
|
+
)
|
266
|
+
|
267
|
+
router.use(
|
268
|
+
apiRoute({
|
269
|
+
method: 'GET',
|
270
|
+
endpoint: '/oauth-sessions',
|
271
|
+
schema: z.object({ sub: subSchema }).strict(),
|
272
|
+
async handler(req, res) {
|
273
|
+
const { account } = await authenticate.call(this, req, res)
|
274
|
+
|
275
|
+
const tokenInfos = await server.tokenManager.listAccountTokens(
|
276
|
+
account.sub,
|
277
|
+
)
|
278
|
+
|
279
|
+
const clientIds = tokenInfos.map((tokenInfo) => tokenInfo.data.clientId)
|
280
|
+
|
281
|
+
const clients = await server.clientManager.loadClients(clientIds, {
|
282
|
+
onError: (err, clientId) => {
|
283
|
+
onError?.(req, res, err, `Failed to load client ${clientId}`)
|
284
|
+
return undefined // metadata won't be available in the UI
|
285
|
+
},
|
286
|
+
})
|
287
|
+
|
288
|
+
// @TODO: We should ideally filter sessions that are expired (or even
|
289
|
+
// expose the expiration date). This requires a change to the way
|
290
|
+
// TokenInfo are stored (see TokenManager#isTokenExpired and
|
291
|
+
// TokenManager#isTokenInactive).
|
292
|
+
return tokenInfos.map(({ id, data }): ActiveOAuthSession => {
|
293
|
+
return {
|
294
|
+
tokenId: id,
|
295
|
+
|
296
|
+
createdAt: data.createdAt.toISOString() as ISODateString,
|
297
|
+
updatedAt: data.updatedAt.toISOString() as ISODateString,
|
298
|
+
|
299
|
+
clientId: data.clientId,
|
300
|
+
clientMetadata: clients.get(data.clientId)?.metadata,
|
301
|
+
|
302
|
+
scope: data.parameters.scope,
|
303
|
+
}
|
304
|
+
})
|
305
|
+
},
|
306
|
+
}),
|
307
|
+
)
|
308
|
+
|
309
|
+
router.use(
|
310
|
+
apiRoute({
|
311
|
+
method: 'GET',
|
312
|
+
endpoint: '/account-sessions',
|
313
|
+
schema: z.object({ sub: subSchema }).strict(),
|
314
|
+
async handler(req, res) {
|
315
|
+
const { account } = await authenticate.call(this, req, res)
|
316
|
+
|
317
|
+
const deviceAccounts = await server.accountManager.listAccountDevices(
|
318
|
+
account.sub,
|
319
|
+
)
|
320
|
+
|
321
|
+
return deviceAccounts.map(
|
322
|
+
(accountSession): ActiveAccountSession => ({
|
323
|
+
deviceId: accountSession.deviceId,
|
324
|
+
deviceMetadata: {
|
325
|
+
ipAddress: accountSession.deviceData.ipAddress,
|
326
|
+
userAgent: accountSession.deviceData.userAgent,
|
327
|
+
lastSeenAt:
|
328
|
+
accountSession.deviceData.lastSeenAt.toISOString() as ISODateString,
|
329
|
+
},
|
330
|
+
|
331
|
+
isCurrentDevice: accountSession.deviceId === this.deviceId,
|
332
|
+
}),
|
333
|
+
)
|
334
|
+
},
|
335
|
+
}),
|
336
|
+
)
|
337
|
+
|
338
|
+
router.use(
|
339
|
+
apiRoute({
|
340
|
+
method: 'POST',
|
341
|
+
endpoint: '/revoke-account-session',
|
342
|
+
schema: z.object({ sub: subSchema, deviceId: deviceIdSchema }).strict(),
|
343
|
+
async handler() {
|
344
|
+
// @NOTE This route is not authenticated. If a user is able to steal
|
345
|
+
// another user's session cookie, we allow them to revoke the device
|
346
|
+
// session.
|
347
|
+
|
348
|
+
await server.accountManager.removeDeviceAccount(
|
349
|
+
this.input.deviceId,
|
350
|
+
this.input.sub,
|
351
|
+
)
|
352
|
+
|
353
|
+
return { success: true }
|
354
|
+
},
|
355
|
+
}),
|
356
|
+
)
|
357
|
+
|
358
|
+
router.use(
|
359
|
+
apiRoute({
|
360
|
+
method: 'POST',
|
361
|
+
endpoint: '/revoke-oauth-session',
|
362
|
+
schema: z.object({ sub: subSchema, tokenId: tokenIdSchema }).strict(),
|
363
|
+
async handler(req, res) {
|
364
|
+
const { account } = await authenticate.call(this, req, res)
|
365
|
+
|
366
|
+
const tokenInfo = await server.tokenManager.getTokenInfo(
|
367
|
+
this.input.tokenId,
|
368
|
+
)
|
369
|
+
|
370
|
+
if (tokenInfo.account.sub !== account.sub) {
|
371
|
+
// report this as though the token was not found
|
372
|
+
throw new InvalidRequestError(`Invalid token`)
|
373
|
+
}
|
374
|
+
|
375
|
+
await server.tokenManager.deleteToken(tokenInfo.id)
|
376
|
+
|
377
|
+
return { success: true }
|
378
|
+
},
|
379
|
+
}),
|
380
|
+
)
|
381
|
+
|
382
|
+
router.use(
|
383
|
+
apiRoute({
|
384
|
+
method: 'POST',
|
385
|
+
endpoint: '/accept',
|
386
|
+
schema: z.object({ sub: z.union([subSchema, signedJwtSchema]) }).strict(),
|
387
|
+
async handler(req, res) {
|
388
|
+
if (!this.requestUri) {
|
389
|
+
throw new InvalidRequestError(
|
390
|
+
'This endpoint can only be used in the context of an OAuth request',
|
391
|
+
)
|
392
|
+
}
|
393
|
+
|
394
|
+
// Any AccessDeniedError caught in this block will result in a redirect
|
395
|
+
// to the client's redirect_uri with an error.
|
396
|
+
try {
|
397
|
+
const { clientId, parameters } = await server.requestManager.get(
|
398
|
+
this.requestUri,
|
399
|
+
this.deviceId,
|
400
|
+
)
|
401
|
+
|
402
|
+
// Any error thrown in this block will be transformed into an
|
403
|
+
// AccessDeniedError.
|
404
|
+
try {
|
405
|
+
const { account, authorizedClients } = await authenticate.call(
|
406
|
+
this,
|
407
|
+
req,
|
408
|
+
res,
|
409
|
+
)
|
410
|
+
|
411
|
+
const client = await server.clientManager.getClient(clientId)
|
412
|
+
|
413
|
+
const code = await server.requestManager.setAuthorized(
|
414
|
+
this.requestUri,
|
415
|
+
client,
|
416
|
+
account,
|
417
|
+
this.deviceId,
|
418
|
+
this.deviceMetadata,
|
419
|
+
)
|
420
|
+
|
421
|
+
const clientData = authorizedClients.get(clientId)
|
422
|
+
if (server.checkConsentRequired(parameters, clientData)) {
|
423
|
+
const scopes = new Set(clientData?.authorizedScopes)
|
424
|
+
|
425
|
+
// Add the newly accepted scopes to the authorized scopes
|
426
|
+
|
427
|
+
// @NOTE `oauthScopeSchema` ensures that `scope` contains no
|
428
|
+
// leading/trailing/duplicate spaces.
|
429
|
+
for (const s of parameters.scope?.split(' ') ?? []) scopes.add(s)
|
430
|
+
|
431
|
+
await server.accountManager.setAuthorizedClient(account, client, {
|
432
|
+
...clientData,
|
433
|
+
authorizedScopes: [...scopes],
|
434
|
+
})
|
435
|
+
}
|
436
|
+
|
437
|
+
const url = buildRedirectUrl(server.issuer, parameters, { code })
|
438
|
+
|
439
|
+
return { url }
|
440
|
+
} catch (err) {
|
441
|
+
// Since we have access to the parameters, we can re-throw an
|
442
|
+
// AccessDeniedError with the redirect_uri parameter.
|
443
|
+
throw AccessDeniedError.from(parameters, err, 'server_error')
|
444
|
+
}
|
445
|
+
} catch (err) {
|
446
|
+
// If any error happened (unauthenticated, invalid request, etc.),
|
447
|
+
// lets make sure the request can no longer be used.
|
448
|
+
try {
|
449
|
+
await server.requestManager.delete(this.requestUri)
|
450
|
+
} catch (err) {
|
451
|
+
onError?.(req, res, err, 'Failed to delete request')
|
452
|
+
}
|
453
|
+
|
454
|
+
if (err instanceof AccessDeniedError && err.parameters.redirect_uri) {
|
455
|
+
// Prefer logging the cause
|
456
|
+
onError?.(req, res, err.cause ?? err, 'Authorization failed')
|
457
|
+
|
458
|
+
const url = buildRedirectUrl(
|
459
|
+
server.issuer,
|
460
|
+
err.parameters,
|
461
|
+
err.toJSON(),
|
462
|
+
)
|
463
|
+
|
464
|
+
return { url }
|
465
|
+
}
|
466
|
+
|
467
|
+
throw err
|
468
|
+
}
|
469
|
+
},
|
470
|
+
}),
|
471
|
+
)
|
472
|
+
|
473
|
+
router.use(
|
474
|
+
apiRoute({
|
475
|
+
method: 'POST',
|
476
|
+
endpoint: '/reject',
|
477
|
+
schema: z.object({}).strict(),
|
478
|
+
rotateDeviceCookies: true,
|
479
|
+
async handler(req, res) {
|
480
|
+
const { requestUri } = this
|
481
|
+
if (!requestUri) {
|
482
|
+
throw new InvalidRequestError(
|
483
|
+
'This endpoint can only be used in the context of an OAuth request',
|
484
|
+
)
|
485
|
+
}
|
486
|
+
|
487
|
+
// Once this endpoint is called, the request will definitely be
|
488
|
+
// rejected.
|
489
|
+
try {
|
490
|
+
// No need to authenticate the user here as they are not authorizing a
|
491
|
+
// particular account (CSRF protection is enough).
|
492
|
+
|
493
|
+
// @NOTE that the client could *technically* trigger this endpoint while
|
494
|
+
// the user is on the authorize page by forging the request (because the
|
495
|
+
// client knows the RequestURI from PAR and has all the info needed to
|
496
|
+
// forge the request, including CSRF). This cannot be used as DoS attack
|
497
|
+
// as the request ID is not guessable and would only result in a bad UX
|
498
|
+
// for misbehaving clients, only for the users of those clients.
|
499
|
+
|
500
|
+
const { parameters } = await server.requestManager.get(
|
501
|
+
requestUri,
|
502
|
+
this.deviceId,
|
503
|
+
)
|
504
|
+
|
505
|
+
const url = buildRedirectUrl(server.issuer, parameters, {
|
506
|
+
error: 'access_denied',
|
507
|
+
error_description: 'The user rejected the request',
|
508
|
+
})
|
509
|
+
|
510
|
+
return { url }
|
511
|
+
} catch (err) {
|
512
|
+
if (err instanceof AccessDeniedError && err.parameters.redirect_uri) {
|
513
|
+
// Prefer logging the cause
|
514
|
+
onError?.(req, res, err.cause ?? err, 'Authorization failed')
|
515
|
+
|
516
|
+
const url = buildRedirectUrl(
|
517
|
+
server.issuer,
|
518
|
+
err.parameters,
|
519
|
+
err.toJSON(),
|
520
|
+
)
|
521
|
+
|
522
|
+
return { url }
|
523
|
+
}
|
524
|
+
|
525
|
+
throw err
|
526
|
+
} finally {
|
527
|
+
await server.requestManager.delete(requestUri).catch((err) => {
|
528
|
+
onError?.(req, res, err, 'Failed to delete request')
|
529
|
+
})
|
530
|
+
}
|
531
|
+
},
|
532
|
+
}),
|
533
|
+
)
|
534
|
+
|
535
|
+
return router.buildMiddleware()
|
536
|
+
|
537
|
+
async function authenticate(
|
538
|
+
this: ApiContext<void, { sub: Sub }>,
|
539
|
+
req: Req,
|
540
|
+
res: Res,
|
541
|
+
) {
|
542
|
+
const authorization = req.headers.authorization?.split(' ')
|
543
|
+
if (authorization?.[0].toLowerCase() === 'bearer') {
|
544
|
+
try {
|
545
|
+
// If there is an authorization header, verify that the ephemeral token it
|
546
|
+
// contains is a jwt bound to the right [sub, device, request].
|
547
|
+
const ephemeralToken = signedJwtSchema.parse(authorization[1])
|
548
|
+
const { payload } =
|
549
|
+
await server.signer.verifyEphemeralToken(ephemeralToken)
|
550
|
+
|
551
|
+
if (
|
552
|
+
payload.sub === this.input.sub &&
|
553
|
+
payload.deviceId === this.deviceId &&
|
554
|
+
payload.requestUri === this.requestUri
|
555
|
+
) {
|
556
|
+
return await server.accountManager.getAccount(payload.sub)
|
557
|
+
}
|
558
|
+
} catch (err) {
|
559
|
+
onError?.(req, res, err, 'Failed to authenticate ephemeral token')
|
560
|
+
// Fall back to session based authentication
|
561
|
+
}
|
562
|
+
}
|
563
|
+
|
564
|
+
try {
|
565
|
+
// Ensures the "sub" has an active session on the device
|
566
|
+
const deviceAccount = await server.accountManager.getDeviceAccount(
|
567
|
+
this.deviceId,
|
568
|
+
this.input.sub,
|
569
|
+
)
|
570
|
+
|
571
|
+
// The session exists but was created too long ago
|
572
|
+
if (server.checkLoginRequired(deviceAccount)) {
|
573
|
+
throw new InvalidRequestError('Login required')
|
574
|
+
}
|
575
|
+
|
576
|
+
return deviceAccount
|
577
|
+
} catch (err) {
|
578
|
+
throw new WWWAuthenticateError(
|
579
|
+
'unauthorized',
|
580
|
+
`User ${this.input.sub} not authenticated on this device`,
|
581
|
+
{ Bearer: {} },
|
582
|
+
err,
|
583
|
+
)
|
584
|
+
}
|
585
|
+
}
|
586
|
+
|
587
|
+
type ApiContext<T extends object | void, I = void> = SubCtx<
|
588
|
+
T,
|
589
|
+
{
|
590
|
+
deviceId: DeviceId
|
591
|
+
deviceMetadata: RequestMetadata
|
592
|
+
|
593
|
+
/**
|
594
|
+
* The parsed input data (json payload if "POST", query params if "GET").
|
595
|
+
*/
|
596
|
+
input: I
|
597
|
+
|
598
|
+
/**
|
599
|
+
* When defined, the request originated from the authorize page.
|
600
|
+
*/
|
601
|
+
requestUri?: RequestUri
|
602
|
+
}
|
603
|
+
>
|
604
|
+
|
605
|
+
type InferValidation<S extends void | z.ZodTypeAny> = S extends z.ZodTypeAny
|
606
|
+
? z.infer<S>
|
607
|
+
: void
|
608
|
+
|
609
|
+
/**
|
610
|
+
* The main purpose of this function is to ensure that the endpoint
|
611
|
+
* implementation matches its type definition from {@link ApiEndpoints}.
|
612
|
+
* @private
|
613
|
+
*/
|
614
|
+
function apiRoute<
|
615
|
+
C extends RouterCtx<Ctx>,
|
616
|
+
M extends 'GET' | 'POST',
|
617
|
+
E extends `/${string}` &
|
618
|
+
// Extract all the endpoint path that match the method (allows for
|
619
|
+
// auto-complete & better error reporting)
|
620
|
+
{
|
621
|
+
[E in keyof ApiEndpoints]: ApiEndpoints[E] extends { method: M }
|
622
|
+
? E
|
623
|
+
: never
|
624
|
+
}[keyof ApiEndpoints],
|
625
|
+
S extends // A schema that validates the POST input or GET params
|
626
|
+
ApiEndpoints[E] extends { method: 'POST'; input: infer I }
|
627
|
+
? z.ZodType<I>
|
628
|
+
: ApiEndpoints[E] extends { method: 'GET'; params: infer P }
|
629
|
+
? z.ZodType<P>
|
630
|
+
: void,
|
631
|
+
>(options: {
|
632
|
+
method: M
|
633
|
+
endpoint: E
|
634
|
+
schema: S
|
635
|
+
rotateDeviceCookies?: boolean
|
636
|
+
handler: (
|
637
|
+
this: ApiContext<RouteCtx<C>, InferValidation<S>>,
|
638
|
+
req: Req,
|
639
|
+
res: Res,
|
640
|
+
) => Awaitable<ApiEndpoints[E]['output']>
|
641
|
+
}): Middleware<C, Req, Res> {
|
642
|
+
return createRoute(
|
643
|
+
options.method,
|
644
|
+
`${API_ENDPOINT_PREFIX}${options.endpoint}`,
|
645
|
+
apiMiddleware(options),
|
646
|
+
)
|
647
|
+
}
|
648
|
+
|
649
|
+
function apiMiddleware<C extends RouterCtx, S extends void | z.ZodTypeAny>({
|
650
|
+
method,
|
651
|
+
schema,
|
652
|
+
rotateDeviceCookies,
|
653
|
+
handler,
|
654
|
+
}: {
|
655
|
+
method: 'GET' | 'POST'
|
656
|
+
schema: S
|
657
|
+
rotateDeviceCookies?: boolean
|
658
|
+
handler: (
|
659
|
+
this: ApiContext<C, InferValidation<S>>,
|
660
|
+
req: Req,
|
661
|
+
res: Res,
|
662
|
+
) => unknown
|
663
|
+
}): Middleware<C, Req, Res> {
|
664
|
+
const parseInput: (this: C, req: Req) => Promise<InferValidation<S>> =
|
665
|
+
schema == null // No schema means endpoint doesn't accept any input
|
666
|
+
? async function (req) {
|
667
|
+
req.resume() // Flush body
|
668
|
+
return undefined
|
669
|
+
}
|
670
|
+
: method === 'POST'
|
671
|
+
? async function (req) {
|
672
|
+
const body = await parseHttpRequest(req, ['json'])
|
673
|
+
return schema.parseAsync(body, { path: ['body'] })
|
674
|
+
}
|
675
|
+
: async function (req) {
|
676
|
+
// @NOTE This should not be necessary with GET requests
|
677
|
+
req.resume().once('error', (_err) => {
|
678
|
+
// Ignore errors when flushing the request body
|
679
|
+
// (e.g. client closed connection)
|
680
|
+
})
|
681
|
+
|
682
|
+
const query = Object.fromEntries(this.url.searchParams)
|
683
|
+
return schema.parseAsync(query, { path: ['query'] })
|
684
|
+
}
|
685
|
+
|
686
|
+
return jsonHandler<C, Req, Res>(async function (req, res) {
|
687
|
+
try {
|
688
|
+
// Prevent caching of API routes
|
689
|
+
res.setHeader('Cache-Control', 'no-store')
|
690
|
+
res.setHeader('Pragma', 'no-cache')
|
691
|
+
|
692
|
+
// Prevent CORS requests
|
693
|
+
validateFetchMode(req, ['same-origin'])
|
694
|
+
validateFetchSite(req, ['same-origin'])
|
695
|
+
validateOrigin(req, issuerOrigin)
|
696
|
+
const referrer = validateReferrer(req, { origin: issuerOrigin })
|
697
|
+
|
698
|
+
// Ensure we are one the right page
|
699
|
+
if (
|
700
|
+
// trailing slashes are not allowed
|
701
|
+
referrer.pathname !== '/oauth/authorize' &&
|
702
|
+
referrer.pathname !== '/account' &&
|
703
|
+
!referrer.pathname.startsWith(`/account/`)
|
704
|
+
) {
|
705
|
+
throw createHttpError(400, `Invalid referrer ${referrer}`)
|
706
|
+
}
|
707
|
+
|
708
|
+
// Check if the request originated from the authorize page
|
709
|
+
const requestUri =
|
710
|
+
referrer.pathname === '/oauth/authorize'
|
711
|
+
? await requestUriSchema.parseAsync(
|
712
|
+
referrer.searchParams.get('request_uri'),
|
713
|
+
)
|
714
|
+
: undefined
|
715
|
+
|
716
|
+
// Validate CSRF token
|
717
|
+
await validateCsrfToken(req, res)
|
718
|
+
|
719
|
+
// Parse and validate the input data
|
720
|
+
const input = await parseInput.call(this, req)
|
721
|
+
|
722
|
+
// Load session data, rotating the session cookie if needed
|
723
|
+
const { deviceId, deviceMetadata } = await server.deviceManager.load(
|
724
|
+
req,
|
725
|
+
res,
|
726
|
+
rotateDeviceCookies,
|
727
|
+
)
|
728
|
+
|
729
|
+
const context = subCtx(this, {
|
730
|
+
input,
|
731
|
+
requestUri,
|
732
|
+
deviceId,
|
733
|
+
deviceMetadata,
|
734
|
+
})
|
735
|
+
|
736
|
+
// Generate the API response
|
737
|
+
const payload = await handler.call(context, req, res)
|
738
|
+
|
739
|
+
return { payload, status: 200 }
|
740
|
+
} catch (err) {
|
741
|
+
onError?.(req, res, err, 'Failed to handle API request')
|
742
|
+
|
743
|
+
// @TODO Rework the API error responses (relying on codes)
|
744
|
+
const payload = buildErrorPayload(err)
|
745
|
+
const status = buildErrorStatus(err)
|
746
|
+
|
747
|
+
return { payload, status }
|
748
|
+
}
|
749
|
+
})
|
750
|
+
}
|
751
|
+
}
|
752
|
+
|
753
|
+
function buildRedirectUrl(
|
754
|
+
iss: string,
|
755
|
+
parameters: OAuthAuthorizationRequestParameters,
|
756
|
+
redirect: AuthorizationRedirectParameters,
|
757
|
+
): string {
|
758
|
+
const url = new URL('/oauth/authorize/redirect', iss)
|
759
|
+
|
760
|
+
url.searchParams.set('redirect_mode', buildRedirectMode(parameters))
|
761
|
+
url.searchParams.set('redirect_uri', buildRedirectUri(parameters))
|
762
|
+
|
763
|
+
for (const [key, value] of buildRedirectParams(iss, parameters, redirect)) {
|
764
|
+
url.searchParams.set(key, value)
|
765
|
+
}
|
766
|
+
|
767
|
+
return url.href
|
768
|
+
}
|
769
|
+
|
770
|
+
export function parseRedirectUrl(url: URL): OAuthRedirectOptions {
|
771
|
+
if (url.pathname !== '/oauth/authorize/redirect') {
|
772
|
+
throw new InvalidRequestError(
|
773
|
+
`Invalid redirect URL: ${url.pathname} is not a valid path`,
|
774
|
+
)
|
775
|
+
}
|
776
|
+
|
777
|
+
const params: [OAuthRedirectQueryParameter, string][] = []
|
778
|
+
|
779
|
+
const state = url.searchParams.get('state')
|
780
|
+
if (state) params.push(['state', state])
|
781
|
+
|
782
|
+
const iss = url.searchParams.get('iss')
|
783
|
+
if (iss) params.push(['iss', iss])
|
784
|
+
|
785
|
+
if (url.searchParams.has('code')) {
|
786
|
+
for (const key of SUCCESS_REDIRECT_KEYS) {
|
787
|
+
const value = url.searchParams.get(key)
|
788
|
+
if (value != null) params.push([key, value])
|
789
|
+
}
|
790
|
+
} else if (url.searchParams.has('error')) {
|
791
|
+
for (const key of ERROR_REDIRECT_KEYS) {
|
792
|
+
const value = url.searchParams.get(key)
|
793
|
+
if (value != null) params.push([key, value])
|
794
|
+
}
|
795
|
+
} else {
|
796
|
+
throw new InvalidRequestError(
|
797
|
+
'Invalid redirect URL: neither code nor error found',
|
798
|
+
)
|
799
|
+
}
|
800
|
+
|
801
|
+
try {
|
802
|
+
const mode: OAuthResponseMode = oauthResponseModeSchema.parse(
|
803
|
+
url.searchParams.get('redirect_mode'),
|
804
|
+
)
|
805
|
+
|
806
|
+
const redirectUri: OAuthRedirectUri = oauthRedirectUriSchema.parse(
|
807
|
+
url.searchParams.get('redirect_uri'),
|
808
|
+
)
|
809
|
+
|
810
|
+
return { mode, redirectUri, params }
|
811
|
+
} catch (err) {
|
812
|
+
throw InvalidRequestError.from(err, 'Invalid redirect URL')
|
813
|
+
}
|
814
|
+
}
|