@atproto/oauth-provider 0.1.0
Sign up to get free protection for your applications and to get access to all the features.
- package/.postcssrc.yml +3 -0
- package/CHANGELOG.md +19 -0
- package/LICENSE.txt +7 -0
- package/dist/access-token/access-token-type.d.ts +6 -0
- package/dist/access-token/access-token-type.d.ts.map +1 -0
- package/dist/access-token/access-token-type.js +10 -0
- package/dist/access-token/access-token-type.js.map +1 -0
- package/dist/account/account-manager.d.ts +14 -0
- package/dist/account/account-manager.d.ts.map +1 -0
- package/dist/account/account-manager.js +39 -0
- package/dist/account/account-manager.js.map +1 -0
- package/dist/account/account-store.d.ts +39 -0
- package/dist/account/account-store.d.ts.map +1 -0
- package/dist/account/account-store.js +19 -0
- package/dist/account/account-store.js.map +1 -0
- package/dist/account/account.d.ts +8 -0
- package/dist/account/account.d.ts.map +1 -0
- package/dist/account/account.js +3 -0
- package/dist/account/account.js.map +1 -0
- package/dist/assets/app/bundle-manifest.json +22 -0
- package/dist/assets/app/main.css +3 -0
- package/dist/assets/app/main.js +20 -0
- package/dist/assets/app/main.js.map +1 -0
- package/dist/assets/asset.d.ts +9 -0
- package/dist/assets/asset.d.ts.map +1 -0
- package/dist/assets/asset.js +3 -0
- package/dist/assets/asset.js.map +1 -0
- package/dist/assets/assets-middleware.d.ts +2 -0
- package/dist/assets/assets-middleware.d.ts.map +1 -0
- package/dist/assets/assets-middleware.js +30 -0
- package/dist/assets/assets-middleware.js.map +1 -0
- package/dist/assets/index.d.ts +4 -0
- package/dist/assets/index.d.ts.map +1 -0
- package/dist/assets/index.js +65 -0
- package/dist/assets/index.js.map +1 -0
- package/dist/client/client-auth.d.ts +13 -0
- package/dist/client/client-auth.d.ts.map +1 -0
- package/dist/client/client-auth.js +35 -0
- package/dist/client/client-auth.js.map +1 -0
- package/dist/client/client-data.d.ts +8 -0
- package/dist/client/client-data.d.ts.map +1 -0
- package/dist/client/client-data.js +3 -0
- package/dist/client/client-data.js.map +1 -0
- package/dist/client/client-id.d.ts +4 -0
- package/dist/client/client-id.d.ts.map +1 -0
- package/dist/client/client-id.js +6 -0
- package/dist/client/client-id.js.map +1 -0
- package/dist/client/client-info.d.ts +13 -0
- package/dist/client/client-info.d.ts.map +1 -0
- package/dist/client/client-info.js +3 -0
- package/dist/client/client-info.js.map +1 -0
- package/dist/client/client-manager.d.ts +38 -0
- package/dist/client/client-manager.d.ts.map +1 -0
- package/dist/client/client-manager.js +534 -0
- package/dist/client/client-manager.js.map +1 -0
- package/dist/client/client-store.d.ts +13 -0
- package/dist/client/client-store.d.ts.map +1 -0
- package/dist/client/client-store.js +39 -0
- package/dist/client/client-store.js.map +1 -0
- package/dist/client/client-utils.d.ts +6 -0
- package/dist/client/client-utils.d.ts.map +1 -0
- package/dist/client/client-utils.js +40 -0
- package/dist/client/client-utils.js.map +1 -0
- package/dist/client/client.d.ts +41 -0
- package/dist/client/client.d.ts.map +1 -0
- package/dist/client/client.js +163 -0
- package/dist/client/client.js.map +1 -0
- package/dist/constants.d.ts +42 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +53 -0
- package/dist/constants.js.map +1 -0
- package/dist/device/device-data.d.ts +20 -0
- package/dist/device/device-data.d.ts.map +1 -0
- package/dist/device/device-data.js +11 -0
- package/dist/device/device-data.js.map +1 -0
- package/dist/device/device-details.d.ts +17 -0
- package/dist/device/device-details.d.ts.map +1 -0
- package/dist/device/device-details.js +34 -0
- package/dist/device/device-details.js.map +1 -0
- package/dist/device/device-id.d.ts +6 -0
- package/dist/device/device-id.d.ts.map +1 -0
- package/dist/device/device-id.js +18 -0
- package/dist/device/device-id.js.map +1 -0
- package/dist/device/device-manager.d.ts +88 -0
- package/dist/device/device-manager.d.ts.map +1 -0
- package/dist/device/device-manager.js +206 -0
- package/dist/device/device-manager.js.map +1 -0
- package/dist/device/device-store.d.ts +15 -0
- package/dist/device/device-store.d.ts.map +1 -0
- package/dist/device/device-store.js +36 -0
- package/dist/device/device-store.js.map +1 -0
- package/dist/device/session-id.d.ts +6 -0
- package/dist/device/session-id.d.ts.map +1 -0
- package/dist/device/session-id.js +18 -0
- package/dist/device/session-id.js.map +1 -0
- package/dist/dpop/dpop-manager.d.ts +33 -0
- package/dist/dpop/dpop-manager.d.ts.map +1 -0
- package/dist/dpop/dpop-manager.js +115 -0
- package/dist/dpop/dpop-manager.js.map +1 -0
- package/dist/dpop/dpop-nonce.d.ts +13 -0
- package/dist/dpop/dpop-nonce.d.ts.map +1 -0
- package/dist/dpop/dpop-nonce.js +94 -0
- package/dist/dpop/dpop-nonce.js.map +1 -0
- package/dist/errors/access-denied-error.d.ts +8 -0
- package/dist/errors/access-denied-error.d.ts.map +1 -0
- package/dist/errors/access-denied-error.js +21 -0
- package/dist/errors/access-denied-error.js.map +1 -0
- package/dist/errors/account-selection-required-error.d.ts +6 -0
- package/dist/errors/account-selection-required-error.d.ts.map +1 -0
- package/dist/errors/account-selection-required-error.js +11 -0
- package/dist/errors/account-selection-required-error.js.map +1 -0
- package/dist/errors/consent-required-error.d.ts +6 -0
- package/dist/errors/consent-required-error.d.ts.map +1 -0
- package/dist/errors/consent-required-error.js +11 -0
- package/dist/errors/consent-required-error.js.map +1 -0
- package/dist/errors/invalid-authorization-details-error.d.ts +20 -0
- package/dist/errors/invalid-authorization-details-error.d.ts.map +1 -0
- package/dist/errors/invalid-authorization-details-error.js +26 -0
- package/dist/errors/invalid-authorization-details-error.js.map +1 -0
- package/dist/errors/invalid-client-error.d.ts +18 -0
- package/dist/errors/invalid-client-error.d.ts.map +1 -0
- package/dist/errors/invalid-client-error.js +24 -0
- package/dist/errors/invalid-client-error.js.map +1 -0
- package/dist/errors/invalid-client-id-error.d.ts +13 -0
- package/dist/errors/invalid-client-id-error.d.ts.map +1 -0
- package/dist/errors/invalid-client-id-error.js +25 -0
- package/dist/errors/invalid-client-id-error.js.map +1 -0
- package/dist/errors/invalid-client-metadata-error.d.ts +13 -0
- package/dist/errors/invalid-client-metadata-error.d.ts.map +1 -0
- package/dist/errors/invalid-client-metadata-error.js +23 -0
- package/dist/errors/invalid-client-metadata-error.js.map +1 -0
- package/dist/errors/invalid-dpop-key-binding-error.d.ts +12 -0
- package/dist/errors/invalid-dpop-key-binding-error.d.ts.map +1 -0
- package/dist/errors/invalid-dpop-key-binding-error.js +20 -0
- package/dist/errors/invalid-dpop-key-binding-error.js.map +1 -0
- package/dist/errors/invalid-dpop-proof-error.d.ts +5 -0
- package/dist/errors/invalid-dpop-proof-error.d.ts.map +1 -0
- package/dist/errors/invalid-dpop-proof-error.js +12 -0
- package/dist/errors/invalid-dpop-proof-error.js.map +1 -0
- package/dist/errors/invalid-grant-error.d.ts +14 -0
- package/dist/errors/invalid-grant-error.d.ts.map +1 -0
- package/dist/errors/invalid-grant-error.js +20 -0
- package/dist/errors/invalid-grant-error.js.map +1 -0
- package/dist/errors/invalid-parameters-error.d.ts +6 -0
- package/dist/errors/invalid-parameters-error.d.ts.map +1 -0
- package/dist/errors/invalid-parameters-error.js +11 -0
- package/dist/errors/invalid-parameters-error.js.map +1 -0
- package/dist/errors/invalid-redirect-uri-error.d.ts +11 -0
- package/dist/errors/invalid-redirect-uri-error.d.ts.map +1 -0
- package/dist/errors/invalid-redirect-uri-error.js +21 -0
- package/dist/errors/invalid-redirect-uri-error.js.map +1 -0
- package/dist/errors/invalid-request-error.d.ts +28 -0
- package/dist/errors/invalid-request-error.d.ts.map +1 -0
- package/dist/errors/invalid-request-error.js +34 -0
- package/dist/errors/invalid-request-error.js.map +1 -0
- package/dist/errors/invalid-token-error.d.ts +16 -0
- package/dist/errors/invalid-token-error.d.ts.map +1 -0
- package/dist/errors/invalid-token-error.js +45 -0
- package/dist/errors/invalid-token-error.js.map +1 -0
- package/dist/errors/login-required-error.d.ts +6 -0
- package/dist/errors/login-required-error.d.ts.map +1 -0
- package/dist/errors/login-required-error.js +11 -0
- package/dist/errors/login-required-error.js.map +1 -0
- package/dist/errors/oauth-error.d.ts +13 -0
- package/dist/errors/oauth-error.d.ts.map +1 -0
- package/dist/errors/oauth-error.js +29 -0
- package/dist/errors/oauth-error.js.map +1 -0
- package/dist/errors/unauthorized-client-error.d.ts +18 -0
- package/dist/errors/unauthorized-client-error.d.ts.map +1 -0
- package/dist/errors/unauthorized-client-error.js +24 -0
- package/dist/errors/unauthorized-client-error.js.map +1 -0
- package/dist/errors/use-dpop-nonce-error.d.ts +18 -0
- package/dist/errors/use-dpop-nonce-error.d.ts.map +1 -0
- package/dist/errors/use-dpop-nonce-error.js +27 -0
- package/dist/errors/use-dpop-nonce-error.js.map +1 -0
- package/dist/errors/www-authenticate-error.d.ts +9 -0
- package/dist/errors/www-authenticate-error.d.ts.map +1 -0
- package/dist/errors/www-authenticate-error.js +46 -0
- package/dist/errors/www-authenticate-error.js.map +1 -0
- package/dist/index.d.ts +14 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +31 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/html/build-document.d.ts +32 -0
- package/dist/lib/html/build-document.d.ts.map +1 -0
- package/dist/lib/html/build-document.js +61 -0
- package/dist/lib/html/build-document.js.map +1 -0
- package/dist/lib/html/escapers.d.ts +9 -0
- package/dist/lib/html/escapers.d.ts.map +1 -0
- package/dist/lib/html/escapers.js +66 -0
- package/dist/lib/html/escapers.js.map +1 -0
- package/dist/lib/html/html.d.ts +13 -0
- package/dist/lib/html/html.d.ts.map +1 -0
- package/dist/lib/html/html.js +53 -0
- package/dist/lib/html/html.js.map +1 -0
- package/dist/lib/html/index.d.ts +4 -0
- package/dist/lib/html/index.d.ts.map +1 -0
- package/dist/lib/html/index.js +21 -0
- package/dist/lib/html/index.js.map +1 -0
- package/dist/lib/html/tags.d.ts +34 -0
- package/dist/lib/html/tags.d.ts.map +1 -0
- package/dist/lib/html/tags.js +47 -0
- package/dist/lib/html/tags.js.map +1 -0
- package/dist/lib/html/util.d.ts +4 -0
- package/dist/lib/html/util.d.ts.map +1 -0
- package/dist/lib/html/util.js +20 -0
- package/dist/lib/html/util.js.map +1 -0
- package/dist/lib/http/accept.d.ts +29 -0
- package/dist/lib/http/accept.d.ts.map +1 -0
- package/dist/lib/http/accept.js +67 -0
- package/dist/lib/http/accept.js.map +1 -0
- package/dist/lib/http/context.d.ts +5 -0
- package/dist/lib/http/context.d.ts.map +1 -0
- package/dist/lib/http/context.js +10 -0
- package/dist/lib/http/context.js.map +1 -0
- package/dist/lib/http/index.d.ts +10 -0
- package/dist/lib/http/index.d.ts.map +1 -0
- package/dist/lib/http/index.js +26 -0
- package/dist/lib/http/index.js.map +1 -0
- package/dist/lib/http/method.d.ts +6 -0
- package/dist/lib/http/method.d.ts.map +1 -0
- package/dist/lib/http/method.js +19 -0
- package/dist/lib/http/method.js.map +1 -0
- package/dist/lib/http/middleware.d.ts +18 -0
- package/dist/lib/http/middleware.d.ts.map +1 -0
- package/dist/lib/http/middleware.js +118 -0
- package/dist/lib/http/middleware.js.map +1 -0
- package/dist/lib/http/parser.d.ts +33 -0
- package/dist/lib/http/parser.d.ts.map +1 -0
- package/dist/lib/http/parser.js +48 -0
- package/dist/lib/http/parser.js.map +1 -0
- package/dist/lib/http/path.d.ts +9 -0
- package/dist/lib/http/path.d.ts.map +1 -0
- package/dist/lib/http/path.js +54 -0
- package/dist/lib/http/path.js.map +1 -0
- package/dist/lib/http/request.d.ts +33 -0
- package/dist/lib/http/request.d.ts.map +1 -0
- package/dist/lib/http/request.js +86 -0
- package/dist/lib/http/request.js.map +1 -0
- package/dist/lib/http/response.d.ts +13 -0
- package/dist/lib/http/response.d.ts.map +1 -0
- package/dist/lib/http/response.js +98 -0
- package/dist/lib/http/response.js.map +1 -0
- package/dist/lib/http/route.d.ts +25 -0
- package/dist/lib/http/route.d.ts.map +1 -0
- package/dist/lib/http/route.js +39 -0
- package/dist/lib/http/route.js.map +1 -0
- package/dist/lib/http/router.d.ts +32 -0
- package/dist/lib/http/router.d.ts.map +1 -0
- package/dist/lib/http/router.js +74 -0
- package/dist/lib/http/router.js.map +1 -0
- package/dist/lib/http/stream.d.ts +13 -0
- package/dist/lib/http/stream.d.ts.map +1 -0
- package/dist/lib/http/stream.js +46 -0
- package/dist/lib/http/stream.js.map +1 -0
- package/dist/lib/http/types.d.ts +7 -0
- package/dist/lib/http/types.d.ts.map +1 -0
- package/dist/lib/http/types.js +3 -0
- package/dist/lib/http/types.js.map +1 -0
- package/dist/lib/http/url.d.ts +8 -0
- package/dist/lib/http/url.d.ts.map +1 -0
- package/dist/lib/http/url.js +22 -0
- package/dist/lib/http/url.js.map +1 -0
- package/dist/lib/redis.d.ts +5 -0
- package/dist/lib/redis.d.ts.map +1 -0
- package/dist/lib/redis.js +22 -0
- package/dist/lib/redis.js.map +1 -0
- package/dist/lib/util/authorization-header.d.ts +4 -0
- package/dist/lib/util/authorization-header.d.ts.map +1 -0
- package/dist/lib/util/authorization-header.js +23 -0
- package/dist/lib/util/authorization-header.js.map +1 -0
- package/dist/lib/util/cast.d.ts +2 -0
- package/dist/lib/util/cast.d.ts.map +1 -0
- package/dist/lib/util/cast.js +10 -0
- package/dist/lib/util/cast.js.map +1 -0
- package/dist/lib/util/crypto.d.ts +3 -0
- package/dist/lib/util/crypto.d.ts.map +1 -0
- package/dist/lib/util/crypto.js +29 -0
- package/dist/lib/util/crypto.js.map +1 -0
- package/dist/lib/util/date.d.ts +3 -0
- package/dist/lib/util/date.d.ts.map +1 -0
- package/dist/lib/util/date.js +12 -0
- package/dist/lib/util/date.js.map +1 -0
- package/dist/lib/util/hostname.d.ts +6 -0
- package/dist/lib/util/hostname.d.ts.map +1 -0
- package/dist/lib/util/hostname.js +24 -0
- package/dist/lib/util/hostname.js.map +1 -0
- package/dist/lib/util/redirect-uri.d.ts +7 -0
- package/dist/lib/util/redirect-uri.d.ts.map +1 -0
- package/dist/lib/util/redirect-uri.js +44 -0
- package/dist/lib/util/redirect-uri.js.map +1 -0
- package/dist/lib/util/time.d.ts +6 -0
- package/dist/lib/util/time.d.ts.map +1 -0
- package/dist/lib/util/time.js +28 -0
- package/dist/lib/util/time.js.map +1 -0
- package/dist/lib/util/type.d.ts +6 -0
- package/dist/lib/util/type.d.ts.map +1 -0
- package/dist/lib/util/type.js +3 -0
- package/dist/lib/util/type.js.map +1 -0
- package/dist/lib/util/well-known.d.ts +3 -0
- package/dist/lib/util/well-known.d.ts.map +1 -0
- package/dist/lib/util/well-known.js +11 -0
- package/dist/lib/util/well-known.js.map +1 -0
- package/dist/metadata/build-metadata.d.ts +14 -0
- package/dist/metadata/build-metadata.d.ts.map +1 -0
- package/dist/metadata/build-metadata.js +132 -0
- package/dist/metadata/build-metadata.js.map +1 -0
- package/dist/oauth-client.d.ts +4 -0
- package/dist/oauth-client.d.ts.map +1 -0
- package/dist/oauth-client.js +19 -0
- package/dist/oauth-client.js.map +1 -0
- package/dist/oauth-dpop.d.ts +3 -0
- package/dist/oauth-dpop.d.ts.map +1 -0
- package/dist/oauth-dpop.js +19 -0
- package/dist/oauth-dpop.js.map +1 -0
- package/dist/oauth-errors.d.ts +20 -0
- package/dist/oauth-errors.d.ts.map +1 -0
- package/dist/oauth-errors.js +43 -0
- package/dist/oauth-errors.js.map +1 -0
- package/dist/oauth-hooks.d.ts +42 -0
- package/dist/oauth-hooks.d.ts.map +1 -0
- package/dist/oauth-hooks.js +3 -0
- package/dist/oauth-hooks.js.map +1 -0
- package/dist/oauth-provider.d.ts +179 -0
- package/dist/oauth-provider.d.ts.map +1 -0
- package/dist/oauth-provider.js +748 -0
- package/dist/oauth-provider.js.map +1 -0
- package/dist/oauth-store.d.ts +11 -0
- package/dist/oauth-store.d.ts.map +1 -0
- package/dist/oauth-store.js +27 -0
- package/dist/oauth-store.js.map +1 -0
- package/dist/oauth-verifier.d.ts +66 -0
- package/dist/oauth-verifier.d.ts.map +1 -0
- package/dist/oauth-verifier.js +94 -0
- package/dist/oauth-verifier.js.map +1 -0
- package/dist/oidc/claims.d.ts +16 -0
- package/dist/oidc/claims.d.ts.map +1 -0
- package/dist/oidc/claims.js +29 -0
- package/dist/oidc/claims.js.map +1 -0
- package/dist/oidc/sub.d.ts +4 -0
- package/dist/oidc/sub.d.ts.map +1 -0
- package/dist/oidc/sub.js +6 -0
- package/dist/oidc/sub.js.map +1 -0
- package/dist/oidc/userinfo.d.ts +7 -0
- package/dist/oidc/userinfo.d.ts.map +1 -0
- package/dist/oidc/userinfo.js +3 -0
- package/dist/oidc/userinfo.js.map +1 -0
- package/dist/output/build-error-payload.d.ts +6 -0
- package/dist/output/build-error-payload.d.ts.map +1 -0
- package/dist/output/build-error-payload.js +108 -0
- package/dist/output/build-error-payload.js.map +1 -0
- package/dist/output/customization.d.ts +37 -0
- package/dist/output/customization.d.ts.map +1 -0
- package/dist/output/customization.js +62 -0
- package/dist/output/customization.js.map +1 -0
- package/dist/output/send-authorize-page.d.ts +43 -0
- package/dist/output/send-authorize-page.d.ts.map +1 -0
- package/dist/output/send-authorize-page.js +49 -0
- package/dist/output/send-authorize-page.js.map +1 -0
- package/dist/output/send-authorize-redirect.d.ts +25 -0
- package/dist/output/send-authorize-redirect.d.ts.map +1 -0
- package/dist/output/send-authorize-redirect.js +72 -0
- package/dist/output/send-authorize-redirect.js.map +1 -0
- package/dist/output/send-error-page.d.ts +5 -0
- package/dist/output/send-error-page.d.ts.map +1 -0
- package/dist/output/send-error-page.js +31 -0
- package/dist/output/send-error-page.js.map +1 -0
- package/dist/output/send-web-page.d.ts +8 -0
- package/dist/output/send-web-page.d.ts.map +1 -0
- package/dist/output/send-web-page.js +48 -0
- package/dist/output/send-web-page.js.map +1 -0
- package/dist/parameters/claims-requested.d.ts +3 -0
- package/dist/parameters/claims-requested.d.ts.map +1 -0
- package/dist/parameters/claims-requested.js +77 -0
- package/dist/parameters/claims-requested.js.map +1 -0
- package/dist/parameters/oidc-payload.d.ts +31 -0
- package/dist/parameters/oidc-payload.d.ts.map +1 -0
- package/dist/parameters/oidc-payload.js +25 -0
- package/dist/parameters/oidc-payload.js.map +1 -0
- package/dist/replay/replay-manager.d.ts +10 -0
- package/dist/replay/replay-manager.d.ts.map +1 -0
- package/dist/replay/replay-manager.js +23 -0
- package/dist/replay/replay-manager.js.map +1 -0
- package/dist/replay/replay-store-memory.d.ts +11 -0
- package/dist/replay/replay-store-memory.d.ts.map +1 -0
- package/dist/replay/replay-store-memory.js +30 -0
- package/dist/replay/replay-store-memory.js.map +1 -0
- package/dist/replay/replay-store-redis.d.ts +16 -0
- package/dist/replay/replay-store-redis.d.ts.map +1 -0
- package/dist/replay/replay-store-redis.js +20 -0
- package/dist/replay/replay-store-redis.js.map +1 -0
- package/dist/replay/replay-store.d.ts +16 -0
- package/dist/replay/replay-store.d.ts.map +1 -0
- package/dist/replay/replay-store.js +22 -0
- package/dist/replay/replay-store.js.map +1 -0
- package/dist/request/code.d.ts +7 -0
- package/dist/request/code.d.ts.map +1 -0
- package/dist/request/code.js +20 -0
- package/dist/request/code.js.map +1 -0
- package/dist/request/request-data.d.ts +21 -0
- package/dist/request/request-data.d.ts.map +1 -0
- package/dist/request/request-data.js +6 -0
- package/dist/request/request-data.js.map +1 -0
- package/dist/request/request-id.d.ts +6 -0
- package/dist/request/request-id.d.ts.map +1 -0
- package/dist/request/request-id.js +18 -0
- package/dist/request/request-id.js.map +1 -0
- package/dist/request/request-info.d.ts +12 -0
- package/dist/request/request-info.d.ts.map +1 -0
- package/dist/request/request-info.js +3 -0
- package/dist/request/request-info.js.map +1 -0
- package/dist/request/request-manager.d.ts +40 -0
- package/dist/request/request-manager.d.ts.map +1 -0
- package/dist/request/request-manager.js +310 -0
- package/dist/request/request-manager.js.map +1 -0
- package/dist/request/request-store-memory.d.ts +16 -0
- package/dist/request/request-store-memory.d.ts.map +1 -0
- package/dist/request/request-store-memory.js +31 -0
- package/dist/request/request-store-memory.js.map +1 -0
- package/dist/request/request-store-redis.d.ts +24 -0
- package/dist/request/request-store-redis.d.ts.map +1 -0
- package/dist/request/request-store-redis.js +58 -0
- package/dist/request/request-store-redis.js.map +1 -0
- package/dist/request/request-store.d.ts +27 -0
- package/dist/request/request-store.d.ts.map +1 -0
- package/dist/request/request-store.js +37 -0
- package/dist/request/request-store.js.map +1 -0
- package/dist/request/request-uri.d.ts +8 -0
- package/dist/request/request-uri.d.ts.map +1 -0
- package/dist/request/request-uri.js +24 -0
- package/dist/request/request-uri.js.map +1 -0
- package/dist/request/types.d.ts +328 -0
- package/dist/request/types.d.ts.map +1 -0
- package/dist/request/types.js +27 -0
- package/dist/request/types.js.map +1 -0
- package/dist/signer/signed-token-payload.d.ts +1694 -0
- package/dist/signer/signed-token-payload.d.ts.map +1 -0
- package/dist/signer/signed-token-payload.js +32 -0
- package/dist/signer/signed-token-payload.js.map +1 -0
- package/dist/signer/signer.d.ts +193 -0
- package/dist/signer/signer.d.ts.map +1 -0
- package/dist/signer/signer.js +101 -0
- package/dist/signer/signer.js.map +1 -0
- package/dist/token/refresh-token.d.ts +7 -0
- package/dist/token/refresh-token.d.ts.map +1 -0
- package/dist/token/refresh-token.js +20 -0
- package/dist/token/refresh-token.js.map +1 -0
- package/dist/token/token-claims.d.ts +1687 -0
- package/dist/token/token-claims.d.ts.map +1 -0
- package/dist/token/token-claims.js +30 -0
- package/dist/token/token-claims.js.map +1 -0
- package/dist/token/token-data.d.ts +20 -0
- package/dist/token/token-data.d.ts.map +1 -0
- package/dist/token/token-data.js +3 -0
- package/dist/token/token-data.js.map +1 -0
- package/dist/token/token-id.d.ts +7 -0
- package/dist/token/token-id.d.ts.map +1 -0
- package/dist/token/token-id.js +20 -0
- package/dist/token/token-id.js.map +1 -0
- package/dist/token/token-manager.d.ts +48 -0
- package/dist/token/token-manager.d.ts.map +1 -0
- package/dist/token/token-manager.js +421 -0
- package/dist/token/token-manager.js.map +1 -0
- package/dist/token/token-store.d.ts +35 -0
- package/dist/token/token-store.d.ts.map +1 -0
- package/dist/token/token-store.js +38 -0
- package/dist/token/token-store.js.map +1 -0
- package/dist/token/types.d.ts +250 -0
- package/dist/token/types.d.ts.map +1 -0
- package/dist/token/types.js +36 -0
- package/dist/token/types.js.map +1 -0
- package/dist/token/verify-token-claims.d.ts +17 -0
- package/dist/token/verify-token-claims.d.ts.map +1 -0
- package/dist/token/verify-token-claims.js +39 -0
- package/dist/token/verify-token-claims.js.map +1 -0
- package/package.json +83 -0
- package/rollup.config.js +55 -0
- package/src/access-token/access-token-type.ts +5 -0
- package/src/account/account-manager.ts +55 -0
- package/src/account/account-store.ts +74 -0
- package/src/account/account.ts +10 -0
- package/src/assets/app/app.tsx +28 -0
- package/src/assets/app/backend-data.ts +65 -0
- package/src/assets/app/components/accept-form.tsx +112 -0
- package/src/assets/app/components/account-identifier.tsx +18 -0
- package/src/assets/app/components/account-picker.tsx +108 -0
- package/src/assets/app/components/client-identifier.tsx +32 -0
- package/src/assets/app/components/client-name.tsx +30 -0
- package/src/assets/app/components/error-card.tsx +41 -0
- package/src/assets/app/components/help-card.tsx +42 -0
- package/src/assets/app/components/layout-title-page.tsx +43 -0
- package/src/assets/app/components/layout-welcome.tsx +58 -0
- package/src/assets/app/components/sign-in-form.tsx +290 -0
- package/src/assets/app/components/sign-up-account-form.tsx +210 -0
- package/src/assets/app/components/sign-up-disclaimer.tsx +44 -0
- package/src/assets/app/components/url-viewer.tsx +70 -0
- package/src/assets/app/cookies.ts +11 -0
- package/src/assets/app/hooks/use-api.ts +104 -0
- package/src/assets/app/hooks/use-bound-dispatch.ts +5 -0
- package/src/assets/app/hooks/use-csrf-token.ts +5 -0
- package/src/assets/app/lib/api.ts +64 -0
- package/src/assets/app/lib/clsx.ts +4 -0
- package/src/assets/app/lib/util.ts +10 -0
- package/src/assets/app/main.css +11 -0
- package/src/assets/app/main.tsx +28 -0
- package/src/assets/app/views/accept-view.tsx +51 -0
- package/src/assets/app/views/authorize-view.tsx +101 -0
- package/src/assets/app/views/error-view.tsx +27 -0
- package/src/assets/app/views/sign-in-view.tsx +121 -0
- package/src/assets/app/views/sign-up-view.tsx +93 -0
- package/src/assets/app/views/welcome-view.tsx +61 -0
- package/src/assets/asset.ts +8 -0
- package/src/assets/assets-middleware.ts +32 -0
- package/src/assets/index.ts +74 -0
- package/src/client/client-auth.ts +45 -0
- package/src/client/client-data.ts +9 -0
- package/src/client/client-id.ts +4 -0
- package/src/client/client-info.ts +13 -0
- package/src/client/client-manager.ts +818 -0
- package/src/client/client-store.ts +38 -0
- package/src/client/client-utils.ts +43 -0
- package/src/client/client.ts +231 -0
- package/src/constants.ts +69 -0
- package/src/device/device-data.ts +11 -0
- package/src/device/device-details.ts +43 -0
- package/src/device/device-id.ts +23 -0
- package/src/device/device-manager.ts +287 -0
- package/src/device/device-store.ts +35 -0
- package/src/device/session-id.ts +22 -0
- package/src/dpop/dpop-manager.ts +147 -0
- package/src/dpop/dpop-nonce.ts +104 -0
- package/src/errors/access-denied-error.ts +26 -0
- package/src/errors/account-selection-required-error.ts +12 -0
- package/src/errors/consent-required-error.ts +12 -0
- package/src/errors/invalid-authorization-details-error.ts +22 -0
- package/src/errors/invalid-client-error.ts +20 -0
- package/src/errors/invalid-client-id-error.ts +20 -0
- package/src/errors/invalid-client-metadata-error.ts +19 -0
- package/src/errors/invalid-dpop-key-binding-error.ts +21 -0
- package/src/errors/invalid-dpop-proof-error.ts +13 -0
- package/src/errors/invalid-grant-error.ts +16 -0
- package/src/errors/invalid-parameters-error.ts +12 -0
- package/src/errors/invalid-redirect-uri-error.ts +17 -0
- package/src/errors/invalid-request-error.ts +30 -0
- package/src/errors/invalid-token-error.ts +59 -0
- package/src/errors/login-required-error.ts +12 -0
- package/src/errors/oauth-error.ts +28 -0
- package/src/errors/unauthorized-client-error.ts +20 -0
- package/src/errors/use-dpop-nonce-error.ts +32 -0
- package/src/errors/www-authenticate-error.ts +65 -0
- package/src/index.ts +15 -0
- package/src/lib/html/README.md +9 -0
- package/src/lib/html/build-document.ts +98 -0
- package/src/lib/html/escapers.ts +66 -0
- package/src/lib/html/html.ts +61 -0
- package/src/lib/html/index.ts +5 -0
- package/src/lib/html/tags.ts +58 -0
- package/src/lib/html/util.ts +21 -0
- package/src/lib/http/README.md +11 -0
- package/src/lib/http/accept.ts +91 -0
- package/src/lib/http/context.ts +11 -0
- package/src/lib/http/index.ts +9 -0
- package/src/lib/http/method.ts +18 -0
- package/src/lib/http/middleware.ts +183 -0
- package/src/lib/http/parser.ts +64 -0
- package/src/lib/http/path.ts +82 -0
- package/src/lib/http/request.ts +141 -0
- package/src/lib/http/response.ts +133 -0
- package/src/lib/http/route.ts +56 -0
- package/src/lib/http/router.ts +118 -0
- package/src/lib/http/stream.ts +78 -0
- package/src/lib/http/types.ts +22 -0
- package/src/lib/http/url.ts +23 -0
- package/src/lib/redis.ts +23 -0
- package/src/lib/util/authorization-header.ts +26 -0
- package/src/lib/util/cast.ts +4 -0
- package/src/lib/util/crypto.ts +27 -0
- package/src/lib/util/date.ts +7 -0
- package/src/lib/util/hostname.ts +19 -0
- package/src/lib/util/redirect-uri.ts +46 -0
- package/src/lib/util/time.ts +33 -0
- package/src/lib/util/type.ts +4 -0
- package/src/lib/util/well-known.ts +8 -0
- package/src/metadata/build-metadata.ts +165 -0
- package/src/oauth-client.ts +3 -0
- package/src/oauth-dpop.ts +2 -0
- package/src/oauth-errors.ts +21 -0
- package/src/oauth-hooks.ts +66 -0
- package/src/oauth-provider.ts +1409 -0
- package/src/oauth-store.ts +11 -0
- package/src/oauth-verifier.ts +219 -0
- package/src/oidc/claims.ts +35 -0
- package/src/oidc/sub.ts +4 -0
- package/src/oidc/userinfo.ts +11 -0
- package/src/output/build-error-payload.ts +143 -0
- package/src/output/customization.ts +96 -0
- package/src/output/send-authorize-page.ts +111 -0
- package/src/output/send-authorize-redirect.ts +130 -0
- package/src/output/send-error-page.ts +41 -0
- package/src/output/send-web-page.ts +66 -0
- package/src/parameters/claims-requested.ts +106 -0
- package/src/parameters/oidc-payload.ts +28 -0
- package/src/replay/replay-manager.ts +38 -0
- package/src/replay/replay-store-memory.ts +36 -0
- package/src/replay/replay-store-redis.ts +31 -0
- package/src/replay/replay-store.ts +44 -0
- package/src/request/code.ts +24 -0
- package/src/request/request-data.ts +26 -0
- package/src/request/request-id.ts +23 -0
- package/src/request/request-info.ts +12 -0
- package/src/request/request-manager.ts +479 -0
- package/src/request/request-store-memory.ts +39 -0
- package/src/request/request-store-redis.ts +71 -0
- package/src/request/request-store.ts +54 -0
- package/src/request/request-uri.ts +29 -0
- package/src/request/types.ts +48 -0
- package/src/signer/signed-token-payload.ts +35 -0
- package/src/signer/signer.ts +165 -0
- package/src/token/refresh-token.ts +31 -0
- package/src/token/token-claims.ts +31 -0
- package/src/token/token-data.ts +33 -0
- package/src/token/token-id.ts +26 -0
- package/src/token/token-manager.ts +591 -0
- package/src/token/token-store.ts +78 -0
- package/src/token/types.ts +86 -0
- package/src/token/verify-token-claims.ts +65 -0
- package/tailwind.config.js +13 -0
- package/tsconfig.backend.json +9 -0
- package/tsconfig.frontend.json +11 -0
- package/tsconfig.json +8 -0
- package/tsconfig.tools.json +8 -0
@@ -0,0 +1,130 @@
|
|
1
|
+
import {
|
2
|
+
OAuthAuthenticationRequestParameters,
|
3
|
+
OAuthTokenType,
|
4
|
+
} from '@atproto/oauth-types'
|
5
|
+
import { ServerResponse } from 'node:http'
|
6
|
+
|
7
|
+
import { Client } from '../client/client.js'
|
8
|
+
import { html, js } from '../lib/html/index.js'
|
9
|
+
import { Code } from '../request/code.js'
|
10
|
+
import { sendWebPage } from './send-web-page.js'
|
11
|
+
|
12
|
+
// https://datatracker.ietf.org/doc/html/draft-ietf-oauth-v2-1-11#section-7.5.4
|
13
|
+
const REDIRECT_STATUS_CODE = 303
|
14
|
+
|
15
|
+
export type AuthorizationResponseParameters = {
|
16
|
+
// Will be added from AuthorizationResultRedirect['issuer']
|
17
|
+
// iss: string // rfc9207
|
18
|
+
|
19
|
+
// Will be added from AuthorizationResultRedirect['parameters']
|
20
|
+
// state?: string
|
21
|
+
|
22
|
+
code?: Code
|
23
|
+
id_token?: string
|
24
|
+
access_token?: string
|
25
|
+
token_type?: OAuthTokenType
|
26
|
+
expires_in?: string
|
27
|
+
|
28
|
+
response?: string // FAPI JARM
|
29
|
+
session_state?: string // OIDC Session Management
|
30
|
+
|
31
|
+
error?: string
|
32
|
+
error_description?: string
|
33
|
+
error_uri?: string
|
34
|
+
}
|
35
|
+
|
36
|
+
export type AuthorizationResultRedirect = {
|
37
|
+
issuer: string
|
38
|
+
client: Client
|
39
|
+
parameters: OAuthAuthenticationRequestParameters
|
40
|
+
redirect: AuthorizationResponseParameters
|
41
|
+
}
|
42
|
+
|
43
|
+
export async function sendAuthorizeRedirect(
|
44
|
+
res: ServerResponse,
|
45
|
+
result: AuthorizationResultRedirect,
|
46
|
+
): Promise<void> {
|
47
|
+
const { issuer, parameters, redirect, client } = result
|
48
|
+
|
49
|
+
const uri = parameters.redirect_uri || client.metadata.redirect_uris[0]
|
50
|
+
const mode = parameters.response_mode || 'query' // @TODO: default should depend on response_type
|
51
|
+
|
52
|
+
const entries: [string, string][] = Object.entries({
|
53
|
+
iss: issuer, // rfc9207
|
54
|
+
state: parameters.state,
|
55
|
+
|
56
|
+
response: redirect.response, // FAPI JARM
|
57
|
+
session_state: redirect.session_state, // OIDC Session Management
|
58
|
+
|
59
|
+
code: redirect.code,
|
60
|
+
id_token: redirect.id_token,
|
61
|
+
access_token: redirect.access_token,
|
62
|
+
expires_in: redirect.expires_in,
|
63
|
+
token_type: redirect.token_type,
|
64
|
+
|
65
|
+
error: redirect.error,
|
66
|
+
error_description: redirect.error_description,
|
67
|
+
error_uri: redirect.error_uri,
|
68
|
+
}).filter((entry): entry is [string, string] => entry[1] != null)
|
69
|
+
|
70
|
+
res.setHeader('Cache-Control', 'no-store')
|
71
|
+
|
72
|
+
switch (mode) {
|
73
|
+
case 'query':
|
74
|
+
return writeQuery(res, uri, entries)
|
75
|
+
case 'fragment':
|
76
|
+
return writeFragment(res, uri, entries)
|
77
|
+
case 'form_post':
|
78
|
+
return writeFormPost(res, uri, entries)
|
79
|
+
}
|
80
|
+
|
81
|
+
// @ts-expect-error fool proof
|
82
|
+
throw new Error(`Unsupported mode: ${mode}`)
|
83
|
+
}
|
84
|
+
|
85
|
+
function writeQuery(
|
86
|
+
res: ServerResponse,
|
87
|
+
uri: string,
|
88
|
+
entries: readonly [string, string][],
|
89
|
+
) {
|
90
|
+
const url = new URL(uri)
|
91
|
+
for (const [key, value] of entries) url.searchParams.set(key, value)
|
92
|
+
res.writeHead(REDIRECT_STATUS_CODE, { Location: url.href }).end()
|
93
|
+
}
|
94
|
+
|
95
|
+
function writeFragment(
|
96
|
+
res: ServerResponse,
|
97
|
+
uri: string,
|
98
|
+
entries: readonly [string, string][],
|
99
|
+
) {
|
100
|
+
const url = new URL(uri)
|
101
|
+
const searchParams = new URLSearchParams()
|
102
|
+
for (const [key, value] of entries) searchParams.set(key, value)
|
103
|
+
url.hash = searchParams.toString()
|
104
|
+
res.writeHead(REDIRECT_STATUS_CODE, { Location: url.href }).end()
|
105
|
+
}
|
106
|
+
|
107
|
+
async function writeFormPost(
|
108
|
+
res: ServerResponse,
|
109
|
+
uri: string,
|
110
|
+
entries: readonly [string, string][],
|
111
|
+
) {
|
112
|
+
// Prevent the Chrome from caching this page
|
113
|
+
// see: https://latesthackingnews.com/2023/12/12/google-updates-chrome-bfcache-for-faster-page-viewing/
|
114
|
+
res.setHeader('Set-Cookie', `bfCacheBypass=foo; max-age=1; SameSite=Lax`)
|
115
|
+
res.setHeader('Cache-Control', 'no-store')
|
116
|
+
res.setHeader('Permissions-Policy', 'otp-credentials=*, document-domain=()')
|
117
|
+
|
118
|
+
return sendWebPage(res, {
|
119
|
+
htmlAttrs: { lang: 'en' },
|
120
|
+
body: html`
|
121
|
+
<form method="post" action="${uri}">
|
122
|
+
${entries.map(([key, value]) => [
|
123
|
+
html`<input type="hidden" name="${key}" value="${value}" />`,
|
124
|
+
])}
|
125
|
+
<input type="submit" value="Continue" />
|
126
|
+
</form>
|
127
|
+
`,
|
128
|
+
scripts: [js`document.forms[0].submit();`],
|
129
|
+
})
|
130
|
+
}
|
@@ -0,0 +1,41 @@
|
|
1
|
+
import { ServerResponse } from 'node:http'
|
2
|
+
|
3
|
+
import { getAsset } from '../assets/index.js'
|
4
|
+
import { cssCode, html } from '../lib/html/index.js'
|
5
|
+
import { buildErrorPayload, buildErrorStatus } from './build-error-payload.js'
|
6
|
+
import {
|
7
|
+
Customization,
|
8
|
+
buildCustomizationCss,
|
9
|
+
buildCustomizationData,
|
10
|
+
} from './customization.js'
|
11
|
+
import { declareBackendData, sendWebPage } from './send-web-page.js'
|
12
|
+
|
13
|
+
export async function sendErrorPage(
|
14
|
+
res: ServerResponse,
|
15
|
+
err: unknown,
|
16
|
+
customization?: Customization,
|
17
|
+
): Promise<void> {
|
18
|
+
const [jsAsset, cssAsset] = await Promise.all([
|
19
|
+
getAsset('main.js'),
|
20
|
+
getAsset('main.css'),
|
21
|
+
])
|
22
|
+
|
23
|
+
return sendWebPage(res, {
|
24
|
+
status: buildErrorStatus(err),
|
25
|
+
scripts: [
|
26
|
+
declareBackendData(
|
27
|
+
'__customizationData',
|
28
|
+
buildCustomizationData(customization),
|
29
|
+
),
|
30
|
+
declareBackendData('__errorData', buildErrorPayload(err)),
|
31
|
+
jsAsset, // Last (to be able to read the global variables)
|
32
|
+
],
|
33
|
+
styles: [
|
34
|
+
cssAsset, // First (to be overridden by customization)
|
35
|
+
cssCode(buildCustomizationCss(customization)),
|
36
|
+
],
|
37
|
+
htmlAttrs: { lang: 'en' },
|
38
|
+
title: 'Error',
|
39
|
+
body: html`<div id="root"></div>`,
|
40
|
+
})
|
41
|
+
}
|
@@ -0,0 +1,66 @@
|
|
1
|
+
import { createHash } from 'node:crypto'
|
2
|
+
import { ServerResponse } from 'node:http'
|
3
|
+
|
4
|
+
import {
|
5
|
+
AssetRef,
|
6
|
+
buildDocument,
|
7
|
+
BuildDocumentOptions,
|
8
|
+
Html,
|
9
|
+
js,
|
10
|
+
} from '../lib/html/index.js'
|
11
|
+
import { writeHtml } from '../lib/http/response.js'
|
12
|
+
|
13
|
+
export function declareBackendData(name: string, data: unknown) {
|
14
|
+
// The script tag is removed after the data is assigned to the global variable
|
15
|
+
// to prevent other scripts from deducing the value of the variable. The "app"
|
16
|
+
// script will read the global variable and then unset it. See
|
17
|
+
// "readBackendData" in "src/assets/app/backend-data.ts".
|
18
|
+
return js`window[${name}]=${data};document.currentScript.remove();`
|
19
|
+
}
|
20
|
+
|
21
|
+
export function sendWebPage(
|
22
|
+
res: ServerResponse,
|
23
|
+
{ status = 200, ...options }: BuildDocumentOptions & { status?: number },
|
24
|
+
): void {
|
25
|
+
// @TODO: make these headers configurable (?)
|
26
|
+
res.setHeader('Cross-Origin-Embedder-Policy', 'credentialless')
|
27
|
+
res.setHeader('Cross-Origin-Resource-Policy', 'same-origin')
|
28
|
+
res.setHeader('Cross-Origin-Opener-Policy', 'same-origin')
|
29
|
+
res.setHeader('Referrer-Policy', 'same-origin')
|
30
|
+
res.setHeader('X-Frame-Options', 'DENY')
|
31
|
+
res.setHeader('X-Content-Type-Options', 'nosniff')
|
32
|
+
res.setHeader('X-XSS-Protection', '0')
|
33
|
+
res.setHeader('Strict-Transport-Security', 'max-age=63072000')
|
34
|
+
res.setHeader(
|
35
|
+
'Content-Security-Policy',
|
36
|
+
[
|
37
|
+
`default-src 'none'`,
|
38
|
+
`frame-ancestors 'none'`,
|
39
|
+
`form-action 'none'`,
|
40
|
+
`base-uri ${options.base?.origin || `'none'`}`,
|
41
|
+
`script-src 'self' ${
|
42
|
+
options.scripts?.map(assetToHash).map(hashToCspRule).join(' ') ?? ''
|
43
|
+
}`,
|
44
|
+
`style-src 'self' ${
|
45
|
+
options.styles?.map(assetToHash).map(hashToCspRule).join(' ') ?? ''
|
46
|
+
}`,
|
47
|
+
`img-src 'self' data: https:`,
|
48
|
+
`connect-src 'self'`,
|
49
|
+
`upgrade-insecure-requests`,
|
50
|
+
].join('; '),
|
51
|
+
)
|
52
|
+
|
53
|
+
const html = buildDocument(options)
|
54
|
+
|
55
|
+
writeHtml(res, html.toString(), status)
|
56
|
+
}
|
57
|
+
|
58
|
+
function assetToHash(asset: Html | AssetRef): string {
|
59
|
+
return asset instanceof Html
|
60
|
+
? createHash('sha256').update(asset.toString()).digest('base64')
|
61
|
+
: asset.sha256
|
62
|
+
}
|
63
|
+
|
64
|
+
function hashToCspRule(hash: string): string {
|
65
|
+
return `'sha256-${hash}'`
|
66
|
+
}
|
@@ -0,0 +1,106 @@
|
|
1
|
+
import {
|
2
|
+
OAuthAuthenticationRequestParameters,
|
3
|
+
OidcClaimsParameter,
|
4
|
+
OidcEntityType,
|
5
|
+
} from '@atproto/oauth-types'
|
6
|
+
import { InvalidRequestError } from '../errors/invalid-request-error.js'
|
7
|
+
|
8
|
+
export function claimRequested(
|
9
|
+
parameters: OAuthAuthenticationRequestParameters,
|
10
|
+
entityType: OidcEntityType,
|
11
|
+
claimName: OidcClaimsParameter,
|
12
|
+
value: unknown,
|
13
|
+
): boolean {
|
14
|
+
if (claimAvailable(parameters, entityType, claimName, value)) {
|
15
|
+
return true
|
16
|
+
}
|
17
|
+
|
18
|
+
const entityClaims = parameters.claims?.[entityType]
|
19
|
+
if (entityClaims?.[claimName]?.essential === true) {
|
20
|
+
// https://openid.net/specs/openid-connect-core-1_0.html#rfc.section.5.5.1
|
21
|
+
//
|
22
|
+
// > By requesting Claims as Essential Claims, the RP indicates to the
|
23
|
+
// > End-User that releasing these Claims will ensure a smooth
|
24
|
+
// > authorization for the specific task requested by the End-User. Note
|
25
|
+
// > that even if the Claims are not available because the End-User did
|
26
|
+
// > not authorize their release or they are not present, the
|
27
|
+
// > Authorization Server MUST NOT generate an error when Claims are not
|
28
|
+
// > returned, whether they are Essential or Voluntary, unless otherwise
|
29
|
+
// > specified in the description of the specific claim.
|
30
|
+
switch (claimName) {
|
31
|
+
case 'acr':
|
32
|
+
// https://openid.net/specs/openid-connect-core-1_0.html#rfc.section.5.5.1.1
|
33
|
+
//
|
34
|
+
// > If this is an Essential Claim and the requirement cannot be met,
|
35
|
+
// > then the Authorization Server MUST treat that outcome as a failed
|
36
|
+
// > authentication attempt.
|
37
|
+
throw new InvalidRequestError(
|
38
|
+
`Unable to provide essential claim: ${claimName}`,
|
39
|
+
)
|
40
|
+
}
|
41
|
+
}
|
42
|
+
|
43
|
+
return false
|
44
|
+
}
|
45
|
+
|
46
|
+
function claimAvailable(
|
47
|
+
parameters: OAuthAuthenticationRequestParameters,
|
48
|
+
entityType: OidcEntityType,
|
49
|
+
claimName: OidcClaimsParameter,
|
50
|
+
value: unknown,
|
51
|
+
): boolean {
|
52
|
+
if (value === undefined) return false
|
53
|
+
|
54
|
+
if (parameters.claims) {
|
55
|
+
const entityClaims = parameters.claims[entityType]
|
56
|
+
if (entityClaims === undefined) return false
|
57
|
+
|
58
|
+
const claimConfig = entityClaims[claimName]
|
59
|
+
if (claimConfig === undefined) return false
|
60
|
+
if (claimConfig === null) return true
|
61
|
+
|
62
|
+
if (
|
63
|
+
claimConfig.value !== undefined &&
|
64
|
+
!compareClaimValue(claimConfig.value, value)
|
65
|
+
) {
|
66
|
+
return false
|
67
|
+
}
|
68
|
+
|
69
|
+
if (
|
70
|
+
claimConfig?.values !== undefined &&
|
71
|
+
!claimConfig.values.some((v) => compareClaimValue(v, value))
|
72
|
+
) {
|
73
|
+
return false
|
74
|
+
}
|
75
|
+
}
|
76
|
+
|
77
|
+
return true
|
78
|
+
}
|
79
|
+
|
80
|
+
type DefinedValue = NonNullable<unknown> | null
|
81
|
+
|
82
|
+
function compareClaimValue(
|
83
|
+
expectedValue: DefinedValue,
|
84
|
+
value: DefinedValue,
|
85
|
+
): boolean {
|
86
|
+
const expectedType = typeof expectedValue
|
87
|
+
const valueType = typeof value
|
88
|
+
|
89
|
+
if (expectedType !== valueType) return false
|
90
|
+
|
91
|
+
switch (typeof expectedValue) {
|
92
|
+
case 'undefined':
|
93
|
+
case 'string':
|
94
|
+
case 'number':
|
95
|
+
case 'boolean':
|
96
|
+
return expectedValue === value
|
97
|
+
case 'object':
|
98
|
+
if (expectedValue === null) return value === null
|
99
|
+
// @TODO (?): allow object comparison
|
100
|
+
// falls through
|
101
|
+
default:
|
102
|
+
throw new InvalidRequestError(
|
103
|
+
`Unable to compare claim value of type ${expectedType}`,
|
104
|
+
)
|
105
|
+
}
|
106
|
+
}
|
@@ -0,0 +1,28 @@
|
|
1
|
+
import { OAuthAuthenticationRequestParameters } from '@atproto/oauth-types'
|
2
|
+
import { Account } from '../account/account.js'
|
3
|
+
import { OIDCStandardPayload, OIDC_SCOPE_CLAIMS } from '../oidc/claims.js'
|
4
|
+
import { claimRequested } from './claims-requested.js'
|
5
|
+
|
6
|
+
export function oidcPayload(
|
7
|
+
params: OAuthAuthenticationRequestParameters,
|
8
|
+
account: Account,
|
9
|
+
) {
|
10
|
+
const payload: OIDCStandardPayload = {}
|
11
|
+
|
12
|
+
const scopes = params.scope ? params.scope?.split(' ') : undefined
|
13
|
+
if (scopes) {
|
14
|
+
for (const [scope, claims] of Object.entries(OIDC_SCOPE_CLAIMS)) {
|
15
|
+
const allowed = scopes.includes(scope)
|
16
|
+
for (const claim of claims) {
|
17
|
+
const value = allowed ? account[claim] : undefined
|
18
|
+
// Should not throw as RequestManager should have already checked
|
19
|
+
// that all the essential claims are available.
|
20
|
+
if (claimRequested(params, 'id_token', claim, value)) {
|
21
|
+
payload[claim] = value as any // All good as long as the account props match the claims
|
22
|
+
}
|
23
|
+
}
|
24
|
+
}
|
25
|
+
}
|
26
|
+
|
27
|
+
return payload
|
28
|
+
}
|
@@ -0,0 +1,38 @@
|
|
1
|
+
import { ClientId } from '../client/client-id.js'
|
2
|
+
import {
|
3
|
+
CLIENT_ASSERTION_MAX_AGE,
|
4
|
+
DPOP_NONCE_MAX_AGE,
|
5
|
+
JAR_MAX_AGE,
|
6
|
+
} from '../constants.js'
|
7
|
+
import { ReplayStore } from './replay-store.js'
|
8
|
+
|
9
|
+
const SECURITY_RATIO = 1.1 // 10% extra time for security
|
10
|
+
const asTimeFrame = (timeFrame: number) => Math.ceil(timeFrame * SECURITY_RATIO)
|
11
|
+
|
12
|
+
export class ReplayManager {
|
13
|
+
constructor(protected readonly replayStore: ReplayStore) {}
|
14
|
+
|
15
|
+
async uniqueAuth(jti: string, clientId: ClientId): Promise<boolean> {
|
16
|
+
return this.replayStore.unique(
|
17
|
+
`Auth@${clientId}`,
|
18
|
+
jti,
|
19
|
+
asTimeFrame(CLIENT_ASSERTION_MAX_AGE),
|
20
|
+
)
|
21
|
+
}
|
22
|
+
|
23
|
+
async uniqueJar(jti: string, clientId: ClientId): Promise<boolean> {
|
24
|
+
return this.replayStore.unique(
|
25
|
+
`JAR@${clientId}`,
|
26
|
+
jti,
|
27
|
+
asTimeFrame(JAR_MAX_AGE),
|
28
|
+
)
|
29
|
+
}
|
30
|
+
|
31
|
+
async uniqueDpop(jti: string, clientId?: ClientId): Promise<boolean> {
|
32
|
+
return this.replayStore.unique(
|
33
|
+
clientId ? `DPoP@${clientId}` : `DPoP`,
|
34
|
+
jti,
|
35
|
+
asTimeFrame(DPOP_NONCE_MAX_AGE),
|
36
|
+
)
|
37
|
+
}
|
38
|
+
}
|
@@ -0,0 +1,36 @@
|
|
1
|
+
import type { ReplayStore } from './replay-store.js'
|
2
|
+
|
3
|
+
export class ReplayStoreMemory implements ReplayStore {
|
4
|
+
private lastCleanup = Date.now()
|
5
|
+
private nonces = new Map<string, number>()
|
6
|
+
|
7
|
+
/**
|
8
|
+
* Returns true if the nonce is unique within the given time frame.
|
9
|
+
*/
|
10
|
+
async unique(
|
11
|
+
namespace: string,
|
12
|
+
nonce: string,
|
13
|
+
timeFrame: number,
|
14
|
+
): Promise<boolean> {
|
15
|
+
this.cleanup()
|
16
|
+
const key = `${namespace}:${nonce}`
|
17
|
+
|
18
|
+
const now = Date.now()
|
19
|
+
|
20
|
+
const exp = this.nonces.get(key)
|
21
|
+
this.nonces.set(key, now + timeFrame)
|
22
|
+
|
23
|
+
return exp == null || exp < now
|
24
|
+
}
|
25
|
+
|
26
|
+
private cleanup() {
|
27
|
+
const now = Date.now()
|
28
|
+
|
29
|
+
if (this.lastCleanup < now - 60_000) {
|
30
|
+
for (const [key, expires] of this.nonces) {
|
31
|
+
if (expires < now) this.nonces.delete(key)
|
32
|
+
}
|
33
|
+
this.lastCleanup = now
|
34
|
+
}
|
35
|
+
}
|
36
|
+
}
|
@@ -0,0 +1,31 @@
|
|
1
|
+
import { Redis } from 'ioredis'
|
2
|
+
|
3
|
+
import { CreateRedisOptions, createRedis } from '../lib/redis.js'
|
4
|
+
import type { ReplayStore } from './replay-store.js'
|
5
|
+
|
6
|
+
export type { CreateRedisOptions, Redis }
|
7
|
+
|
8
|
+
export type ReplayStoreRedisOptions = {
|
9
|
+
redis: CreateRedisOptions
|
10
|
+
}
|
11
|
+
|
12
|
+
export class ReplayStoreRedis implements ReplayStore {
|
13
|
+
private readonly redis: Redis
|
14
|
+
|
15
|
+
constructor(options: ReplayStoreRedisOptions) {
|
16
|
+
this.redis = createRedis(options.redis)
|
17
|
+
}
|
18
|
+
|
19
|
+
/**
|
20
|
+
* Returns true if the nonce is unique within the given time frame.
|
21
|
+
*/
|
22
|
+
async unique(
|
23
|
+
namespace: string,
|
24
|
+
nonce: string,
|
25
|
+
timeFrame: number,
|
26
|
+
): Promise<boolean> {
|
27
|
+
const key = `nonces:${namespace}:${nonce}`
|
28
|
+
const prev = await this.redis.set(key, '1', 'PX', timeFrame, 'GET')
|
29
|
+
return prev == null
|
30
|
+
}
|
31
|
+
}
|
@@ -0,0 +1,44 @@
|
|
1
|
+
import { Awaitable } from '../lib/util/type.js'
|
2
|
+
|
3
|
+
// Export all types needed to implement the ReplayStore interface
|
4
|
+
export type { Awaitable }
|
5
|
+
|
6
|
+
export interface ReplayStore {
|
7
|
+
/**
|
8
|
+
* Returns true if the nonce is unique within the given time frame. While not
|
9
|
+
* strictly necessary for security purposes, the namespace should be used to
|
10
|
+
* mitigate denial of service attacks from one client to the other.
|
11
|
+
*
|
12
|
+
* @param timeFrame expressed in milliseconds. Will never exceed 24 hours.
|
13
|
+
*/
|
14
|
+
unique(
|
15
|
+
namespace: string,
|
16
|
+
nonce: string,
|
17
|
+
timeFrame: number,
|
18
|
+
): Awaitable<boolean>
|
19
|
+
}
|
20
|
+
|
21
|
+
export function isReplayStore(
|
22
|
+
implementation: Record<string, unknown> & Partial<ReplayStore>,
|
23
|
+
): implementation is Record<string, unknown> & ReplayStore {
|
24
|
+
return typeof implementation.unique === 'function'
|
25
|
+
}
|
26
|
+
|
27
|
+
export function ifReplayStore(
|
28
|
+
implementation?: Record<string, unknown> & Partial<ReplayStore>,
|
29
|
+
): ReplayStore | undefined {
|
30
|
+
if (implementation && isReplayStore(implementation)) {
|
31
|
+
return implementation
|
32
|
+
}
|
33
|
+
|
34
|
+
return undefined
|
35
|
+
}
|
36
|
+
|
37
|
+
export function asReplayStore(
|
38
|
+
implementation?: Record<string, unknown> & Partial<ReplayStore>,
|
39
|
+
): ReplayStore {
|
40
|
+
const store = ifReplayStore(implementation)
|
41
|
+
if (store) return store
|
42
|
+
|
43
|
+
throw new Error('Invalid ReplayStore implementation')
|
44
|
+
}
|
@@ -0,0 +1,24 @@
|
|
1
|
+
import { z } from 'zod'
|
2
|
+
|
3
|
+
import { CODE_BYTES_LENGTH, CODE_PREFIX } from '../constants.js'
|
4
|
+
import { randomHexId } from '../lib/util/crypto.js'
|
5
|
+
|
6
|
+
export const CODE_LENGTH = CODE_PREFIX.length + CODE_BYTES_LENGTH * 2 // hex encoding
|
7
|
+
|
8
|
+
export const codeSchema = z
|
9
|
+
.string()
|
10
|
+
.length(CODE_LENGTH) // hex encoding
|
11
|
+
.refine(
|
12
|
+
(v): v is `${typeof CODE_PREFIX}${string}` => v.startsWith(CODE_PREFIX),
|
13
|
+
{
|
14
|
+
message: `Invalid code format`,
|
15
|
+
},
|
16
|
+
)
|
17
|
+
|
18
|
+
export const isCode = (data: unknown): data is Code =>
|
19
|
+
codeSchema.safeParse(data).success
|
20
|
+
|
21
|
+
export type Code = z.infer<typeof codeSchema>
|
22
|
+
export const generateCode = async (): Promise<Code> => {
|
23
|
+
return `${CODE_PREFIX}${await randomHexId(CODE_BYTES_LENGTH)}`
|
24
|
+
}
|
@@ -0,0 +1,26 @@
|
|
1
|
+
import { OAuthAuthenticationRequestParameters } from '@atproto/oauth-types'
|
2
|
+
|
3
|
+
import { ClientAuth } from '../client/client-auth.js'
|
4
|
+
import { ClientId } from '../client/client-id.js'
|
5
|
+
import { DeviceId } from '../device/device-id.js'
|
6
|
+
import { Sub } from '../oidc/sub.js'
|
7
|
+
import { Code } from './code.js'
|
8
|
+
|
9
|
+
export type RequestData = {
|
10
|
+
clientId: ClientId
|
11
|
+
clientAuth: ClientAuth
|
12
|
+
parameters: Readonly<OAuthAuthenticationRequestParameters>
|
13
|
+
expiresAt: Date
|
14
|
+
deviceId: DeviceId | null
|
15
|
+
sub: Sub | null
|
16
|
+
code: Code | null
|
17
|
+
}
|
18
|
+
|
19
|
+
export type RequestDataAuthorized = RequestData & {
|
20
|
+
sub: Sub
|
21
|
+
deviceId: DeviceId
|
22
|
+
}
|
23
|
+
|
24
|
+
export const isRequestDataAuthorized = (
|
25
|
+
data: RequestData,
|
26
|
+
): data is RequestDataAuthorized => data.sub !== null && data.deviceId !== null
|
@@ -0,0 +1,23 @@
|
|
1
|
+
import { z } from 'zod'
|
2
|
+
|
3
|
+
import { REQUEST_ID_BYTES_LENGTH, REQUEST_ID_PREFIX } from '../constants.js'
|
4
|
+
import { randomHexId } from '../lib/util/crypto.js'
|
5
|
+
|
6
|
+
export const REQUEST_ID_LENGTH =
|
7
|
+
REQUEST_ID_PREFIX.length + REQUEST_ID_BYTES_LENGTH * 2 // hex encoding
|
8
|
+
|
9
|
+
export const requestIdSchema = z
|
10
|
+
.string()
|
11
|
+
.length(REQUEST_ID_LENGTH)
|
12
|
+
.refine(
|
13
|
+
(v): v is `${typeof REQUEST_ID_PREFIX}${string}` =>
|
14
|
+
v.startsWith(REQUEST_ID_PREFIX),
|
15
|
+
{
|
16
|
+
message: `Invalid request ID format`,
|
17
|
+
},
|
18
|
+
)
|
19
|
+
|
20
|
+
export type RequestId = z.infer<typeof requestIdSchema>
|
21
|
+
export const generateRequestId = async (): Promise<RequestId> => {
|
22
|
+
return `${REQUEST_ID_PREFIX}${await randomHexId(REQUEST_ID_BYTES_LENGTH)}`
|
23
|
+
}
|
@@ -0,0 +1,12 @@
|
|
1
|
+
import { OAuthAuthenticationRequestParameters } from '@atproto/oauth-types'
|
2
|
+
import { ClientAuth } from '../client/client-auth.js'
|
3
|
+
import { RequestId } from './request-id.js'
|
4
|
+
import { RequestUri } from './request-uri.js'
|
5
|
+
|
6
|
+
export type RequestInfo = {
|
7
|
+
id: RequestId
|
8
|
+
uri: RequestUri
|
9
|
+
parameters: Readonly<OAuthAuthenticationRequestParameters>
|
10
|
+
expiresAt: Date
|
11
|
+
clientAuth: ClientAuth
|
12
|
+
}
|