@openstax/ts-utils 1.33.1 → 1.34.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.cfnlintrc +2 -0
- package/.github/CODEOWNERS +1 -0
- package/.github/workflows/ci.yml +36 -0
- package/.github/workflows/lint.yml +55 -0
- package/.nvmrc +1 -0
- package/.syncignore +4 -0
- package/.syncpackrc +18 -0
- package/CONTRIBUTING.md +96 -0
- package/LICENSE +661 -0
- package/Procfile +1 -0
- package/README.md +62 -90
- package/app.json +23 -0
- package/cspell.json +32 -0
- package/deploy/constants.env +21 -0
- package/deploy/deploy.bash +157 -0
- package/deploy/deployment-alt-region.cfn.yml +70 -0
- package/deploy/deployment.cfn.yml +650 -0
- package/deploy/destroy-deployment.bash +23 -0
- package/deploy/shared.cfn.yml +94 -0
- package/docs/lambda-build.md +35 -0
- package/package.json +12 -228
- package/packages/frontend/README.md +46 -0
- package/packages/frontend/package.json +101 -0
- package/packages/frontend/public/favicon.ico +0 -0
- package/packages/frontend/public/index.html +107 -0
- package/packages/frontend/public/maintenance.html +59 -0
- package/packages/frontend/public/manifest.json +15 -0
- package/packages/frontend/public/robots.txt +3 -0
- package/packages/frontend/script/make-certificate.bash +49 -0
- package/packages/frontend/script/server/cli.js +11 -0
- package/packages/frontend/script/server/index.js +47 -0
- package/packages/frontend/script/start.bash +22 -0
- package/packages/frontend/script/trust-localhost.bash +7 -0
- package/packages/frontend/src/auth/authProvider.ts +10 -0
- package/packages/frontend/src/auth/useAuth.ts +33 -0
- package/packages/frontend/src/components/Pagination.tsx +26 -0
- package/packages/frontend/src/configProvider/index.ts +53 -0
- package/packages/frontend/src/configProvider/use.ts +41 -0
- package/packages/frontend/src/core/context/services.spec.tsx +39 -0
- package/packages/frontend/src/core/context/services.tsx +16 -0
- package/packages/frontend/src/core/index.spec.ts +7 -0
- package/packages/frontend/src/core/index.ts +20 -0
- package/packages/frontend/src/core/services.tsx +14 -0
- package/packages/frontend/src/core/types.ts +3 -0
- package/packages/frontend/src/example/api.ts +28 -0
- package/packages/frontend/src/example/components/Layout.tsx +23 -0
- package/packages/frontend/src/example/screens/Home.spec.tsx +68 -0
- package/packages/frontend/src/example/screens/Home.tsx +78 -0
- package/packages/frontend/src/example/screens/ThingList.spec.tsx +60 -0
- package/packages/frontend/src/example/screens/ThingList.tsx +75 -0
- package/packages/frontend/src/example/screens/ThingView.spec.tsx +71 -0
- package/packages/frontend/src/example/screens/ThingView.tsx +47 -0
- package/packages/frontend/src/example/screens/index.ts +9 -0
- package/packages/frontend/src/index.css +159 -0
- package/packages/frontend/src/index.tsx +67 -0
- package/packages/frontend/src/react-app-env.d.ts +1 -0
- package/packages/frontend/src/routing/components/RouteLink.spec.tsx +55 -0
- package/packages/frontend/src/routing/components/RouteLink.tsx +35 -0
- package/packages/frontend/src/routing/middleware.ts +6 -0
- package/packages/frontend/src/routing/useQuery.ts +14 -0
- package/packages/frontend/src/setupProxy.js +19 -0
- package/packages/frontend/src/setupTests.ts +9 -0
- package/packages/frontend/src/tests/testServices.tsx +23 -0
- package/packages/frontend/tsconfig.json +27 -0
- package/packages/lambda/.eslintrc.js +64 -0
- package/packages/lambda/jest-global-setup.js +3 -0
- package/packages/lambda/jest-setup-after-env.js +1 -0
- package/packages/lambda/jest.config.js +31 -0
- package/packages/lambda/jest.resolver.js +17 -0
- package/packages/lambda/package.json +68 -0
- package/packages/lambda/script/build.bash +19 -0
- package/packages/lambda/script/bundle-functions.bash +10 -0
- package/packages/lambda/script/lambdaLocalProxy.js +16 -0
- package/packages/lambda/script/lambdaLocalProxy.spec.ts +147 -0
- package/packages/lambda/script/utils/getRouteData.ts +7 -0
- package/packages/lambda/script/utils/routeDataLoader.js +8 -0
- package/packages/lambda/script/utils/routeDataLoader.spec.ts +8 -0
- package/packages/lambda/src/functions/serviceApi/core/index.ts +7 -0
- package/packages/lambda/src/functions/serviceApi/core/request.spec.ts +38 -0
- package/packages/lambda/src/functions/serviceApi/core/request.ts +42 -0
- package/packages/lambda/src/functions/serviceApi/core/routes.spec.ts +7 -0
- package/packages/lambda/src/functions/serviceApi/core/routes.ts +10 -0
- package/packages/lambda/src/functions/serviceApi/core/services.ts +9 -0
- package/packages/lambda/src/functions/serviceApi/core/types.ts +13 -0
- package/packages/lambda/src/functions/serviceApi/entry/lambda/https-xray.ts +4 -0
- package/packages/lambda/src/functions/serviceApi/entry/lambda/index.spec.ts +48 -0
- package/packages/lambda/src/functions/serviceApi/entry/lambda/index.ts +58 -0
- package/packages/lambda/src/functions/serviceApi/entry/lambda/services.ts +36 -0
- package/packages/lambda/src/functions/serviceApi/entry/local.ts +71 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentSearchMiddleware.spec.ts +16 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentSearchMiddleware.ts +41 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentStoreMiddleware.spec.ts +78 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentStoreMiddleware.ts +70 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/routes.spec.ts +306 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/example/routes.ts +176 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/index.spec.ts +263 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/index.ts +134 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/authMiddleware.spec.ts +23 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/authMiddleware.ts +32 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/configMiddleware.spec.ts +10 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/configMiddleware.ts +7 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/frontendFileServerMiddleware.spec.ts +13 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/frontendFileServerMiddleware.ts +23 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/paginationMiddleware.spec.ts +9 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/paginationMiddleware.ts +9 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/searchMiddleware.spec.ts +12 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/searchMiddleware.ts +21 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/userRoleValidatorMiddleware.spec.ts +21 -0
- package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/userRoleValidatorMiddleware.ts +18 -0
- package/packages/lambda/tsconfig.json +30 -0
- package/packages/lambda/webpack.config.js +97 -0
- package/packages/utils/.eslintrc.js +64 -0
- package/packages/utils/README.md +118 -0
- package/packages/utils/jest-global-setup.js +3 -0
- package/packages/utils/jest.config.js +25 -0
- package/packages/utils/jest.resolver.js +17 -0
- package/packages/utils/package.json +238 -0
- package/packages/utils/src/assertions/index.spec.ts +126 -0
- package/{dist/esm/assertions/index.js → packages/utils/src/assertions/index.ts} +64 -49
- package/packages/utils/src/aws/ssmService.ts +7 -0
- package/packages/utils/src/config/awsParameterConfig.ts +24 -0
- package/packages/utils/src/config/envConfig.ts +58 -0
- package/packages/utils/src/config/index.spec.ts +165 -0
- package/{dist/esm/config/index.d.ts → packages/utils/src/config/index.ts} +29 -13
- package/packages/utils/src/config/lambdaParameterConfig.ts +49 -0
- package/{dist/esm/config/replaceConfig.js → packages/utils/src/config/replaceConfig.ts} +16 -6
- package/packages/utils/src/config/resolveConfigValue.ts +10 -0
- package/packages/utils/src/errors/index.spec.ts +35 -0
- package/{dist/esm/errors/index.js → packages/utils/src/errors/index.ts} +57 -41
- package/packages/utils/src/fetch/fetchStatusRetry.spec.ts +197 -0
- package/packages/utils/src/fetch/fetchStatusRetry.ts +33 -0
- package/packages/utils/src/fetch/index.spec.ts +34 -0
- package/packages/utils/src/fetch/index.ts +87 -0
- package/packages/utils/src/guards/index.spec.ts +58 -0
- package/{dist/esm/guards/index.d.ts → packages/utils/src/guards/index.ts} +10 -7
- package/packages/utils/src/index.spec.ts +471 -0
- package/packages/utils/src/middleware/apiErrorHandler.spec.ts +65 -0
- package/packages/utils/src/middleware/apiErrorHandler.ts +67 -0
- package/packages/utils/src/middleware/apiSlowResponseMiddleware.spec.ts +184 -0
- package/packages/utils/src/middleware/apiSlowResponseMiddleware.ts +71 -0
- package/packages/utils/src/middleware/index.spec.ts +99 -0
- package/{dist/cjs/middleware/index.d.ts → packages/utils/src/middleware/index.ts} +53 -5
- package/packages/utils/src/middleware/lambdaCorsResponseMiddleware.spec.ts +103 -0
- package/packages/utils/src/middleware/lambdaCorsResponseMiddleware.ts +52 -0
- package/packages/utils/src/middleware/throwNotFoundMiddleware.spec.ts +20 -0
- package/packages/utils/src/middleware/throwNotFoundMiddleware.ts +11 -0
- package/packages/utils/src/misc/hashValue.ts +18 -0
- package/packages/utils/src/misc/helpers.ts +259 -0
- package/packages/utils/src/misc/merge.ts +48 -0
- package/{dist/esm/misc/partitionSequence.js → packages/utils/src/misc/partitionSequence.ts} +23 -15
- package/packages/utils/src/pagination/index.spec.ts +150 -0
- package/packages/utils/src/pagination/index.ts +117 -0
- package/{dist/esm/routing/helpers.js → packages/utils/src/routing/helpers.ts} +42 -30
- package/packages/utils/src/routing/index.spec.ts +553 -0
- package/packages/utils/src/routing/index.ts +424 -0
- package/packages/utils/src/routing/validators/zod.spec.ts +16 -0
- package/packages/utils/src/routing/validators/zod.ts +14 -0
- package/packages/utils/src/services/accountsGateway/README.md +3 -0
- package/packages/utils/src/services/accountsGateway/index.spec.ts +518 -0
- package/packages/utils/src/services/accountsGateway/index.ts +251 -0
- package/packages/utils/src/services/apiGateway/README.md +93 -0
- package/packages/utils/src/services/apiGateway/index.spec.ts +254 -0
- package/packages/utils/src/services/apiGateway/index.ts +189 -0
- package/packages/utils/src/services/authProvider/README.md +21 -0
- package/packages/utils/src/services/authProvider/browser.spec.ts +391 -0
- package/packages/utils/src/services/authProvider/browser.ts +209 -0
- package/packages/utils/src/services/authProvider/decryption.spec.ts +337 -0
- package/packages/utils/src/services/authProvider/decryption.ts +98 -0
- package/packages/utils/src/services/authProvider/index.ts +93 -0
- package/packages/utils/src/services/authProvider/stub.spec.ts +29 -0
- package/packages/utils/src/services/authProvider/subrequest.spec.ts +105 -0
- package/packages/utils/src/services/authProvider/subrequest.ts +68 -0
- package/packages/utils/src/services/authProvider/utils/decryptAndVerify.spec.ts +128 -0
- package/packages/utils/src/services/authProvider/utils/decryptAndVerify.ts +106 -0
- package/packages/utils/src/services/authProvider/utils/embeddedAuthProvider.spec.ts +26 -0
- package/packages/utils/src/services/authProvider/utils/embeddedAuthProvider.ts +57 -0
- package/packages/utils/src/services/authProvider/utils/userRoleValidator.spec.ts +135 -0
- package/packages/utils/src/services/authProvider/utils/userRoleValidator.ts +49 -0
- package/packages/utils/src/services/authProvider/utils/userSubrequest.spec.ts +26 -0
- package/packages/utils/src/services/authProvider/utils/userSubrequest.ts +10 -0
- package/packages/utils/src/services/documentStore/dynamoEncoding.ts +57 -0
- package/packages/utils/src/services/documentStore/fileSystemAssert.spec.ts +43 -0
- package/packages/utils/src/services/documentStore/fileSystemAssert.ts +10 -0
- package/{dist/cjs/services/documentStore/index.d.ts → packages/utils/src/services/documentStore/index.ts} +8 -8
- package/packages/utils/src/services/documentStore/unversioned/README.md +13 -0
- package/packages/utils/src/services/documentStore/unversioned/dynamodb.spec.ts +859 -0
- package/packages/utils/src/services/documentStore/unversioned/dynamodb.ts +243 -0
- package/packages/utils/src/services/documentStore/unversioned/file-system.spec.ts +629 -0
- package/packages/utils/src/services/documentStore/unversioned/file-system.ts +194 -0
- package/{dist/cjs/services/documentStore/unversioned/index.d.ts → packages/utils/src/services/documentStore/unversioned/index.ts} +2 -0
- package/packages/utils/src/services/documentStore/versioned/README.md +13 -0
- package/packages/utils/src/services/documentStore/versioned/dynamodb.spec.ts +376 -0
- package/packages/utils/src/services/documentStore/versioned/dynamodb.ts +167 -0
- package/packages/utils/src/services/documentStore/versioned/file-system.spec.ts +262 -0
- package/packages/utils/src/services/documentStore/versioned/file-system.ts +90 -0
- package/packages/utils/src/services/documentStore/versioned/index.ts +25 -0
- package/packages/utils/src/services/exercisesGateway/README.md +5 -0
- package/packages/utils/src/services/exercisesGateway/index.spec.ts +326 -0
- package/packages/utils/src/services/exercisesGateway/index.ts +163 -0
- package/packages/utils/src/services/fileServer/index.spec.ts +88 -0
- package/packages/utils/src/services/fileServer/index.ts +43 -0
- package/packages/utils/src/services/fileServer/localFileServer.spec.ts +182 -0
- package/packages/utils/src/services/fileServer/localFileServer.ts +159 -0
- package/packages/utils/src/services/fileServer/s3FileServer.spec.ts +266 -0
- package/packages/utils/src/services/fileServer/s3FileServer.ts +155 -0
- package/packages/utils/src/services/launchParams/index.spec.ts +366 -0
- package/packages/utils/src/services/launchParams/signer.ts +73 -0
- package/packages/utils/src/services/launchParams/verifier.ts +120 -0
- package/packages/utils/src/services/logger/console.spec.ts +29 -0
- package/{dist/esm/services/logger/console.js → packages/utils/src/services/logger/console.ts} +5 -2
- package/packages/utils/src/services/logger/index.spec.ts +65 -0
- package/{dist/esm/services/logger/index.d.ts → packages/utils/src/services/logger/index.ts} +23 -9
- package/packages/utils/src/services/lrsGateway/README.md +5 -0
- package/packages/utils/src/services/lrsGateway/addStatementDefaultFields.ts +22 -0
- package/packages/utils/src/services/lrsGateway/attempt-utils.spec.ts +847 -0
- package/packages/utils/src/services/lrsGateway/attempt-utils.ts +358 -0
- package/packages/utils/src/services/lrsGateway/file-system.spec.ts +363 -0
- package/packages/utils/src/services/lrsGateway/file-system.ts +165 -0
- package/packages/utils/src/services/lrsGateway/index.spec.ts +194 -0
- package/packages/utils/src/services/lrsGateway/index.ts +257 -0
- package/packages/utils/src/services/lrsGateway/xapiUtils.spec.ts +887 -0
- package/packages/utils/src/services/lrsGateway/xapiUtils.ts +262 -0
- package/packages/utils/src/services/postgresConnection/index.spec.ts +170 -0
- package/packages/utils/src/services/postgresConnection/index.ts +84 -0
- package/packages/utils/src/services/searchProvider/README.md +3 -0
- package/packages/utils/src/services/searchProvider/index.ts +59 -0
- package/packages/utils/src/services/searchProvider/memorySearchTheBadWay.spec.ts +526 -0
- package/packages/utils/src/services/searchProvider/memorySearchTheBadWay.ts +223 -0
- package/packages/utils/src/services/searchProvider/openSearch.spec.ts +926 -0
- package/packages/utils/src/services/searchProvider/openSearch.ts +195 -0
- package/{dist/esm/types.d.ts → packages/utils/src/types.ts} +34 -6
- package/packages/utils/tsconfig.json +31 -0
- package/packages/utils/tsconfig.without-specs.cjs.json +7 -0
- package/packages/utils/tsconfig.without-specs.esm.json +7 -0
- package/packages/utils/tsconfig.without-specs.json +6 -0
- package/scripts/build.bash +24 -0
- package/scripts/ci.bash +10 -0
- package/scripts/start.bash +29 -0
- package/dist/cjs/assertions/index.d.ts +0 -89
- package/dist/cjs/assertions/index.js +0 -157
- package/dist/cjs/aws/ssmService.d.ts +0 -5
- package/dist/cjs/aws/ssmService.js +0 -9
- package/dist/cjs/config/awsParameterConfig.d.ts +0 -10
- package/dist/cjs/config/awsParameterConfig.js +0 -26
- package/dist/cjs/config/envConfig.d.ts +0 -24
- package/dist/cjs/config/envConfig.js +0 -57
- package/dist/cjs/config/index.d.ts +0 -48
- package/dist/cjs/config/index.js +0 -35
- package/dist/cjs/config/lambdaParameterConfig.d.ts +0 -12
- package/dist/cjs/config/lambdaParameterConfig.js +0 -45
- package/dist/cjs/config/replaceConfig.d.ts +0 -14
- package/dist/cjs/config/replaceConfig.js +0 -22
- package/dist/cjs/config/resolveConfigValue.d.ts +0 -5
- package/dist/cjs/config/resolveConfigValue.js +0 -12
- package/dist/cjs/errors/index.d.ts +0 -88
- package/dist/cjs/errors/index.js +0 -123
- package/dist/cjs/fetch/fetchStatusRetry.d.ts +0 -8
- package/dist/cjs/fetch/fetchStatusRetry.js +0 -27
- package/dist/cjs/fetch/index.d.ts +0 -64
- package/dist/cjs/fetch/index.js +0 -55
- package/dist/cjs/guards/index.d.ts +0 -38
- package/dist/cjs/guards/index.js +0 -44
- package/dist/cjs/index.js +0 -20
- package/dist/cjs/middleware/apiErrorHandler.d.ts +0 -24
- package/dist/cjs/middleware/apiErrorHandler.js +0 -42
- package/dist/cjs/middleware/apiSlowResponseMiddleware.d.ts +0 -23
- package/dist/cjs/middleware/apiSlowResponseMiddleware.js +0 -54
- package/dist/cjs/middleware/index.js +0 -48
- package/dist/cjs/middleware/lambdaCorsResponseMiddleware.d.ts +0 -20
- package/dist/cjs/middleware/lambdaCorsResponseMiddleware.js +0 -44
- package/dist/cjs/middleware/throwNotFoundMiddleware.d.ts +0 -4
- package/dist/cjs/middleware/throwNotFoundMiddleware.js +0 -14
- package/dist/cjs/misc/hashValue.d.ts +0 -10
- package/dist/cjs/misc/hashValue.js +0 -17
- package/dist/cjs/misc/helpers.d.ts +0 -124
- package/dist/cjs/misc/helpers.js +0 -214
- package/dist/cjs/misc/merge.d.ts +0 -21
- package/dist/cjs/misc/merge.js +0 -45
- package/dist/cjs/misc/partitionSequence.d.ts +0 -35
- package/dist/cjs/misc/partitionSequence.js +0 -55
- package/dist/cjs/pagination/index.d.ts +0 -91
- package/dist/cjs/pagination/index.js +0 -83
- package/dist/cjs/routing/helpers.d.ts +0 -57
- package/dist/cjs/routing/helpers.js +0 -90
- package/dist/cjs/routing/index.d.ts +0 -290
- package/dist/cjs/routing/index.js +0 -295
- package/dist/cjs/routing/validators/zod.d.ts +0 -4
- package/dist/cjs/routing/validators/zod.js +0 -14
- package/dist/cjs/services/accountsGateway/index.d.ts +0 -92
- package/dist/cjs/services/accountsGateway/index.js +0 -138
- package/dist/cjs/services/apiGateway/index.d.ts +0 -68
- package/dist/cjs/services/apiGateway/index.js +0 -118
- package/dist/cjs/services/authProvider/browser.d.ts +0 -40
- package/dist/cjs/services/authProvider/browser.js +0 -155
- package/dist/cjs/services/authProvider/decryption.d.ts +0 -19
- package/dist/cjs/services/authProvider/decryption.js +0 -73
- package/dist/cjs/services/authProvider/index.d.ts +0 -63
- package/dist/cjs/services/authProvider/index.js +0 -34
- package/dist/cjs/services/authProvider/subrequest.d.ts +0 -13
- package/dist/cjs/services/authProvider/subrequest.js +0 -49
- package/dist/cjs/services/authProvider/utils/decryptAndVerify.d.ts +0 -28
- package/dist/cjs/services/authProvider/utils/decryptAndVerify.js +0 -91
- package/dist/cjs/services/authProvider/utils/embeddedAuthProvider.d.ts +0 -26
- package/dist/cjs/services/authProvider/utils/embeddedAuthProvider.js +0 -47
- package/dist/cjs/services/authProvider/utils/userRoleValidator.d.ts +0 -13
- package/dist/cjs/services/authProvider/utils/userRoleValidator.js +0 -37
- package/dist/cjs/services/authProvider/utils/userSubrequest.d.ts +0 -3
- package/dist/cjs/services/authProvider/utils/userSubrequest.js +0 -13
- package/dist/cjs/services/documentStore/dynamoEncoding.d.ts +0 -10
- package/dist/cjs/services/documentStore/dynamoEncoding.js +0 -52
- package/dist/cjs/services/documentStore/fileSystemAssert.d.ts +0 -1
- package/dist/cjs/services/documentStore/fileSystemAssert.js +0 -14
- package/dist/cjs/services/documentStore/index.js +0 -2
- package/dist/cjs/services/documentStore/unversioned/dynamodb.d.ts +0 -31
- package/dist/cjs/services/documentStore/unversioned/dynamodb.js +0 -233
- package/dist/cjs/services/documentStore/unversioned/file-system.d.ts +0 -32
- package/dist/cjs/services/documentStore/unversioned/file-system.js +0 -214
- package/dist/cjs/services/documentStore/unversioned/index.js +0 -2
- package/dist/cjs/services/documentStore/versioned/dynamodb.d.ts +0 -25
- package/dist/cjs/services/documentStore/versioned/dynamodb.js +0 -143
- package/dist/cjs/services/documentStore/versioned/file-system.d.ts +0 -25
- package/dist/cjs/services/documentStore/versioned/file-system.js +0 -73
- package/dist/cjs/services/documentStore/versioned/index.d.ts +0 -17
- package/dist/cjs/services/documentStore/versioned/index.js +0 -2
- package/dist/cjs/services/exercisesGateway/index.d.ts +0 -67
- package/dist/cjs/services/exercisesGateway/index.js +0 -107
- package/dist/cjs/services/fileServer/index.d.ts +0 -30
- package/dist/cjs/services/fileServer/index.js +0 -19
- package/dist/cjs/services/fileServer/localFileServer.d.ts +0 -13
- package/dist/cjs/services/fileServer/localFileServer.js +0 -132
- package/dist/cjs/services/fileServer/s3FileServer.d.ts +0 -14
- package/dist/cjs/services/fileServer/s3FileServer.js +0 -132
- package/dist/cjs/services/launchParams/index.js +0 -7
- package/dist/cjs/services/launchParams/signer.d.ts +0 -23
- package/dist/cjs/services/launchParams/signer.js +0 -58
- package/dist/cjs/services/launchParams/verifier.d.ts +0 -21
- package/dist/cjs/services/launchParams/verifier.js +0 -129
- package/dist/cjs/services/logger/console.d.ts +0 -4
- package/dist/cjs/services/logger/console.js +0 -12
- package/dist/cjs/services/logger/index.d.ts +0 -39
- package/dist/cjs/services/logger/index.js +0 -31
- package/dist/cjs/services/lrsGateway/addStatementDefaultFields.d.ts +0 -5
- package/dist/cjs/services/lrsGateway/addStatementDefaultFields.js +0 -21
- package/dist/cjs/services/lrsGateway/attempt-utils.d.ts +0 -70
- package/dist/cjs/services/lrsGateway/attempt-utils.js +0 -258
- package/dist/cjs/services/lrsGateway/file-system.d.ts +0 -15
- package/dist/cjs/services/lrsGateway/file-system.js +0 -150
- package/dist/cjs/services/lrsGateway/index.d.ts +0 -122
- package/dist/cjs/services/lrsGateway/index.js +0 -148
- package/dist/cjs/services/lrsGateway/xapiUtils.d.ts +0 -68
- package/dist/cjs/services/lrsGateway/xapiUtils.js +0 -109
- package/dist/cjs/services/postgresConnection/index.d.ts +0 -28
- package/dist/cjs/services/postgresConnection/index.js +0 -65
- package/dist/cjs/services/searchProvider/index.d.ts +0 -67
- package/dist/cjs/services/searchProvider/index.js +0 -2
- package/dist/cjs/services/searchProvider/memorySearchTheBadWay.d.ts +0 -20
- package/dist/cjs/services/searchProvider/memorySearchTheBadWay.js +0 -191
- package/dist/cjs/services/searchProvider/openSearch.d.ts +0 -28
- package/dist/cjs/services/searchProvider/openSearch.js +0 -154
- package/dist/cjs/tsconfig.without-specs.cjs.tsbuildinfo +0 -1
- package/dist/cjs/types.d.ts +0 -31
- package/dist/cjs/types.js +0 -2
- package/dist/esm/assertions/index.d.ts +0 -89
- package/dist/esm/aws/ssmService.d.ts +0 -5
- package/dist/esm/aws/ssmService.js +0 -6
- package/dist/esm/config/awsParameterConfig.d.ts +0 -10
- package/dist/esm/config/awsParameterConfig.js +0 -22
- package/dist/esm/config/envConfig.d.ts +0 -24
- package/dist/esm/config/envConfig.js +0 -53
- package/dist/esm/config/index.js +0 -17
- package/dist/esm/config/lambdaParameterConfig.d.ts +0 -12
- package/dist/esm/config/lambdaParameterConfig.js +0 -38
- package/dist/esm/config/replaceConfig.d.ts +0 -14
- package/dist/esm/config/resolveConfigValue.d.ts +0 -5
- package/dist/esm/config/resolveConfigValue.js +0 -8
- package/dist/esm/errors/index.d.ts +0 -88
- package/dist/esm/fetch/fetchStatusRetry.d.ts +0 -8
- package/dist/esm/fetch/fetchStatusRetry.js +0 -23
- package/dist/esm/fetch/index.d.ts +0 -64
- package/dist/esm/fetch/index.js +0 -46
- package/dist/esm/guards/index.js +0 -36
- package/dist/esm/index.d.ts +0 -4
- package/dist/esm/index.js +0 -4
- package/dist/esm/middleware/apiErrorHandler.d.ts +0 -24
- package/dist/esm/middleware/apiErrorHandler.js +0 -38
- package/dist/esm/middleware/apiSlowResponseMiddleware.d.ts +0 -23
- package/dist/esm/middleware/apiSlowResponseMiddleware.js +0 -50
- package/dist/esm/middleware/index.d.ts +0 -47
- package/dist/esm/middleware/index.js +0 -44
- package/dist/esm/middleware/lambdaCorsResponseMiddleware.d.ts +0 -20
- package/dist/esm/middleware/lambdaCorsResponseMiddleware.js +0 -40
- package/dist/esm/middleware/throwNotFoundMiddleware.d.ts +0 -4
- package/dist/esm/middleware/throwNotFoundMiddleware.js +0 -10
- package/dist/esm/misc/hashValue.d.ts +0 -10
- package/dist/esm/misc/hashValue.js +0 -13
- package/dist/esm/misc/helpers.d.ts +0 -124
- package/dist/esm/misc/helpers.js +0 -199
- package/dist/esm/misc/merge.d.ts +0 -21
- package/dist/esm/misc/merge.js +0 -40
- package/dist/esm/misc/partitionSequence.d.ts +0 -35
- package/dist/esm/pagination/index.d.ts +0 -91
- package/dist/esm/pagination/index.js +0 -77
- package/dist/esm/routing/helpers.d.ts +0 -57
- package/dist/esm/routing/index.d.ts +0 -290
- package/dist/esm/routing/index.js +0 -246
- package/dist/esm/routing/validators/zod.d.ts +0 -4
- package/dist/esm/routing/validators/zod.js +0 -10
- package/dist/esm/services/accountsGateway/index.d.ts +0 -92
- package/dist/esm/services/accountsGateway/index.js +0 -131
- package/dist/esm/services/apiGateway/index.d.ts +0 -68
- package/dist/esm/services/apiGateway/index.js +0 -77
- package/dist/esm/services/authProvider/browser.d.ts +0 -40
- package/dist/esm/services/authProvider/browser.js +0 -151
- package/dist/esm/services/authProvider/decryption.d.ts +0 -19
- package/dist/esm/services/authProvider/decryption.js +0 -69
- package/dist/esm/services/authProvider/index.d.ts +0 -63
- package/dist/esm/services/authProvider/index.js +0 -26
- package/dist/esm/services/authProvider/subrequest.d.ts +0 -13
- package/dist/esm/services/authProvider/subrequest.js +0 -45
- package/dist/esm/services/authProvider/utils/decryptAndVerify.d.ts +0 -28
- package/dist/esm/services/authProvider/utils/decryptAndVerify.js +0 -85
- package/dist/esm/services/authProvider/utils/embeddedAuthProvider.d.ts +0 -26
- package/dist/esm/services/authProvider/utils/embeddedAuthProvider.js +0 -40
- package/dist/esm/services/authProvider/utils/userRoleValidator.d.ts +0 -13
- package/dist/esm/services/authProvider/utils/userRoleValidator.js +0 -33
- package/dist/esm/services/authProvider/utils/userSubrequest.d.ts +0 -3
- package/dist/esm/services/authProvider/utils/userSubrequest.js +0 -6
- package/dist/esm/services/documentStore/dynamoEncoding.d.ts +0 -10
- package/dist/esm/services/documentStore/dynamoEncoding.js +0 -45
- package/dist/esm/services/documentStore/fileSystemAssert.d.ts +0 -1
- package/dist/esm/services/documentStore/fileSystemAssert.js +0 -10
- package/dist/esm/services/documentStore/index.d.ts +0 -14
- package/dist/esm/services/documentStore/index.js +0 -1
- package/dist/esm/services/documentStore/unversioned/dynamodb.d.ts +0 -31
- package/dist/esm/services/documentStore/unversioned/dynamodb.js +0 -226
- package/dist/esm/services/documentStore/unversioned/file-system.d.ts +0 -32
- package/dist/esm/services/documentStore/unversioned/file-system.js +0 -174
- package/dist/esm/services/documentStore/unversioned/index.d.ts +0 -2
- package/dist/esm/services/documentStore/unversioned/index.js +0 -1
- package/dist/esm/services/documentStore/versioned/dynamodb.d.ts +0 -25
- package/dist/esm/services/documentStore/versioned/dynamodb.js +0 -139
- package/dist/esm/services/documentStore/versioned/file-system.d.ts +0 -25
- package/dist/esm/services/documentStore/versioned/file-system.js +0 -69
- package/dist/esm/services/documentStore/versioned/index.d.ts +0 -17
- package/dist/esm/services/documentStore/versioned/index.js +0 -1
- package/dist/esm/services/exercisesGateway/index.d.ts +0 -67
- package/dist/esm/services/exercisesGateway/index.js +0 -70
- package/dist/esm/services/fileServer/index.d.ts +0 -30
- package/dist/esm/services/fileServer/index.js +0 -13
- package/dist/esm/services/fileServer/localFileServer.d.ts +0 -13
- package/dist/esm/services/fileServer/localFileServer.js +0 -125
- package/dist/esm/services/fileServer/s3FileServer.d.ts +0 -14
- package/dist/esm/services/fileServer/s3FileServer.js +0 -125
- package/dist/esm/services/launchParams/index.d.ts +0 -2
- package/dist/esm/services/launchParams/index.js +0 -2
- package/dist/esm/services/launchParams/signer.d.ts +0 -23
- package/dist/esm/services/launchParams/signer.js +0 -51
- package/dist/esm/services/launchParams/verifier.d.ts +0 -21
- package/dist/esm/services/launchParams/verifier.js +0 -92
- package/dist/esm/services/logger/console.d.ts +0 -4
- package/dist/esm/services/logger/index.js +0 -27
- package/dist/esm/services/lrsGateway/addStatementDefaultFields.d.ts +0 -5
- package/dist/esm/services/lrsGateway/addStatementDefaultFields.js +0 -14
- package/dist/esm/services/lrsGateway/attempt-utils.d.ts +0 -70
- package/dist/esm/services/lrsGateway/attempt-utils.js +0 -236
- package/dist/esm/services/lrsGateway/file-system.d.ts +0 -15
- package/dist/esm/services/lrsGateway/file-system.js +0 -110
- package/dist/esm/services/lrsGateway/index.d.ts +0 -122
- package/dist/esm/services/lrsGateway/index.js +0 -111
- package/dist/esm/services/lrsGateway/xapiUtils.d.ts +0 -68
- package/dist/esm/services/lrsGateway/xapiUtils.js +0 -99
- package/dist/esm/services/postgresConnection/index.d.ts +0 -28
- package/dist/esm/services/postgresConnection/index.js +0 -58
- package/dist/esm/services/searchProvider/index.d.ts +0 -67
- package/dist/esm/services/searchProvider/index.js +0 -1
- package/dist/esm/services/searchProvider/memorySearchTheBadWay.d.ts +0 -20
- package/dist/esm/services/searchProvider/memorySearchTheBadWay.js +0 -187
- package/dist/esm/services/searchProvider/openSearch.d.ts +0 -28
- package/dist/esm/services/searchProvider/openSearch.js +0 -150
- package/dist/esm/tsconfig.without-specs.esm.tsbuildinfo +0 -1
- package/dist/esm/types.js +0 -1
- package/script/bin/.init-params-script.bash.swp +0 -0
- /package/{script → packages/utils/script}/bin/copy-from-template.bash +0 -0
- /package/{script → packages/utils/script}/bin/delete-stack.bash +0 -0
- /package/{script → packages/utils/script}/bin/deploy.bash +0 -0
- /package/{script → packages/utils/script}/bin/destroy-deployment.bash +0 -0
- /package/{script → packages/utils/script}/bin/empty-bucket.bash +0 -0
- /package/{script → packages/utils/script}/bin/get-arg.bash +0 -0
- /package/{script → packages/utils/script}/bin/get-deployed-environments.bash +0 -0
- /package/{script → packages/utils/script}/bin/get-env-param.bash +0 -0
- /package/{script → packages/utils/script}/bin/get-kwarg.bash +0 -0
- /package/{script → packages/utils/script}/bin/get-stack-param.bash +0 -0
- /package/{script → packages/utils/script}/bin/has-flag.bash +0 -0
- /package/{script → packages/utils/script}/bin/init-constants-script.bash +0 -0
- /package/{script → packages/utils/script}/bin/init-params-script.bash +0 -0
- /package/{script → packages/utils/script}/bin/stack-exists.bash +0 -0
- /package/{script → packages/utils/script}/bin/update-utils.bash +0 -0
- /package/{script → packages/utils/script}/bin/upload-pager-duty-endpoints.bash +0 -0
- /package/{script → packages/utils/script}/bin/upload-params.bash +0 -0
- /package/{script → packages/utils/script}/bin/which.bash +0 -0
- /package/{script → packages/utils/script}/bin-entry.bash +0 -0
- /package/{script → packages/utils/script}/build.bash +0 -0
- /package/{dist/cjs/index.d.ts → packages/utils/src/index.ts} +0 -0
- /package/{dist/cjs/services/launchParams/index.d.ts → packages/utils/src/services/launchParams/index.ts} +0 -0
|
@@ -0,0 +1,259 @@
|
|
|
1
|
+
import type { Logger } from '../services/logger';
|
|
2
|
+
|
|
3
|
+
/*
|
|
4
|
+
* there was a reason i made these instead of using lodash/fp but i forget what it was. i think maybe
|
|
5
|
+
* these do more validation that the second function gets a compatible object.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
/**
|
|
9
|
+
* Returns a function that gets the value of the given key from an object
|
|
10
|
+
*
|
|
11
|
+
* @param key
|
|
12
|
+
* @example
|
|
13
|
+
* const getAuthor = getKeyValue('author');
|
|
14
|
+
* const author = getAuthor(book);
|
|
15
|
+
*/
|
|
16
|
+
export const getKeyValue = <K extends string>(key: K) => <O extends {[_key in K]?: any}>(obj: O): O[K] => obj[key];
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Returns a function that gets the value of the given key from an object, or the given default
|
|
20
|
+
* value if the key is not present.
|
|
21
|
+
*
|
|
22
|
+
* @param key
|
|
23
|
+
* @param defaultValue a default value matching the type of the value at the given key
|
|
24
|
+
* @example
|
|
25
|
+
* const getAuthorOrNope = getKeyValueOr('author', 'nope');
|
|
26
|
+
* const authorOrNope = getAuthorOrNope(book);
|
|
27
|
+
*/
|
|
28
|
+
export const getKeyValueOr = <K extends string, V, Od extends {[_key in K]?: any} = {[_key in K]?: any}>(key: K, defaultValue: V) => <O extends (Od extends undefined ? {[_key in K]?: any} : Od)>(obj: O): NonNullable<O[K]> | V => obj[key] || defaultValue;
|
|
29
|
+
|
|
30
|
+
/**
|
|
31
|
+
* Returns a function that sets the value of the given key on an object.
|
|
32
|
+
*
|
|
33
|
+
* @param key
|
|
34
|
+
* @example
|
|
35
|
+
* const putAuthor = putKeyValue('author');
|
|
36
|
+
* const newBook = putAuthor(book, 'tom');
|
|
37
|
+
*/
|
|
38
|
+
export const putKeyValue = <K extends string>(key: K) => <O extends {[_key in K]?: any}>(obj: O, value: O[K]): O => ({...obj, [key]: value});
|
|
39
|
+
|
|
40
|
+
/**
|
|
41
|
+
* Coerces a value into an array. If the value is already an array, it is returned as-is.
|
|
42
|
+
* If the value is undefined, an empty array is returned.
|
|
43
|
+
*
|
|
44
|
+
* @param thing the value
|
|
45
|
+
* @returns an array
|
|
46
|
+
*/
|
|
47
|
+
export const coerceArray = <T>(thing: undefined | T | T[]): T[] => thing instanceof Array ? thing : thing !== undefined ? [thing] : [];
|
|
48
|
+
|
|
49
|
+
type FlowFn<A, R> = (...args: [A]) => R;
|
|
50
|
+
type AnyFlowFn = FlowFn<any, any>;
|
|
51
|
+
type FlowFnResult<F, A> = F extends FlowFn<A, infer R> ? R : never;
|
|
52
|
+
type FlowResult<C, A> = C extends [infer F1, ...infer Cr]
|
|
53
|
+
? F1 extends AnyFlowFn
|
|
54
|
+
? Cr extends never[]
|
|
55
|
+
? FlowFnResult<F1, A>
|
|
56
|
+
: FlowResult<Cr, FlowFnResult<F1, A>>
|
|
57
|
+
: never
|
|
58
|
+
: never
|
|
59
|
+
;
|
|
60
|
+
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
|
61
|
+
type FlowChainArg<C> = C extends [infer F1, ...infer _]
|
|
62
|
+
? F1 extends FlowFn<infer A, any>
|
|
63
|
+
? A
|
|
64
|
+
: never
|
|
65
|
+
: never
|
|
66
|
+
;
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* this is like lodash/flow but it uses a recursive type instead of hard-coding parameters
|
|
70
|
+
*/
|
|
71
|
+
export const flow = <C extends AnyFlowFn[]>(...chain: C) => (param: FlowChainArg<C>): FlowResult<C, FlowChainArg<C>> => {
|
|
72
|
+
let result: any = param;
|
|
73
|
+
|
|
74
|
+
for (const fn of chain) {
|
|
75
|
+
result = fn(result);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
return result;
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
export type RetryOptions = {
|
|
82
|
+
logger?: Logger;
|
|
83
|
+
wait?: number;
|
|
84
|
+
splay?: number;
|
|
85
|
+
retries?: number;
|
|
86
|
+
n?: number;
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Retries a function with a delay between retries.
|
|
91
|
+
* @param fn The function to retry
|
|
92
|
+
* @param [options] retry options
|
|
93
|
+
* @param [options.wait=100] base wait of first retry (number), further retries are: t=n*wait,
|
|
94
|
+
* default: 100
|
|
95
|
+
* @param [options.splay=0.5] percentage to modify t by. 0.5 is 50%, which would make
|
|
96
|
+
* t=n*wait+rand(wait*splay*-1, wait*splay), default: 0.5
|
|
97
|
+
* @param [options.retries=2] number of times to retry. (2 retries means it'll try a max of 3 times),
|
|
98
|
+
* default: 2
|
|
99
|
+
* @param [options.n=0] the starting retry index. probably don't set this. unless you really want a very steep
|
|
100
|
+
* initial wait cliff with incremental increases, then maybe you'd set this. default: 0
|
|
101
|
+
* @returns A promise that resolves with the result of the function, or rejects with the last error.
|
|
102
|
+
*/
|
|
103
|
+
export const retryWithDelay = <R>(fn: (abort?: (e: Error) => never) => Promise<R>, options?: RetryOptions): Promise<R> => {
|
|
104
|
+
const {wait, splay, retries, n} = {wait: 100, splay: 0.5, retries: 2, n: 0, ...options};
|
|
105
|
+
|
|
106
|
+
if (n >= retries) {
|
|
107
|
+
return fn();
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
let aborted = false;
|
|
111
|
+
|
|
112
|
+
const timeout = (n+1)*wait+(Math.random() * 2 - 1) * splay * wait;
|
|
113
|
+
const retry = (e: Error) => new Promise<R>((resolve, reject) => {
|
|
114
|
+
if (aborted) {
|
|
115
|
+
reject(e);
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
options?.logger?.log(`failed try ${n + 1} of ${retries}. ${e.message}`);
|
|
119
|
+
setTimeout(() => retryWithDelay(fn, {...options, n: n+1}).then(resolve, reject), timeout);
|
|
120
|
+
}
|
|
121
|
+
});
|
|
122
|
+
const abort = (e: Error): never => {
|
|
123
|
+
aborted = true;
|
|
124
|
+
throw e;
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
return fn(abort).catch(retry);
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
/**
|
|
131
|
+
* a shameful helper to avoid needing to test code coverage of branches
|
|
132
|
+
*/
|
|
133
|
+
export const fnIf = <T1, T2>(condition: boolean, trueValue: T1, falseValue: T2) => condition ? trueValue : falseValue;
|
|
134
|
+
|
|
135
|
+
/**
|
|
136
|
+
* maps the array and returns the first result that matches the predicate
|
|
137
|
+
* avoids processing extra elements that would happen with .map().find() or .reduce
|
|
138
|
+
*
|
|
139
|
+
* eg the third element of the array is never processed:
|
|
140
|
+
* const result = mapFind([1,2,3], x => 'hello'.charAt(x), x => x === 'l');
|
|
141
|
+
*/
|
|
142
|
+
export const mapFind = <I, R>(array: I[], mapper: (item: I) => R, predicate: (result: R) => boolean = (r: any) => !!r): R | undefined => {
|
|
143
|
+
for (const item of array) {
|
|
144
|
+
const mapped = mapper(item);
|
|
145
|
+
|
|
146
|
+
if (predicate(mapped)) {
|
|
147
|
+
return mapped;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
};
|
|
151
|
+
|
|
152
|
+
/**
|
|
153
|
+
* returns a function that will only ever call the given function once, returning the first result for every subsequent call
|
|
154
|
+
*
|
|
155
|
+
* does not cache rejected promises, to avoid cache poisoning on failed async requests
|
|
156
|
+
*
|
|
157
|
+
* eg:
|
|
158
|
+
* const heavyFunction = () => 'hello';
|
|
159
|
+
* const butOnlyOnce = once(() => 'hello');
|
|
160
|
+
*
|
|
161
|
+
* heavyFunction() // returns `hello`;
|
|
162
|
+
* butOnlyOnce() // returns `hello`;
|
|
163
|
+
*/
|
|
164
|
+
export const once = <F extends (...args: any[]) => any>(fn: F): F => {
|
|
165
|
+
const initialValue = {} as any as ReturnType<F>;
|
|
166
|
+
let result: ReturnType<F> = initialValue;
|
|
167
|
+
return ((...args: Parameters<F>) => {
|
|
168
|
+
if (result !== initialValue) {
|
|
169
|
+
return result;
|
|
170
|
+
}
|
|
171
|
+
result = fn(...args);
|
|
172
|
+
if (typeof result === 'object' && result as object instanceof Promise) {
|
|
173
|
+
// Clear the result when possible but do not return a Promise that resolves to the initialValue
|
|
174
|
+
(result as Promise<any>).catch(() => result = initialValue);
|
|
175
|
+
}
|
|
176
|
+
// If this is a rejected Promise, it should be returned before catch() actually overwrites it
|
|
177
|
+
return result;
|
|
178
|
+
}) as F;
|
|
179
|
+
};
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* memoizes a function with any number of arguments
|
|
183
|
+
*
|
|
184
|
+
* does not cache rejected promises, to avoid cache poisoning on failed async requests
|
|
185
|
+
*/
|
|
186
|
+
export const memoize = <F extends (...args: any[]) => any>(fn: F): F => {
|
|
187
|
+
interface CacheLayer {
|
|
188
|
+
weakLayers?: WeakMap<any, CacheLayer>;
|
|
189
|
+
strongLayers?: Map<any, CacheLayer>;
|
|
190
|
+
result?: ReturnType<F>;
|
|
191
|
+
}
|
|
192
|
+
const cache: CacheLayer = {};
|
|
193
|
+
|
|
194
|
+
const resolveCache = (cacheLayer: CacheLayer, [first, ...rest]: any[]): CacheLayer => {
|
|
195
|
+
if (!first) {
|
|
196
|
+
return cacheLayer;
|
|
197
|
+
}
|
|
198
|
+
|
|
199
|
+
const layers = first instanceof Object
|
|
200
|
+
? (cacheLayer.weakLayers = (cacheLayer.weakLayers || (typeof WeakMap === 'undefined' ? undefined : new WeakMap())))
|
|
201
|
+
: (cacheLayer.strongLayers = (cacheLayer.strongLayers || new Map()))
|
|
202
|
+
;
|
|
203
|
+
|
|
204
|
+
// argument is an object and WeakMap is not supported
|
|
205
|
+
if (!layers) {
|
|
206
|
+
return {};
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
const layer = layers.get(first) || {};
|
|
210
|
+
|
|
211
|
+
if (!layers.has(first)) {
|
|
212
|
+
layers.set(first, layer);
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
return resolveCache(layer, rest);
|
|
216
|
+
};
|
|
217
|
+
|
|
218
|
+
return ((...args: Parameters<F>) => {
|
|
219
|
+
const thisCache = resolveCache(cache, args);
|
|
220
|
+
if ('result' in thisCache) {
|
|
221
|
+
return thisCache.result;
|
|
222
|
+
}
|
|
223
|
+
thisCache.result = fn(...args);
|
|
224
|
+
if (typeof thisCache.result === 'object' && thisCache.result as object instanceof Promise) {
|
|
225
|
+
// Clear the result when possible but do not return a Promise that resolves to the initialValue
|
|
226
|
+
(thisCache.result as Promise<any>).catch(() => delete thisCache.result);
|
|
227
|
+
}
|
|
228
|
+
return thisCache.result;
|
|
229
|
+
}) as F;
|
|
230
|
+
};
|
|
231
|
+
|
|
232
|
+
/**
|
|
233
|
+
* rounds number to given number of places
|
|
234
|
+
*
|
|
235
|
+
* eg:
|
|
236
|
+
* roundToPrecision(1234.123, 2); // returns 1200
|
|
237
|
+
* roundToPrecision(1234.123, -2); // returns 1234.12
|
|
238
|
+
* roundToPrecision(1234.125, -2); // returns 1234.13
|
|
239
|
+
*/
|
|
240
|
+
export const roundToPrecision = (num: number, places: number) => {
|
|
241
|
+
const multiplier = Math.pow(10, -1 * places);
|
|
242
|
+
// https://stackoverflow.com/a/11832950/14809536
|
|
243
|
+
return Math.round((num + Number.EPSILON) * multiplier) / multiplier;
|
|
244
|
+
};
|
|
245
|
+
|
|
246
|
+
/**
|
|
247
|
+
* a silly utility to help typescript realize an array is a tuple
|
|
248
|
+
*
|
|
249
|
+
* eg:
|
|
250
|
+
* const a = [5, 'string'] // type is `Array<string | number>`
|
|
251
|
+
* const t = tuple(5, 'string') type is `[5, 'string']`
|
|
252
|
+
*
|
|
253
|
+
* both have the same javascript value, but one is forced to be a tuple, which
|
|
254
|
+
* is nice if its structure is important. examples are like the React.useState
|
|
255
|
+
* pattern where there are two return values in a tuple, or if you're feeding
|
|
256
|
+
* Object.fromEntries
|
|
257
|
+
*
|
|
258
|
+
*/
|
|
259
|
+
export const tuple = <A extends any[]>(...args: A) => args;
|
|
@@ -0,0 +1,48 @@
|
|
|
1
|
+
import { isPlainObject } from '../guards';
|
|
2
|
+
import type { UnionToIntersection } from '../types';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Takes two objects and returns an array of the keys that are common to both, with a type
|
|
6
|
+
* limited to those keys.
|
|
7
|
+
*
|
|
8
|
+
* @param thing1 some object
|
|
9
|
+
* @param thing2 another object
|
|
10
|
+
* @returns an array of keys, type-limited to those keys
|
|
11
|
+
*/
|
|
12
|
+
export const getCommonProperties = <T1 extends {}, T2 extends {}>(thing1: T1, thing2: T2) =>
|
|
13
|
+
Object.keys(thing1).filter((key) => Object.keys(thing2).includes(key)) as Array<keyof T1 & keyof T2>;
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* recursive merge properties of inputs. values are merged if they are
|
|
17
|
+
* plain objects or arrays, otherwise if the same property exists in both
|
|
18
|
+
* objects the value from the second argument will win.
|
|
19
|
+
*
|
|
20
|
+
* unlike lodash merge, this will not change object references for values that
|
|
21
|
+
* exist only in one parameter.
|
|
22
|
+
*
|
|
23
|
+
* @example merge({thing: 'one'}, {thing: 'two', otherKey: 'one'}, {coolKey: 'coolValue'});
|
|
24
|
+
*/
|
|
25
|
+
export const merge = <T extends Array<{}>>(...[thing1, ...tail]: T): UnionToIntersection<T[number]> => {
|
|
26
|
+
|
|
27
|
+
const mergedTail = tail.length > 0
|
|
28
|
+
? merge(...tail)
|
|
29
|
+
: null;
|
|
30
|
+
|
|
31
|
+
if (!mergedTail) {
|
|
32
|
+
return thing1 as UnionToIntersection<T[number]>;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
return {
|
|
36
|
+
...thing1,
|
|
37
|
+
...mergedTail,
|
|
38
|
+
...getCommonProperties(thing1, mergedTail).reduce((result, property) => ({
|
|
39
|
+
...result,
|
|
40
|
+
...(isPlainObject(thing1[property]) && isPlainObject(mergedTail[property])
|
|
41
|
+
? {[property]: merge(thing1[property], mergedTail[property])}
|
|
42
|
+
: (Array.isArray(thing1[property]) && Array.isArray(mergedTail[property]))
|
|
43
|
+
? {[property]: [...thing1[property] as unknown as [], ...mergedTail[property] as unknown as []]}
|
|
44
|
+
: {}
|
|
45
|
+
),
|
|
46
|
+
}), {}),
|
|
47
|
+
} as UnionToIntersection<T[number]>;
|
|
48
|
+
};
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import deepEqual from 'deep-equal';
|
|
2
|
+
|
|
2
3
|
/**
|
|
3
4
|
* partitions a sequence based on a partition function returning {value: any; matches?: boolean}
|
|
4
5
|
* - if the function returns `matches` explicitly then adjacent matching elements will
|
|
@@ -30,19 +31,26 @@ import deepEqual from 'deep-equal';
|
|
|
30
31
|
* , [1,2,3,5,6,8]
|
|
31
32
|
* )
|
|
32
33
|
*/
|
|
33
|
-
export const partitionSequence =
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
34
|
+
export const partitionSequence = <T, P>(
|
|
35
|
+
getPartition: (thing: T, previous?: P) => {matches?: boolean; value: P},
|
|
36
|
+
sequence: T[]
|
|
37
|
+
) => {
|
|
38
|
+
const appendItem = (result: [P, T[]][], item: T): [P, T[]][] => {
|
|
39
|
+
const current = result[result.length - 1];
|
|
40
|
+
const itemPartition = getPartition(item, current?.[0]);
|
|
41
|
+
|
|
42
|
+
if (current && (
|
|
43
|
+
(itemPartition.matches === undefined && deepEqual(current[0], itemPartition.value))
|
|
44
|
+
|| itemPartition.matches
|
|
45
|
+
)) {
|
|
46
|
+
current[0] = itemPartition.value;
|
|
47
|
+
current[1].push(item);
|
|
48
|
+
} else {
|
|
49
|
+
result.push([itemPartition.value, [item]]);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
return result;
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
return sequence.reduce(appendItem, [] as [P, T[]][]);
|
|
48
56
|
};
|
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { getKeyValue, putKeyValue } from '..';
|
|
2
|
+
import { InvalidRequestError } from '../errors';
|
|
3
|
+
import { createPaginationMiddleware, loadMorePagination, pageNumberPagination } from '../pagination';
|
|
4
|
+
import { makeCreateRoute, QueryParams } from '../routing';
|
|
5
|
+
|
|
6
|
+
interface RouteRequest {
|
|
7
|
+
method: string;
|
|
8
|
+
path: string;
|
|
9
|
+
query: {[key: string]: string | undefined};
|
|
10
|
+
}
|
|
11
|
+
interface AppServices {
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
const createRoute = makeCreateRoute<AppServices, RouteRequest>();
|
|
15
|
+
const route = createRoute({name: 'testRoute', path: '/foobar', handler: (_: {}) => ({body: 'body'})});
|
|
16
|
+
|
|
17
|
+
describe('createPaginationMiddleware', () => {
|
|
18
|
+
it('removes unused query from request', () => {
|
|
19
|
+
const middleware = createPaginationMiddleware<RouteRequest>()({
|
|
20
|
+
getQueryParams: getKeyValue('query'),
|
|
21
|
+
setUnusedQueryParams: putKeyValue('query'),
|
|
22
|
+
paginator: jest.fn(() => ({getUnusedQueryParams: () => ({unused: 'query'})}))
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
const services = middleware()({request: {method: 'GET', path: '/foo', query: {some: 'query'}}}, {route, params: {}});
|
|
26
|
+
|
|
27
|
+
expect(services.request.query).toEqual({unused: 'query'});
|
|
28
|
+
});
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
describe('loadMorePaginationMiddleware', () => {
|
|
32
|
+
it('gives page token as string', () => {
|
|
33
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: 'foobar'}, {route, params: {}});
|
|
34
|
+
expect(pagination.getPageTokenString()).toBe('foobar');
|
|
35
|
+
});
|
|
36
|
+
|
|
37
|
+
it('gives page token as number', () => {
|
|
38
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: '7'}, {route, params: {}});
|
|
39
|
+
expect(pagination.getPageTokenNumber()).toBe(7);
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
it('gives page token as number (default)', () => {
|
|
43
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({}, {route, params: {}});
|
|
44
|
+
expect(pagination.getPageTokenNumber()).toBeUndefined();
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it('throws on invalid number', () => {
|
|
48
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: 'foobar'}, {route, params: {}});
|
|
49
|
+
expect(() => pagination.getPageTokenNumber()).toThrow();
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
it('builds a response', () => {
|
|
53
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: 'foobar'}, {route, params: {}});
|
|
54
|
+
expect(pagination.getPaginationResponse({items: [1,2,3], nextPageToken: undefined})).toEqual({
|
|
55
|
+
items: [1,2,3],
|
|
56
|
+
meta: {},
|
|
57
|
+
links: {},
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
it('builds a response with next page token', () => {
|
|
62
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: 'foobar'}, {route, params: {}});
|
|
63
|
+
expect(pagination.getPaginationResponse({items: [1,2,3], nextPageToken: 'next token'})).toEqual({
|
|
64
|
+
items: [1,2,3],
|
|
65
|
+
meta: {nextPageToken: 'next token'},
|
|
66
|
+
links: {nextPage: '/foobar?pageToken=next%20token'},
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
it('gets remaining query params', () => {
|
|
71
|
+
const pagination = loadMorePagination<typeof route, QueryParams>({pageToken: '7', otherParam: 'other'}, {route, params: {}});
|
|
72
|
+
expect(pagination.getUnusedQueryParams()).toEqual({otherParam: 'other'});
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
describe('pageNumberPaginationMiddleware', () => {
|
|
77
|
+
it('gives pagination params', () => {
|
|
78
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({page: '2'}, {route, params: {}});
|
|
79
|
+
expect(pagination.getPaginationParams()).toEqual({page: 2});
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it('gives pagination params (default)', () => {
|
|
83
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({}, {route, params: {}});
|
|
84
|
+
expect(pagination.getPaginationParams()).toEqual({page: undefined});
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
it('throws on invalid number', () => {
|
|
88
|
+
expect(() => pageNumberPagination<typeof route, QueryParams>({page: 'asdf'}, {route, params: {}})).toThrow(InvalidRequestError);
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
it('build pagination response', () => {
|
|
92
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({page: '2'}, {route, params: {}});
|
|
93
|
+
expect(pagination.getPaginationResponse({items: [1,2,3], pageSize: 5, currentPage: 2, totalItems: 15, totalPages: 3})).toEqual({
|
|
94
|
+
items: [1,2,3],
|
|
95
|
+
meta: {
|
|
96
|
+
pageSize: 5,
|
|
97
|
+
currentPage: 2,
|
|
98
|
+
totalItems: 15,
|
|
99
|
+
totalPages: 3,
|
|
100
|
+
},
|
|
101
|
+
links: {
|
|
102
|
+
firstPage: '/foobar?page=1',
|
|
103
|
+
lastPage: '/foobar?page=3',
|
|
104
|
+
nextPage: '/foobar?page=3',
|
|
105
|
+
prevPage: '/foobar?page=1',
|
|
106
|
+
}
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
|
|
110
|
+
it('omits previous when there isn\'t one', () => {
|
|
111
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({page: '1'}, {route, params: {}});
|
|
112
|
+
expect(pagination.getPaginationResponse({items: [1,2,3], pageSize: 5, currentPage: 1, totalItems: 15, totalPages: 3})).toEqual({
|
|
113
|
+
items: [1,2,3],
|
|
114
|
+
meta: {
|
|
115
|
+
pageSize: 5,
|
|
116
|
+
currentPage: 1,
|
|
117
|
+
totalItems: 15,
|
|
118
|
+
totalPages: 3,
|
|
119
|
+
},
|
|
120
|
+
links: {
|
|
121
|
+
firstPage: '/foobar?page=1',
|
|
122
|
+
lastPage: '/foobar?page=3',
|
|
123
|
+
nextPage: '/foobar?page=2',
|
|
124
|
+
}
|
|
125
|
+
});
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
it('omits next when there isn\'t one', () => {
|
|
129
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({page: '3'}, {route, params: {}});
|
|
130
|
+
expect(pagination.getPaginationResponse({items: [1,2,3], pageSize: 5, currentPage: 3, totalItems: 15, totalPages: 3})).toEqual({
|
|
131
|
+
items: [1,2,3],
|
|
132
|
+
meta: {
|
|
133
|
+
pageSize: 5,
|
|
134
|
+
currentPage: 3,
|
|
135
|
+
totalItems: 15,
|
|
136
|
+
totalPages: 3,
|
|
137
|
+
},
|
|
138
|
+
links: {
|
|
139
|
+
firstPage: '/foobar?page=1',
|
|
140
|
+
lastPage: '/foobar?page=3',
|
|
141
|
+
prevPage: '/foobar?page=2',
|
|
142
|
+
}
|
|
143
|
+
});
|
|
144
|
+
});
|
|
145
|
+
|
|
146
|
+
it('gets remaining query params', () => {
|
|
147
|
+
const pagination = pageNumberPagination<typeof route, QueryParams>({page: '3', otherParam: 'other'}, {route, params: {}});
|
|
148
|
+
expect(pagination.getUnusedQueryParams()).toEqual({otherParam: 'other'});
|
|
149
|
+
});
|
|
150
|
+
});
|
|
@@ -0,0 +1,117 @@
|
|
|
1
|
+
import { notNaN } from '../assertions';
|
|
2
|
+
import { InvalidRequestError } from '../errors';
|
|
3
|
+
import { QueryParams, renderAnyRouteUrl, RouteMatchRecord } from '../routing';
|
|
4
|
+
|
|
5
|
+
export type PaginationHandler<Pa, Q extends QueryParams> = <R>(queryParams: Q, match: RouteMatchRecord<R>) => Pa & {
|
|
6
|
+
getUnusedQueryParams: () => Q;
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
/**
|
|
10
|
+
* helper to create middleware with the given paginator. aside from taking care of annoying to write pagination logic, these helpers also make
|
|
11
|
+
* sure that all item list responses have the same formatting.
|
|
12
|
+
*
|
|
13
|
+
* eg:
|
|
14
|
+
* const getQueryParams = getKeyValueOr('queryStringParameters', {} as QueryParams);
|
|
15
|
+
* const setUnusedQueryParams = putKeyValue('queryStringParameters');
|
|
16
|
+
*
|
|
17
|
+
* export const loadMorePaginationMiddleware = createPaginationMiddleware<ApiRouteRequest>()({getQueryParams, setUnusedQueryParams, paginator: loadMorePagination});
|
|
18
|
+
* export const pageNumberPaginationMiddleware = createPaginationMiddleware<ApiRouteRequest>()({getQueryParams, setUnusedQueryParams, paginator: pageNumberPagination});
|
|
19
|
+
*
|
|
20
|
+
* eg the pagination middleware then provides your necessary inputs (getPageToken... in this case) and formats the response:
|
|
21
|
+
* const result = await services.myDocumentStore.getVersions(key, services.pagination.getPageTokenNumber());
|
|
22
|
+
*
|
|
23
|
+
* if (!result) {
|
|
24
|
+
* throw new NotFoundError('requested item not found');
|
|
25
|
+
* }
|
|
26
|
+
*
|
|
27
|
+
* return apiJsonResponse(200, services.pagination.getPaginationResponse(result));
|
|
28
|
+
*/
|
|
29
|
+
export const createPaginationMiddleware = <Ri, Q extends QueryParams = {[key: string]: string | undefined}, R = any>() =>
|
|
30
|
+
<Pa>({getQueryParams, setUnusedQueryParams, paginator}: {
|
|
31
|
+
getQueryParams: (request: Ri) => Q;
|
|
32
|
+
setUnusedQueryParams: (request: Ri, query: Q) => Ri;
|
|
33
|
+
paginator: PaginationHandler<Pa, Q>;
|
|
34
|
+
}) => <M extends {request: Ri}>() => (middleware: M, match: RouteMatchRecord<R>) => {
|
|
35
|
+
const queryParams = getQueryParams(middleware.request);
|
|
36
|
+
const pagination = paginator(queryParams, match);
|
|
37
|
+
|
|
38
|
+
// remove pagination params from downstream logic
|
|
39
|
+
middleware.request = setUnusedQueryParams(middleware.request, pagination.getUnusedQueryParams());
|
|
40
|
+
|
|
41
|
+
return {...middleware, pagination};
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
export const loadMorePagination = <R, Q extends QueryParams>(queryParams: Q, {route, params}: RouteMatchRecord<R>) => {
|
|
45
|
+
const {pageToken, ...otherParams} = queryParams;
|
|
46
|
+
|
|
47
|
+
return {
|
|
48
|
+
getUnusedQueryParams: () => otherParams,
|
|
49
|
+
getPageTokenString: () => pageToken,
|
|
50
|
+
getPageTokenNumber: () => pageToken && typeof pageToken === 'string'
|
|
51
|
+
? notNaN(parseInt(pageToken, 10), () => { throw new InvalidRequestError(); })
|
|
52
|
+
: undefined
|
|
53
|
+
,
|
|
54
|
+
getPaginationResponse: <T>({items, ...meta}: LoadMorePaginationResultInput<T>) => {
|
|
55
|
+
return {
|
|
56
|
+
items,
|
|
57
|
+
meta,
|
|
58
|
+
links: {
|
|
59
|
+
nextPage: meta.nextPageToken
|
|
60
|
+
? renderAnyRouteUrl(route, params, {...queryParams, pageToken: meta.nextPageToken.toString()})
|
|
61
|
+
: undefined,
|
|
62
|
+
}
|
|
63
|
+
};
|
|
64
|
+
},
|
|
65
|
+
};
|
|
66
|
+
};
|
|
67
|
+
|
|
68
|
+
/**
|
|
69
|
+
* if you're writing a data loader, this is what you need to return to be compatible with the loadMore style paginator.
|
|
70
|
+
* this is how the response formatter knows the token for the next page to put in the response metadata.
|
|
71
|
+
* */
|
|
72
|
+
export interface LoadMorePaginationResultInput<T> {
|
|
73
|
+
items: T[];
|
|
74
|
+
nextPageToken: string | number | undefined;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
export const pageNumberPagination = <R, Q extends QueryParams>(queryParams: Q, {route, params}: RouteMatchRecord<R>) => {
|
|
78
|
+
const {page, ...otherParams} = queryParams;
|
|
79
|
+
|
|
80
|
+
const numberPage = page && typeof page === 'string'
|
|
81
|
+
? notNaN(parseInt(page, 10), () => { throw new InvalidRequestError(); })
|
|
82
|
+
: undefined;
|
|
83
|
+
|
|
84
|
+
return {
|
|
85
|
+
getUnusedQueryParams: () => otherParams,
|
|
86
|
+
getPaginationParams: () => ({page: numberPage}),
|
|
87
|
+
getPaginationResponse: <T>({items, ...meta}: PageNumberPaginationResultInput<T>) => {
|
|
88
|
+
|
|
89
|
+
return {
|
|
90
|
+
items,
|
|
91
|
+
meta,
|
|
92
|
+
links: {
|
|
93
|
+
firstPage: renderAnyRouteUrl(route, params, {...queryParams, page: '1'}),
|
|
94
|
+
lastPage: renderAnyRouteUrl(route, params, {...queryParams, page: meta.totalPages.toString()}),
|
|
95
|
+
nextPage: meta.currentPage < meta.totalPages
|
|
96
|
+
? renderAnyRouteUrl(route, params, {...queryParams, page: (meta.currentPage + 1).toString()})
|
|
97
|
+
: undefined,
|
|
98
|
+
prevPage: meta.currentPage > 1
|
|
99
|
+
? renderAnyRouteUrl(route, params, {...queryParams, page: (meta.currentPage - 1).toString()})
|
|
100
|
+
: undefined
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
},
|
|
104
|
+
};
|
|
105
|
+
};
|
|
106
|
+
|
|
107
|
+
/**
|
|
108
|
+
* if you're writing a data loader, this is what you need to return to be compatible with the pageNumber style paginator.
|
|
109
|
+
* this is how the response formatter knows the information to put in the response metadata.
|
|
110
|
+
* */
|
|
111
|
+
export interface PageNumberPaginationResultInput<T> {
|
|
112
|
+
items: T[];
|
|
113
|
+
pageSize: number;
|
|
114
|
+
currentPage: number;
|
|
115
|
+
totalItems: number;
|
|
116
|
+
totalPages: number;
|
|
117
|
+
}
|