@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.
Files changed (504) hide show
  1. package/.cfnlintrc +2 -0
  2. package/.github/CODEOWNERS +1 -0
  3. package/.github/workflows/ci.yml +36 -0
  4. package/.github/workflows/lint.yml +55 -0
  5. package/.nvmrc +1 -0
  6. package/.syncignore +4 -0
  7. package/.syncpackrc +18 -0
  8. package/CONTRIBUTING.md +96 -0
  9. package/LICENSE +661 -0
  10. package/Procfile +1 -0
  11. package/README.md +62 -90
  12. package/app.json +23 -0
  13. package/cspell.json +32 -0
  14. package/deploy/constants.env +21 -0
  15. package/deploy/deploy.bash +157 -0
  16. package/deploy/deployment-alt-region.cfn.yml +70 -0
  17. package/deploy/deployment.cfn.yml +650 -0
  18. package/deploy/destroy-deployment.bash +23 -0
  19. package/deploy/shared.cfn.yml +94 -0
  20. package/docs/lambda-build.md +35 -0
  21. package/package.json +12 -228
  22. package/packages/frontend/README.md +46 -0
  23. package/packages/frontend/package.json +101 -0
  24. package/packages/frontend/public/favicon.ico +0 -0
  25. package/packages/frontend/public/index.html +107 -0
  26. package/packages/frontend/public/maintenance.html +59 -0
  27. package/packages/frontend/public/manifest.json +15 -0
  28. package/packages/frontend/public/robots.txt +3 -0
  29. package/packages/frontend/script/make-certificate.bash +49 -0
  30. package/packages/frontend/script/server/cli.js +11 -0
  31. package/packages/frontend/script/server/index.js +47 -0
  32. package/packages/frontend/script/start.bash +22 -0
  33. package/packages/frontend/script/trust-localhost.bash +7 -0
  34. package/packages/frontend/src/auth/authProvider.ts +10 -0
  35. package/packages/frontend/src/auth/useAuth.ts +33 -0
  36. package/packages/frontend/src/components/Pagination.tsx +26 -0
  37. package/packages/frontend/src/configProvider/index.ts +53 -0
  38. package/packages/frontend/src/configProvider/use.ts +41 -0
  39. package/packages/frontend/src/core/context/services.spec.tsx +39 -0
  40. package/packages/frontend/src/core/context/services.tsx +16 -0
  41. package/packages/frontend/src/core/index.spec.ts +7 -0
  42. package/packages/frontend/src/core/index.ts +20 -0
  43. package/packages/frontend/src/core/services.tsx +14 -0
  44. package/packages/frontend/src/core/types.ts +3 -0
  45. package/packages/frontend/src/example/api.ts +28 -0
  46. package/packages/frontend/src/example/components/Layout.tsx +23 -0
  47. package/packages/frontend/src/example/screens/Home.spec.tsx +68 -0
  48. package/packages/frontend/src/example/screens/Home.tsx +78 -0
  49. package/packages/frontend/src/example/screens/ThingList.spec.tsx +60 -0
  50. package/packages/frontend/src/example/screens/ThingList.tsx +75 -0
  51. package/packages/frontend/src/example/screens/ThingView.spec.tsx +71 -0
  52. package/packages/frontend/src/example/screens/ThingView.tsx +47 -0
  53. package/packages/frontend/src/example/screens/index.ts +9 -0
  54. package/packages/frontend/src/index.css +159 -0
  55. package/packages/frontend/src/index.tsx +67 -0
  56. package/packages/frontend/src/react-app-env.d.ts +1 -0
  57. package/packages/frontend/src/routing/components/RouteLink.spec.tsx +55 -0
  58. package/packages/frontend/src/routing/components/RouteLink.tsx +35 -0
  59. package/packages/frontend/src/routing/middleware.ts +6 -0
  60. package/packages/frontend/src/routing/useQuery.ts +14 -0
  61. package/packages/frontend/src/setupProxy.js +19 -0
  62. package/packages/frontend/src/setupTests.ts +9 -0
  63. package/packages/frontend/src/tests/testServices.tsx +23 -0
  64. package/packages/frontend/tsconfig.json +27 -0
  65. package/packages/lambda/.eslintrc.js +64 -0
  66. package/packages/lambda/jest-global-setup.js +3 -0
  67. package/packages/lambda/jest-setup-after-env.js +1 -0
  68. package/packages/lambda/jest.config.js +31 -0
  69. package/packages/lambda/jest.resolver.js +17 -0
  70. package/packages/lambda/package.json +68 -0
  71. package/packages/lambda/script/build.bash +19 -0
  72. package/packages/lambda/script/bundle-functions.bash +10 -0
  73. package/packages/lambda/script/lambdaLocalProxy.js +16 -0
  74. package/packages/lambda/script/lambdaLocalProxy.spec.ts +147 -0
  75. package/packages/lambda/script/utils/getRouteData.ts +7 -0
  76. package/packages/lambda/script/utils/routeDataLoader.js +8 -0
  77. package/packages/lambda/script/utils/routeDataLoader.spec.ts +8 -0
  78. package/packages/lambda/src/functions/serviceApi/core/index.ts +7 -0
  79. package/packages/lambda/src/functions/serviceApi/core/request.spec.ts +38 -0
  80. package/packages/lambda/src/functions/serviceApi/core/request.ts +42 -0
  81. package/packages/lambda/src/functions/serviceApi/core/routes.spec.ts +7 -0
  82. package/packages/lambda/src/functions/serviceApi/core/routes.ts +10 -0
  83. package/packages/lambda/src/functions/serviceApi/core/services.ts +9 -0
  84. package/packages/lambda/src/functions/serviceApi/core/types.ts +13 -0
  85. package/packages/lambda/src/functions/serviceApi/entry/lambda/https-xray.ts +4 -0
  86. package/packages/lambda/src/functions/serviceApi/entry/lambda/index.spec.ts +48 -0
  87. package/packages/lambda/src/functions/serviceApi/entry/lambda/index.ts +58 -0
  88. package/packages/lambda/src/functions/serviceApi/entry/lambda/services.ts +36 -0
  89. package/packages/lambda/src/functions/serviceApi/entry/local.ts +71 -0
  90. package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentSearchMiddleware.spec.ts +16 -0
  91. package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentSearchMiddleware.ts +41 -0
  92. package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentStoreMiddleware.spec.ts +78 -0
  93. package/packages/lambda/src/functions/serviceApi/versions/v0/example/documentStoreMiddleware.ts +70 -0
  94. package/packages/lambda/src/functions/serviceApi/versions/v0/example/routes.spec.ts +306 -0
  95. package/packages/lambda/src/functions/serviceApi/versions/v0/example/routes.ts +176 -0
  96. package/packages/lambda/src/functions/serviceApi/versions/v0/index.spec.ts +263 -0
  97. package/packages/lambda/src/functions/serviceApi/versions/v0/index.ts +134 -0
  98. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/authMiddleware.spec.ts +23 -0
  99. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/authMiddleware.ts +32 -0
  100. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/configMiddleware.spec.ts +10 -0
  101. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/configMiddleware.ts +7 -0
  102. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/frontendFileServerMiddleware.spec.ts +13 -0
  103. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/frontendFileServerMiddleware.ts +23 -0
  104. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/paginationMiddleware.spec.ts +9 -0
  105. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/paginationMiddleware.ts +9 -0
  106. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/searchMiddleware.spec.ts +12 -0
  107. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/searchMiddleware.ts +21 -0
  108. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/userRoleValidatorMiddleware.spec.ts +21 -0
  109. package/packages/lambda/src/functions/serviceApi/versions/v0/middleware/userRoleValidatorMiddleware.ts +18 -0
  110. package/packages/lambda/tsconfig.json +30 -0
  111. package/packages/lambda/webpack.config.js +97 -0
  112. package/packages/utils/.eslintrc.js +64 -0
  113. package/packages/utils/README.md +118 -0
  114. package/packages/utils/jest-global-setup.js +3 -0
  115. package/packages/utils/jest.config.js +25 -0
  116. package/packages/utils/jest.resolver.js +17 -0
  117. package/packages/utils/package.json +238 -0
  118. package/packages/utils/src/assertions/index.spec.ts +126 -0
  119. package/{dist/esm/assertions/index.js → packages/utils/src/assertions/index.ts} +64 -49
  120. package/packages/utils/src/aws/ssmService.ts +7 -0
  121. package/packages/utils/src/config/awsParameterConfig.ts +24 -0
  122. package/packages/utils/src/config/envConfig.ts +58 -0
  123. package/packages/utils/src/config/index.spec.ts +165 -0
  124. package/{dist/esm/config/index.d.ts → packages/utils/src/config/index.ts} +29 -13
  125. package/packages/utils/src/config/lambdaParameterConfig.ts +49 -0
  126. package/{dist/esm/config/replaceConfig.js → packages/utils/src/config/replaceConfig.ts} +16 -6
  127. package/packages/utils/src/config/resolveConfigValue.ts +10 -0
  128. package/packages/utils/src/errors/index.spec.ts +35 -0
  129. package/{dist/esm/errors/index.js → packages/utils/src/errors/index.ts} +57 -41
  130. package/packages/utils/src/fetch/fetchStatusRetry.spec.ts +197 -0
  131. package/packages/utils/src/fetch/fetchStatusRetry.ts +33 -0
  132. package/packages/utils/src/fetch/index.spec.ts +34 -0
  133. package/packages/utils/src/fetch/index.ts +87 -0
  134. package/packages/utils/src/guards/index.spec.ts +58 -0
  135. package/{dist/esm/guards/index.d.ts → packages/utils/src/guards/index.ts} +10 -7
  136. package/packages/utils/src/index.spec.ts +471 -0
  137. package/packages/utils/src/middleware/apiErrorHandler.spec.ts +65 -0
  138. package/packages/utils/src/middleware/apiErrorHandler.ts +67 -0
  139. package/packages/utils/src/middleware/apiSlowResponseMiddleware.spec.ts +184 -0
  140. package/packages/utils/src/middleware/apiSlowResponseMiddleware.ts +71 -0
  141. package/packages/utils/src/middleware/index.spec.ts +99 -0
  142. package/{dist/cjs/middleware/index.d.ts → packages/utils/src/middleware/index.ts} +53 -5
  143. package/packages/utils/src/middleware/lambdaCorsResponseMiddleware.spec.ts +103 -0
  144. package/packages/utils/src/middleware/lambdaCorsResponseMiddleware.ts +52 -0
  145. package/packages/utils/src/middleware/throwNotFoundMiddleware.spec.ts +20 -0
  146. package/packages/utils/src/middleware/throwNotFoundMiddleware.ts +11 -0
  147. package/packages/utils/src/misc/hashValue.ts +18 -0
  148. package/packages/utils/src/misc/helpers.ts +259 -0
  149. package/packages/utils/src/misc/merge.ts +48 -0
  150. package/{dist/esm/misc/partitionSequence.js → packages/utils/src/misc/partitionSequence.ts} +23 -15
  151. package/packages/utils/src/pagination/index.spec.ts +150 -0
  152. package/packages/utils/src/pagination/index.ts +117 -0
  153. package/{dist/esm/routing/helpers.js → packages/utils/src/routing/helpers.ts} +42 -30
  154. package/packages/utils/src/routing/index.spec.ts +553 -0
  155. package/packages/utils/src/routing/index.ts +424 -0
  156. package/packages/utils/src/routing/validators/zod.spec.ts +16 -0
  157. package/packages/utils/src/routing/validators/zod.ts +14 -0
  158. package/packages/utils/src/services/accountsGateway/README.md +3 -0
  159. package/packages/utils/src/services/accountsGateway/index.spec.ts +518 -0
  160. package/packages/utils/src/services/accountsGateway/index.ts +251 -0
  161. package/packages/utils/src/services/apiGateway/README.md +93 -0
  162. package/packages/utils/src/services/apiGateway/index.spec.ts +254 -0
  163. package/packages/utils/src/services/apiGateway/index.ts +189 -0
  164. package/packages/utils/src/services/authProvider/README.md +21 -0
  165. package/packages/utils/src/services/authProvider/browser.spec.ts +391 -0
  166. package/packages/utils/src/services/authProvider/browser.ts +209 -0
  167. package/packages/utils/src/services/authProvider/decryption.spec.ts +337 -0
  168. package/packages/utils/src/services/authProvider/decryption.ts +98 -0
  169. package/packages/utils/src/services/authProvider/index.ts +93 -0
  170. package/packages/utils/src/services/authProvider/stub.spec.ts +29 -0
  171. package/packages/utils/src/services/authProvider/subrequest.spec.ts +105 -0
  172. package/packages/utils/src/services/authProvider/subrequest.ts +68 -0
  173. package/packages/utils/src/services/authProvider/utils/decryptAndVerify.spec.ts +128 -0
  174. package/packages/utils/src/services/authProvider/utils/decryptAndVerify.ts +106 -0
  175. package/packages/utils/src/services/authProvider/utils/embeddedAuthProvider.spec.ts +26 -0
  176. package/packages/utils/src/services/authProvider/utils/embeddedAuthProvider.ts +57 -0
  177. package/packages/utils/src/services/authProvider/utils/userRoleValidator.spec.ts +135 -0
  178. package/packages/utils/src/services/authProvider/utils/userRoleValidator.ts +49 -0
  179. package/packages/utils/src/services/authProvider/utils/userSubrequest.spec.ts +26 -0
  180. package/packages/utils/src/services/authProvider/utils/userSubrequest.ts +10 -0
  181. package/packages/utils/src/services/documentStore/dynamoEncoding.ts +57 -0
  182. package/packages/utils/src/services/documentStore/fileSystemAssert.spec.ts +43 -0
  183. package/packages/utils/src/services/documentStore/fileSystemAssert.ts +10 -0
  184. package/{dist/cjs/services/documentStore/index.d.ts → packages/utils/src/services/documentStore/index.ts} +8 -8
  185. package/packages/utils/src/services/documentStore/unversioned/README.md +13 -0
  186. package/packages/utils/src/services/documentStore/unversioned/dynamodb.spec.ts +859 -0
  187. package/packages/utils/src/services/documentStore/unversioned/dynamodb.ts +243 -0
  188. package/packages/utils/src/services/documentStore/unversioned/file-system.spec.ts +629 -0
  189. package/packages/utils/src/services/documentStore/unversioned/file-system.ts +194 -0
  190. package/{dist/cjs/services/documentStore/unversioned/index.d.ts → packages/utils/src/services/documentStore/unversioned/index.ts} +2 -0
  191. package/packages/utils/src/services/documentStore/versioned/README.md +13 -0
  192. package/packages/utils/src/services/documentStore/versioned/dynamodb.spec.ts +376 -0
  193. package/packages/utils/src/services/documentStore/versioned/dynamodb.ts +167 -0
  194. package/packages/utils/src/services/documentStore/versioned/file-system.spec.ts +262 -0
  195. package/packages/utils/src/services/documentStore/versioned/file-system.ts +90 -0
  196. package/packages/utils/src/services/documentStore/versioned/index.ts +25 -0
  197. package/packages/utils/src/services/exercisesGateway/README.md +5 -0
  198. package/packages/utils/src/services/exercisesGateway/index.spec.ts +326 -0
  199. package/packages/utils/src/services/exercisesGateway/index.ts +163 -0
  200. package/packages/utils/src/services/fileServer/index.spec.ts +88 -0
  201. package/packages/utils/src/services/fileServer/index.ts +43 -0
  202. package/packages/utils/src/services/fileServer/localFileServer.spec.ts +182 -0
  203. package/packages/utils/src/services/fileServer/localFileServer.ts +159 -0
  204. package/packages/utils/src/services/fileServer/s3FileServer.spec.ts +266 -0
  205. package/packages/utils/src/services/fileServer/s3FileServer.ts +155 -0
  206. package/packages/utils/src/services/launchParams/index.spec.ts +366 -0
  207. package/packages/utils/src/services/launchParams/signer.ts +73 -0
  208. package/packages/utils/src/services/launchParams/verifier.ts +120 -0
  209. package/packages/utils/src/services/logger/console.spec.ts +29 -0
  210. package/{dist/esm/services/logger/console.js → packages/utils/src/services/logger/console.ts} +5 -2
  211. package/packages/utils/src/services/logger/index.spec.ts +65 -0
  212. package/{dist/esm/services/logger/index.d.ts → packages/utils/src/services/logger/index.ts} +23 -9
  213. package/packages/utils/src/services/lrsGateway/README.md +5 -0
  214. package/packages/utils/src/services/lrsGateway/addStatementDefaultFields.ts +22 -0
  215. package/packages/utils/src/services/lrsGateway/attempt-utils.spec.ts +847 -0
  216. package/packages/utils/src/services/lrsGateway/attempt-utils.ts +358 -0
  217. package/packages/utils/src/services/lrsGateway/file-system.spec.ts +363 -0
  218. package/packages/utils/src/services/lrsGateway/file-system.ts +165 -0
  219. package/packages/utils/src/services/lrsGateway/index.spec.ts +194 -0
  220. package/packages/utils/src/services/lrsGateway/index.ts +257 -0
  221. package/packages/utils/src/services/lrsGateway/xapiUtils.spec.ts +887 -0
  222. package/packages/utils/src/services/lrsGateway/xapiUtils.ts +262 -0
  223. package/packages/utils/src/services/postgresConnection/index.spec.ts +170 -0
  224. package/packages/utils/src/services/postgresConnection/index.ts +84 -0
  225. package/packages/utils/src/services/searchProvider/README.md +3 -0
  226. package/packages/utils/src/services/searchProvider/index.ts +59 -0
  227. package/packages/utils/src/services/searchProvider/memorySearchTheBadWay.spec.ts +526 -0
  228. package/packages/utils/src/services/searchProvider/memorySearchTheBadWay.ts +223 -0
  229. package/packages/utils/src/services/searchProvider/openSearch.spec.ts +926 -0
  230. package/packages/utils/src/services/searchProvider/openSearch.ts +195 -0
  231. package/{dist/esm/types.d.ts → packages/utils/src/types.ts} +34 -6
  232. package/packages/utils/tsconfig.json +31 -0
  233. package/packages/utils/tsconfig.without-specs.cjs.json +7 -0
  234. package/packages/utils/tsconfig.without-specs.esm.json +7 -0
  235. package/packages/utils/tsconfig.without-specs.json +6 -0
  236. package/scripts/build.bash +24 -0
  237. package/scripts/ci.bash +10 -0
  238. package/scripts/start.bash +29 -0
  239. package/dist/cjs/assertions/index.d.ts +0 -89
  240. package/dist/cjs/assertions/index.js +0 -157
  241. package/dist/cjs/aws/ssmService.d.ts +0 -5
  242. package/dist/cjs/aws/ssmService.js +0 -9
  243. package/dist/cjs/config/awsParameterConfig.d.ts +0 -10
  244. package/dist/cjs/config/awsParameterConfig.js +0 -26
  245. package/dist/cjs/config/envConfig.d.ts +0 -24
  246. package/dist/cjs/config/envConfig.js +0 -57
  247. package/dist/cjs/config/index.d.ts +0 -48
  248. package/dist/cjs/config/index.js +0 -35
  249. package/dist/cjs/config/lambdaParameterConfig.d.ts +0 -12
  250. package/dist/cjs/config/lambdaParameterConfig.js +0 -45
  251. package/dist/cjs/config/replaceConfig.d.ts +0 -14
  252. package/dist/cjs/config/replaceConfig.js +0 -22
  253. package/dist/cjs/config/resolveConfigValue.d.ts +0 -5
  254. package/dist/cjs/config/resolveConfigValue.js +0 -12
  255. package/dist/cjs/errors/index.d.ts +0 -88
  256. package/dist/cjs/errors/index.js +0 -123
  257. package/dist/cjs/fetch/fetchStatusRetry.d.ts +0 -8
  258. package/dist/cjs/fetch/fetchStatusRetry.js +0 -27
  259. package/dist/cjs/fetch/index.d.ts +0 -64
  260. package/dist/cjs/fetch/index.js +0 -55
  261. package/dist/cjs/guards/index.d.ts +0 -38
  262. package/dist/cjs/guards/index.js +0 -44
  263. package/dist/cjs/index.js +0 -20
  264. package/dist/cjs/middleware/apiErrorHandler.d.ts +0 -24
  265. package/dist/cjs/middleware/apiErrorHandler.js +0 -42
  266. package/dist/cjs/middleware/apiSlowResponseMiddleware.d.ts +0 -23
  267. package/dist/cjs/middleware/apiSlowResponseMiddleware.js +0 -54
  268. package/dist/cjs/middleware/index.js +0 -48
  269. package/dist/cjs/middleware/lambdaCorsResponseMiddleware.d.ts +0 -20
  270. package/dist/cjs/middleware/lambdaCorsResponseMiddleware.js +0 -44
  271. package/dist/cjs/middleware/throwNotFoundMiddleware.d.ts +0 -4
  272. package/dist/cjs/middleware/throwNotFoundMiddleware.js +0 -14
  273. package/dist/cjs/misc/hashValue.d.ts +0 -10
  274. package/dist/cjs/misc/hashValue.js +0 -17
  275. package/dist/cjs/misc/helpers.d.ts +0 -124
  276. package/dist/cjs/misc/helpers.js +0 -214
  277. package/dist/cjs/misc/merge.d.ts +0 -21
  278. package/dist/cjs/misc/merge.js +0 -45
  279. package/dist/cjs/misc/partitionSequence.d.ts +0 -35
  280. package/dist/cjs/misc/partitionSequence.js +0 -55
  281. package/dist/cjs/pagination/index.d.ts +0 -91
  282. package/dist/cjs/pagination/index.js +0 -83
  283. package/dist/cjs/routing/helpers.d.ts +0 -57
  284. package/dist/cjs/routing/helpers.js +0 -90
  285. package/dist/cjs/routing/index.d.ts +0 -290
  286. package/dist/cjs/routing/index.js +0 -295
  287. package/dist/cjs/routing/validators/zod.d.ts +0 -4
  288. package/dist/cjs/routing/validators/zod.js +0 -14
  289. package/dist/cjs/services/accountsGateway/index.d.ts +0 -92
  290. package/dist/cjs/services/accountsGateway/index.js +0 -138
  291. package/dist/cjs/services/apiGateway/index.d.ts +0 -68
  292. package/dist/cjs/services/apiGateway/index.js +0 -118
  293. package/dist/cjs/services/authProvider/browser.d.ts +0 -40
  294. package/dist/cjs/services/authProvider/browser.js +0 -155
  295. package/dist/cjs/services/authProvider/decryption.d.ts +0 -19
  296. package/dist/cjs/services/authProvider/decryption.js +0 -73
  297. package/dist/cjs/services/authProvider/index.d.ts +0 -63
  298. package/dist/cjs/services/authProvider/index.js +0 -34
  299. package/dist/cjs/services/authProvider/subrequest.d.ts +0 -13
  300. package/dist/cjs/services/authProvider/subrequest.js +0 -49
  301. package/dist/cjs/services/authProvider/utils/decryptAndVerify.d.ts +0 -28
  302. package/dist/cjs/services/authProvider/utils/decryptAndVerify.js +0 -91
  303. package/dist/cjs/services/authProvider/utils/embeddedAuthProvider.d.ts +0 -26
  304. package/dist/cjs/services/authProvider/utils/embeddedAuthProvider.js +0 -47
  305. package/dist/cjs/services/authProvider/utils/userRoleValidator.d.ts +0 -13
  306. package/dist/cjs/services/authProvider/utils/userRoleValidator.js +0 -37
  307. package/dist/cjs/services/authProvider/utils/userSubrequest.d.ts +0 -3
  308. package/dist/cjs/services/authProvider/utils/userSubrequest.js +0 -13
  309. package/dist/cjs/services/documentStore/dynamoEncoding.d.ts +0 -10
  310. package/dist/cjs/services/documentStore/dynamoEncoding.js +0 -52
  311. package/dist/cjs/services/documentStore/fileSystemAssert.d.ts +0 -1
  312. package/dist/cjs/services/documentStore/fileSystemAssert.js +0 -14
  313. package/dist/cjs/services/documentStore/index.js +0 -2
  314. package/dist/cjs/services/documentStore/unversioned/dynamodb.d.ts +0 -31
  315. package/dist/cjs/services/documentStore/unversioned/dynamodb.js +0 -233
  316. package/dist/cjs/services/documentStore/unversioned/file-system.d.ts +0 -32
  317. package/dist/cjs/services/documentStore/unversioned/file-system.js +0 -214
  318. package/dist/cjs/services/documentStore/unversioned/index.js +0 -2
  319. package/dist/cjs/services/documentStore/versioned/dynamodb.d.ts +0 -25
  320. package/dist/cjs/services/documentStore/versioned/dynamodb.js +0 -143
  321. package/dist/cjs/services/documentStore/versioned/file-system.d.ts +0 -25
  322. package/dist/cjs/services/documentStore/versioned/file-system.js +0 -73
  323. package/dist/cjs/services/documentStore/versioned/index.d.ts +0 -17
  324. package/dist/cjs/services/documentStore/versioned/index.js +0 -2
  325. package/dist/cjs/services/exercisesGateway/index.d.ts +0 -67
  326. package/dist/cjs/services/exercisesGateway/index.js +0 -107
  327. package/dist/cjs/services/fileServer/index.d.ts +0 -30
  328. package/dist/cjs/services/fileServer/index.js +0 -19
  329. package/dist/cjs/services/fileServer/localFileServer.d.ts +0 -13
  330. package/dist/cjs/services/fileServer/localFileServer.js +0 -132
  331. package/dist/cjs/services/fileServer/s3FileServer.d.ts +0 -14
  332. package/dist/cjs/services/fileServer/s3FileServer.js +0 -132
  333. package/dist/cjs/services/launchParams/index.js +0 -7
  334. package/dist/cjs/services/launchParams/signer.d.ts +0 -23
  335. package/dist/cjs/services/launchParams/signer.js +0 -58
  336. package/dist/cjs/services/launchParams/verifier.d.ts +0 -21
  337. package/dist/cjs/services/launchParams/verifier.js +0 -129
  338. package/dist/cjs/services/logger/console.d.ts +0 -4
  339. package/dist/cjs/services/logger/console.js +0 -12
  340. package/dist/cjs/services/logger/index.d.ts +0 -39
  341. package/dist/cjs/services/logger/index.js +0 -31
  342. package/dist/cjs/services/lrsGateway/addStatementDefaultFields.d.ts +0 -5
  343. package/dist/cjs/services/lrsGateway/addStatementDefaultFields.js +0 -21
  344. package/dist/cjs/services/lrsGateway/attempt-utils.d.ts +0 -70
  345. package/dist/cjs/services/lrsGateway/attempt-utils.js +0 -258
  346. package/dist/cjs/services/lrsGateway/file-system.d.ts +0 -15
  347. package/dist/cjs/services/lrsGateway/file-system.js +0 -150
  348. package/dist/cjs/services/lrsGateway/index.d.ts +0 -122
  349. package/dist/cjs/services/lrsGateway/index.js +0 -148
  350. package/dist/cjs/services/lrsGateway/xapiUtils.d.ts +0 -68
  351. package/dist/cjs/services/lrsGateway/xapiUtils.js +0 -109
  352. package/dist/cjs/services/postgresConnection/index.d.ts +0 -28
  353. package/dist/cjs/services/postgresConnection/index.js +0 -65
  354. package/dist/cjs/services/searchProvider/index.d.ts +0 -67
  355. package/dist/cjs/services/searchProvider/index.js +0 -2
  356. package/dist/cjs/services/searchProvider/memorySearchTheBadWay.d.ts +0 -20
  357. package/dist/cjs/services/searchProvider/memorySearchTheBadWay.js +0 -191
  358. package/dist/cjs/services/searchProvider/openSearch.d.ts +0 -28
  359. package/dist/cjs/services/searchProvider/openSearch.js +0 -154
  360. package/dist/cjs/tsconfig.without-specs.cjs.tsbuildinfo +0 -1
  361. package/dist/cjs/types.d.ts +0 -31
  362. package/dist/cjs/types.js +0 -2
  363. package/dist/esm/assertions/index.d.ts +0 -89
  364. package/dist/esm/aws/ssmService.d.ts +0 -5
  365. package/dist/esm/aws/ssmService.js +0 -6
  366. package/dist/esm/config/awsParameterConfig.d.ts +0 -10
  367. package/dist/esm/config/awsParameterConfig.js +0 -22
  368. package/dist/esm/config/envConfig.d.ts +0 -24
  369. package/dist/esm/config/envConfig.js +0 -53
  370. package/dist/esm/config/index.js +0 -17
  371. package/dist/esm/config/lambdaParameterConfig.d.ts +0 -12
  372. package/dist/esm/config/lambdaParameterConfig.js +0 -38
  373. package/dist/esm/config/replaceConfig.d.ts +0 -14
  374. package/dist/esm/config/resolveConfigValue.d.ts +0 -5
  375. package/dist/esm/config/resolveConfigValue.js +0 -8
  376. package/dist/esm/errors/index.d.ts +0 -88
  377. package/dist/esm/fetch/fetchStatusRetry.d.ts +0 -8
  378. package/dist/esm/fetch/fetchStatusRetry.js +0 -23
  379. package/dist/esm/fetch/index.d.ts +0 -64
  380. package/dist/esm/fetch/index.js +0 -46
  381. package/dist/esm/guards/index.js +0 -36
  382. package/dist/esm/index.d.ts +0 -4
  383. package/dist/esm/index.js +0 -4
  384. package/dist/esm/middleware/apiErrorHandler.d.ts +0 -24
  385. package/dist/esm/middleware/apiErrorHandler.js +0 -38
  386. package/dist/esm/middleware/apiSlowResponseMiddleware.d.ts +0 -23
  387. package/dist/esm/middleware/apiSlowResponseMiddleware.js +0 -50
  388. package/dist/esm/middleware/index.d.ts +0 -47
  389. package/dist/esm/middleware/index.js +0 -44
  390. package/dist/esm/middleware/lambdaCorsResponseMiddleware.d.ts +0 -20
  391. package/dist/esm/middleware/lambdaCorsResponseMiddleware.js +0 -40
  392. package/dist/esm/middleware/throwNotFoundMiddleware.d.ts +0 -4
  393. package/dist/esm/middleware/throwNotFoundMiddleware.js +0 -10
  394. package/dist/esm/misc/hashValue.d.ts +0 -10
  395. package/dist/esm/misc/hashValue.js +0 -13
  396. package/dist/esm/misc/helpers.d.ts +0 -124
  397. package/dist/esm/misc/helpers.js +0 -199
  398. package/dist/esm/misc/merge.d.ts +0 -21
  399. package/dist/esm/misc/merge.js +0 -40
  400. package/dist/esm/misc/partitionSequence.d.ts +0 -35
  401. package/dist/esm/pagination/index.d.ts +0 -91
  402. package/dist/esm/pagination/index.js +0 -77
  403. package/dist/esm/routing/helpers.d.ts +0 -57
  404. package/dist/esm/routing/index.d.ts +0 -290
  405. package/dist/esm/routing/index.js +0 -246
  406. package/dist/esm/routing/validators/zod.d.ts +0 -4
  407. package/dist/esm/routing/validators/zod.js +0 -10
  408. package/dist/esm/services/accountsGateway/index.d.ts +0 -92
  409. package/dist/esm/services/accountsGateway/index.js +0 -131
  410. package/dist/esm/services/apiGateway/index.d.ts +0 -68
  411. package/dist/esm/services/apiGateway/index.js +0 -77
  412. package/dist/esm/services/authProvider/browser.d.ts +0 -40
  413. package/dist/esm/services/authProvider/browser.js +0 -151
  414. package/dist/esm/services/authProvider/decryption.d.ts +0 -19
  415. package/dist/esm/services/authProvider/decryption.js +0 -69
  416. package/dist/esm/services/authProvider/index.d.ts +0 -63
  417. package/dist/esm/services/authProvider/index.js +0 -26
  418. package/dist/esm/services/authProvider/subrequest.d.ts +0 -13
  419. package/dist/esm/services/authProvider/subrequest.js +0 -45
  420. package/dist/esm/services/authProvider/utils/decryptAndVerify.d.ts +0 -28
  421. package/dist/esm/services/authProvider/utils/decryptAndVerify.js +0 -85
  422. package/dist/esm/services/authProvider/utils/embeddedAuthProvider.d.ts +0 -26
  423. package/dist/esm/services/authProvider/utils/embeddedAuthProvider.js +0 -40
  424. package/dist/esm/services/authProvider/utils/userRoleValidator.d.ts +0 -13
  425. package/dist/esm/services/authProvider/utils/userRoleValidator.js +0 -33
  426. package/dist/esm/services/authProvider/utils/userSubrequest.d.ts +0 -3
  427. package/dist/esm/services/authProvider/utils/userSubrequest.js +0 -6
  428. package/dist/esm/services/documentStore/dynamoEncoding.d.ts +0 -10
  429. package/dist/esm/services/documentStore/dynamoEncoding.js +0 -45
  430. package/dist/esm/services/documentStore/fileSystemAssert.d.ts +0 -1
  431. package/dist/esm/services/documentStore/fileSystemAssert.js +0 -10
  432. package/dist/esm/services/documentStore/index.d.ts +0 -14
  433. package/dist/esm/services/documentStore/index.js +0 -1
  434. package/dist/esm/services/documentStore/unversioned/dynamodb.d.ts +0 -31
  435. package/dist/esm/services/documentStore/unversioned/dynamodb.js +0 -226
  436. package/dist/esm/services/documentStore/unversioned/file-system.d.ts +0 -32
  437. package/dist/esm/services/documentStore/unversioned/file-system.js +0 -174
  438. package/dist/esm/services/documentStore/unversioned/index.d.ts +0 -2
  439. package/dist/esm/services/documentStore/unversioned/index.js +0 -1
  440. package/dist/esm/services/documentStore/versioned/dynamodb.d.ts +0 -25
  441. package/dist/esm/services/documentStore/versioned/dynamodb.js +0 -139
  442. package/dist/esm/services/documentStore/versioned/file-system.d.ts +0 -25
  443. package/dist/esm/services/documentStore/versioned/file-system.js +0 -69
  444. package/dist/esm/services/documentStore/versioned/index.d.ts +0 -17
  445. package/dist/esm/services/documentStore/versioned/index.js +0 -1
  446. package/dist/esm/services/exercisesGateway/index.d.ts +0 -67
  447. package/dist/esm/services/exercisesGateway/index.js +0 -70
  448. package/dist/esm/services/fileServer/index.d.ts +0 -30
  449. package/dist/esm/services/fileServer/index.js +0 -13
  450. package/dist/esm/services/fileServer/localFileServer.d.ts +0 -13
  451. package/dist/esm/services/fileServer/localFileServer.js +0 -125
  452. package/dist/esm/services/fileServer/s3FileServer.d.ts +0 -14
  453. package/dist/esm/services/fileServer/s3FileServer.js +0 -125
  454. package/dist/esm/services/launchParams/index.d.ts +0 -2
  455. package/dist/esm/services/launchParams/index.js +0 -2
  456. package/dist/esm/services/launchParams/signer.d.ts +0 -23
  457. package/dist/esm/services/launchParams/signer.js +0 -51
  458. package/dist/esm/services/launchParams/verifier.d.ts +0 -21
  459. package/dist/esm/services/launchParams/verifier.js +0 -92
  460. package/dist/esm/services/logger/console.d.ts +0 -4
  461. package/dist/esm/services/logger/index.js +0 -27
  462. package/dist/esm/services/lrsGateway/addStatementDefaultFields.d.ts +0 -5
  463. package/dist/esm/services/lrsGateway/addStatementDefaultFields.js +0 -14
  464. package/dist/esm/services/lrsGateway/attempt-utils.d.ts +0 -70
  465. package/dist/esm/services/lrsGateway/attempt-utils.js +0 -236
  466. package/dist/esm/services/lrsGateway/file-system.d.ts +0 -15
  467. package/dist/esm/services/lrsGateway/file-system.js +0 -110
  468. package/dist/esm/services/lrsGateway/index.d.ts +0 -122
  469. package/dist/esm/services/lrsGateway/index.js +0 -111
  470. package/dist/esm/services/lrsGateway/xapiUtils.d.ts +0 -68
  471. package/dist/esm/services/lrsGateway/xapiUtils.js +0 -99
  472. package/dist/esm/services/postgresConnection/index.d.ts +0 -28
  473. package/dist/esm/services/postgresConnection/index.js +0 -58
  474. package/dist/esm/services/searchProvider/index.d.ts +0 -67
  475. package/dist/esm/services/searchProvider/index.js +0 -1
  476. package/dist/esm/services/searchProvider/memorySearchTheBadWay.d.ts +0 -20
  477. package/dist/esm/services/searchProvider/memorySearchTheBadWay.js +0 -187
  478. package/dist/esm/services/searchProvider/openSearch.d.ts +0 -28
  479. package/dist/esm/services/searchProvider/openSearch.js +0 -150
  480. package/dist/esm/tsconfig.without-specs.esm.tsbuildinfo +0 -1
  481. package/dist/esm/types.js +0 -1
  482. package/script/bin/.init-params-script.bash.swp +0 -0
  483. /package/{script → packages/utils/script}/bin/copy-from-template.bash +0 -0
  484. /package/{script → packages/utils/script}/bin/delete-stack.bash +0 -0
  485. /package/{script → packages/utils/script}/bin/deploy.bash +0 -0
  486. /package/{script → packages/utils/script}/bin/destroy-deployment.bash +0 -0
  487. /package/{script → packages/utils/script}/bin/empty-bucket.bash +0 -0
  488. /package/{script → packages/utils/script}/bin/get-arg.bash +0 -0
  489. /package/{script → packages/utils/script}/bin/get-deployed-environments.bash +0 -0
  490. /package/{script → packages/utils/script}/bin/get-env-param.bash +0 -0
  491. /package/{script → packages/utils/script}/bin/get-kwarg.bash +0 -0
  492. /package/{script → packages/utils/script}/bin/get-stack-param.bash +0 -0
  493. /package/{script → packages/utils/script}/bin/has-flag.bash +0 -0
  494. /package/{script → packages/utils/script}/bin/init-constants-script.bash +0 -0
  495. /package/{script → packages/utils/script}/bin/init-params-script.bash +0 -0
  496. /package/{script → packages/utils/script}/bin/stack-exists.bash +0 -0
  497. /package/{script → packages/utils/script}/bin/update-utils.bash +0 -0
  498. /package/{script → packages/utils/script}/bin/upload-pager-duty-endpoints.bash +0 -0
  499. /package/{script → packages/utils/script}/bin/upload-params.bash +0 -0
  500. /package/{script → packages/utils/script}/bin/which.bash +0 -0
  501. /package/{script → packages/utils/script}/bin-entry.bash +0 -0
  502. /package/{script → packages/utils/script}/build.bash +0 -0
  503. /package/{dist/cjs/index.d.ts → packages/utils/src/index.ts} +0 -0
  504. /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 = (getPartition, sequence) => {
34
- const appendItem = (result, item) => {
35
- const current = result[result.length - 1];
36
- const itemPartition = getPartition(item, current === null || current === void 0 ? void 0 : current[0]);
37
- if (current && ((itemPartition.matches === undefined && deepEqual(current[0], itemPartition.value))
38
- || itemPartition.matches)) {
39
- current[0] = itemPartition.value;
40
- current[1].push(item);
41
- }
42
- else {
43
- result.push([itemPartition.value, [item]]);
44
- }
45
- return result;
46
- };
47
- return sequence.reduce(appendItem, []);
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
+ }