skuba 0.0.0-master-20230318044948

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 (563) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +67 -0
  3. package/config/eslint.js +3 -0
  4. package/config/jest.js +1 -0
  5. package/config/prettier.d.ts +3 -0
  6. package/config/prettier.js +5 -0
  7. package/config/tsconfig.json +9 -0
  8. package/jest/moduleNameMapper.js +58 -0
  9. package/jest/moduleNameMapper.test.ts +64 -0
  10. package/jest/transform.js +50 -0
  11. package/jest/transform.test.ts +11 -0
  12. package/jest/tsConfig.js +26 -0
  13. package/jest-preset.js +34 -0
  14. package/lib/api/buildkite/annotate.d.ts +26 -0
  15. package/lib/api/buildkite/annotate.js +49 -0
  16. package/lib/api/buildkite/annotate.js.map +7 -0
  17. package/lib/api/buildkite/index.d.ts +3 -0
  18. package/lib/api/buildkite/index.js +32 -0
  19. package/lib/api/buildkite/index.js.map +7 -0
  20. package/lib/api/buildkite/md.d.ts +6 -0
  21. package/lib/api/buildkite/md.js +33 -0
  22. package/lib/api/buildkite/md.js.map +7 -0
  23. package/lib/api/git/commit.d.ts +15 -0
  24. package/lib/api/git/commit.js +52 -0
  25. package/lib/api/git/commit.js.map +7 -0
  26. package/lib/api/git/commitAllChanges.d.ts +19 -0
  27. package/lib/api/git/commitAllChanges.js +65 -0
  28. package/lib/api/git/commitAllChanges.js.map +7 -0
  29. package/lib/api/git/currentBranch.d.ts +10 -0
  30. package/lib/api/git/currentBranch.js +58 -0
  31. package/lib/api/git/currentBranch.js.map +7 -0
  32. package/lib/api/git/getChangedFiles.d.ts +20 -0
  33. package/lib/api/git/getChangedFiles.js +63 -0
  34. package/lib/api/git/getChangedFiles.js.map +7 -0
  35. package/lib/api/git/index.d.ts +10 -0
  36. package/lib/api/git/index.js +55 -0
  37. package/lib/api/git/index.js.map +7 -0
  38. package/lib/api/git/log.d.ts +19 -0
  39. package/lib/api/git/log.js +71 -0
  40. package/lib/api/git/log.js.map +7 -0
  41. package/lib/api/git/pull.d.ts +35 -0
  42. package/lib/api/git/pull.js +69 -0
  43. package/lib/api/git/pull.js.map +7 -0
  44. package/lib/api/git/push.d.ts +43 -0
  45. package/lib/api/git/push.js +70 -0
  46. package/lib/api/git/push.js.map +7 -0
  47. package/lib/api/git/remote.d.ts +20 -0
  48. package/lib/api/git/remote.js +72 -0
  49. package/lib/api/git/remote.js.map +7 -0
  50. package/lib/api/git/reset.d.ts +12 -0
  51. package/lib/api/git/reset.js +62 -0
  52. package/lib/api/git/reset.js.map +7 -0
  53. package/lib/api/git/statusMatrix.d.ts +7 -0
  54. package/lib/api/git/statusMatrix.js +47 -0
  55. package/lib/api/git/statusMatrix.js.map +7 -0
  56. package/lib/api/github/checkRun.d.ts +44 -0
  57. package/lib/api/github/checkRun.js +81 -0
  58. package/lib/api/github/checkRun.js.map +7 -0
  59. package/lib/api/github/environment.d.ts +18 -0
  60. package/lib/api/github/environment.js +48 -0
  61. package/lib/api/github/environment.js.map +7 -0
  62. package/lib/api/github/index.d.ts +7 -0
  63. package/lib/api/github/index.js +48 -0
  64. package/lib/api/github/index.js.map +7 -0
  65. package/lib/api/github/issueComment.d.ts +59 -0
  66. package/lib/api/github/issueComment.js +83 -0
  67. package/lib/api/github/issueComment.js.map +7 -0
  68. package/lib/api/github/pullRequest.d.ts +21 -0
  69. package/lib/api/github/pullRequest.js +68 -0
  70. package/lib/api/github/pullRequest.js.map +7 -0
  71. package/lib/api/github/push.d.ts +79 -0
  72. package/lib/api/github/push.js +158 -0
  73. package/lib/api/github/push.js.map +7 -0
  74. package/lib/api/jest/index.d.ts +144 -0
  75. package/lib/api/jest/index.js +41 -0
  76. package/lib/api/jest/index.js.map +7 -0
  77. package/lib/api/net/compose.d.ts +4 -0
  78. package/lib/api/net/compose.js +50 -0
  79. package/lib/api/net/compose.js.map +7 -0
  80. package/lib/api/net/index.d.ts +1 -0
  81. package/lib/api/net/index.js +29 -0
  82. package/lib/api/net/index.js.map +7 -0
  83. package/lib/api/net/socket.d.ts +5 -0
  84. package/lib/api/net/socket.js +68 -0
  85. package/lib/api/net/socket.js.map +7 -0
  86. package/lib/api/net/waitFor.d.ts +22 -0
  87. package/lib/api/net/waitFor.js +40 -0
  88. package/lib/api/net/waitFor.js.map +7 -0
  89. package/lib/cli/adapter/eslint.d.ts +14 -0
  90. package/lib/cli/adapter/eslint.js +112 -0
  91. package/lib/cli/adapter/eslint.js.map +7 -0
  92. package/lib/cli/adapter/prettier.d.ts +45 -0
  93. package/lib/cli/adapter/prettier.js +171 -0
  94. package/lib/cli/adapter/prettier.js.map +7 -0
  95. package/lib/cli/build/args.d.ts +7 -0
  96. package/lib/cli/build/args.js +73 -0
  97. package/lib/cli/build/args.js.map +7 -0
  98. package/lib/cli/build/esbuild.d.ts +5 -0
  99. package/lib/cli/build/esbuild.js +133 -0
  100. package/lib/cli/build/esbuild.js.map +7 -0
  101. package/lib/cli/build/index.d.ts +1 -0
  102. package/lib/cli/build/index.js +73 -0
  103. package/lib/cli/build/index.js.map +7 -0
  104. package/lib/cli/build/tsc.d.ts +1 -0
  105. package/lib/cli/build/tsc.js +36 -0
  106. package/lib/cli/build/tsc.js.map +7 -0
  107. package/lib/cli/buildPackage.d.ts +1 -0
  108. package/lib/cli/buildPackage.js +56 -0
  109. package/lib/cli/buildPackage.js.map +7 -0
  110. package/lib/cli/configure/addEmptyExports.d.ts +6 -0
  111. package/lib/cli/configure/addEmptyExports.js +76 -0
  112. package/lib/cli/configure/addEmptyExports.js.map +7 -0
  113. package/lib/cli/configure/analyseConfiguration.d.ts +9 -0
  114. package/lib/cli/configure/analyseConfiguration.js +69 -0
  115. package/lib/cli/configure/analyseConfiguration.js.map +7 -0
  116. package/lib/cli/configure/analyseDependencies.d.ts +10 -0
  117. package/lib/cli/configure/analyseDependencies.js +136 -0
  118. package/lib/cli/configure/analyseDependencies.js.map +7 -0
  119. package/lib/cli/configure/analysis/diff.d.ts +1 -0
  120. package/lib/cli/configure/analysis/diff.js +45 -0
  121. package/lib/cli/configure/analysis/diff.js.map +7 -0
  122. package/lib/cli/configure/analysis/files.d.ts +1 -0
  123. package/lib/cli/configure/analysis/files.js +58 -0
  124. package/lib/cli/configure/analysis/files.js.map +7 -0
  125. package/lib/cli/configure/analysis/git.d.ts +1 -0
  126. package/lib/cli/configure/analysis/git.js +54 -0
  127. package/lib/cli/configure/analysis/git.js.map +7 -0
  128. package/lib/cli/configure/analysis/package.d.ts +13 -0
  129. package/lib/cli/configure/analysis/package.js +110 -0
  130. package/lib/cli/configure/analysis/package.js.map +7 -0
  131. package/lib/cli/configure/analysis/project.d.ts +3 -0
  132. package/lib/cli/configure/analysis/project.js +105 -0
  133. package/lib/cli/configure/analysis/project.js.map +7 -0
  134. package/lib/cli/configure/dependencies/index.d.ts +5 -0
  135. package/lib/cli/configure/dependencies/index.js +41 -0
  136. package/lib/cli/configure/dependencies/index.js.map +7 -0
  137. package/lib/cli/configure/dependencies/seekDatadogCustomMetrics.d.ts +2 -0
  138. package/lib/cli/configure/dependencies/seekDatadogCustomMetrics.js +59 -0
  139. package/lib/cli/configure/dependencies/seekDatadogCustomMetrics.js.map +7 -0
  140. package/lib/cli/configure/dependencies/seekKoala.d.ts +2 -0
  141. package/lib/cli/configure/dependencies/seekKoala.js +56 -0
  142. package/lib/cli/configure/dependencies/seekKoala.js.map +7 -0
  143. package/lib/cli/configure/dependencies/skuba.d.ts +2 -0
  144. package/lib/cli/configure/dependencies/skuba.js +52 -0
  145. package/lib/cli/configure/dependencies/skuba.js.map +7 -0
  146. package/lib/cli/configure/dependencies/skubaDeps.d.ts +2 -0
  147. package/lib/cli/configure/dependencies/skubaDeps.js +56 -0
  148. package/lib/cli/configure/dependencies/skubaDeps.js.map +7 -0
  149. package/lib/cli/configure/dependencies/skubaDive.d.ts +3 -0
  150. package/lib/cli/configure/dependencies/skubaDive.js +70 -0
  151. package/lib/cli/configure/dependencies/skubaDive.js.map +7 -0
  152. package/lib/cli/configure/ensureTemplateCompletion.d.ts +9 -0
  153. package/lib/cli/configure/ensureTemplateCompletion.js +77 -0
  154. package/lib/cli/configure/ensureTemplateCompletion.js.map +7 -0
  155. package/lib/cli/configure/getEntryPoint.d.ts +11 -0
  156. package/lib/cli/configure/getEntryPoint.js +70 -0
  157. package/lib/cli/configure/getEntryPoint.js.map +7 -0
  158. package/lib/cli/configure/getProjectType.d.ts +9 -0
  159. package/lib/cli/configure/getProjectType.js +52 -0
  160. package/lib/cli/configure/getProjectType.js.map +7 -0
  161. package/lib/cli/configure/index.d.ts +1 -0
  162. package/lib/cli/configure/index.js +148 -0
  163. package/lib/cli/configure/index.js.map +7 -0
  164. package/lib/cli/configure/modules/eslint.d.ts +2 -0
  165. package/lib/cli/configure/modules/eslint.js +61 -0
  166. package/lib/cli/configure/modules/eslint.js.map +7 -0
  167. package/lib/cli/configure/modules/ignore.d.ts +2 -0
  168. package/lib/cli/configure/modules/ignore.js +40 -0
  169. package/lib/cli/configure/modules/ignore.js.map +7 -0
  170. package/lib/cli/configure/modules/index.d.ts +2 -0
  171. package/lib/cli/configure/modules/index.js +54 -0
  172. package/lib/cli/configure/modules/index.js.map +7 -0
  173. package/lib/cli/configure/modules/jest.d.ts +2 -0
  174. package/lib/cli/configure/modules/jest.js +86 -0
  175. package/lib/cli/configure/modules/jest.js.map +7 -0
  176. package/lib/cli/configure/modules/nodemon.d.ts +2 -0
  177. package/lib/cli/configure/modules/nodemon.js +30 -0
  178. package/lib/cli/configure/modules/nodemon.js.map +7 -0
  179. package/lib/cli/configure/modules/package.d.ts +2 -0
  180. package/lib/cli/configure/modules/package.js +117 -0
  181. package/lib/cli/configure/modules/package.js.map +7 -0
  182. package/lib/cli/configure/modules/prettier.d.ts +2 -0
  183. package/lib/cli/configure/modules/prettier.js +52 -0
  184. package/lib/cli/configure/modules/prettier.js.map +7 -0
  185. package/lib/cli/configure/modules/renovate.d.ts +3 -0
  186. package/lib/cli/configure/modules/renovate.js +69 -0
  187. package/lib/cli/configure/modules/renovate.js.map +7 -0
  188. package/lib/cli/configure/modules/serverless.d.ts +2 -0
  189. package/lib/cli/configure/modules/serverless.js +36 -0
  190. package/lib/cli/configure/modules/serverless.js.map +7 -0
  191. package/lib/cli/configure/modules/skubaDive.d.ts +2 -0
  192. package/lib/cli/configure/modules/skubaDive.js +72 -0
  193. package/lib/cli/configure/modules/skubaDive.js.map +7 -0
  194. package/lib/cli/configure/modules/tsconfig.d.ts +2 -0
  195. package/lib/cli/configure/modules/tsconfig.js +87 -0
  196. package/lib/cli/configure/modules/tsconfig.js.map +7 -0
  197. package/lib/cli/configure/modules/tslint.d.ts +2 -0
  198. package/lib/cli/configure/modules/tslint.js +30 -0
  199. package/lib/cli/configure/modules/tslint.js.map +7 -0
  200. package/lib/cli/configure/patchRenovateConfig.d.ts +1 -0
  201. package/lib/cli/configure/patchRenovateConfig.js +132 -0
  202. package/lib/cli/configure/patchRenovateConfig.js.map +7 -0
  203. package/lib/cli/configure/processing/deleteFiles.d.ts +5 -0
  204. package/lib/cli/configure/processing/deleteFiles.js +31 -0
  205. package/lib/cli/configure/processing/deleteFiles.js.map +7 -0
  206. package/lib/cli/configure/processing/ignoreFile.d.ts +8 -0
  207. package/lib/cli/configure/processing/ignoreFile.js +74 -0
  208. package/lib/cli/configure/processing/ignoreFile.js.map +7 -0
  209. package/lib/cli/configure/processing/javascript.d.ts +2 -0
  210. package/lib/cli/configure/processing/javascript.js +38 -0
  211. package/lib/cli/configure/processing/javascript.js.map +7 -0
  212. package/lib/cli/configure/processing/json.d.ts +2 -0
  213. package/lib/cli/configure/processing/json.js +57 -0
  214. package/lib/cli/configure/processing/json.js.map +7 -0
  215. package/lib/cli/configure/processing/loadFiles.d.ts +5 -0
  216. package/lib/cli/configure/processing/loadFiles.js +31 -0
  217. package/lib/cli/configure/processing/loadFiles.js.map +7 -0
  218. package/lib/cli/configure/processing/module.d.ts +10 -0
  219. package/lib/cli/configure/processing/module.js +39 -0
  220. package/lib/cli/configure/processing/module.js.map +7 -0
  221. package/lib/cli/configure/processing/package.d.ts +73 -0
  222. package/lib/cli/configure/processing/package.js +87 -0
  223. package/lib/cli/configure/processing/package.js.map +7 -0
  224. package/lib/cli/configure/processing/prettier.d.ts +4 -0
  225. package/lib/cli/configure/processing/prettier.js +44 -0
  226. package/lib/cli/configure/processing/prettier.js.map +7 -0
  227. package/lib/cli/configure/processing/record.d.ts +11 -0
  228. package/lib/cli/configure/processing/record.js +65 -0
  229. package/lib/cli/configure/processing/record.js.map +7 -0
  230. package/lib/cli/configure/processing/typescript.d.ts +23 -0
  231. package/lib/cli/configure/processing/typescript.js +197 -0
  232. package/lib/cli/configure/processing/typescript.js.map +7 -0
  233. package/lib/cli/configure/refreshIgnoreFiles.d.ts +3 -0
  234. package/lib/cli/configure/refreshIgnoreFiles.js +78 -0
  235. package/lib/cli/configure/refreshIgnoreFiles.js.map +7 -0
  236. package/lib/cli/configure/types.d.ts +26 -0
  237. package/lib/cli/configure/types.js +17 -0
  238. package/lib/cli/configure/types.js.map +7 -0
  239. package/lib/cli/format.d.ts +1 -0
  240. package/lib/cli/format.js +70 -0
  241. package/lib/cli/format.js.map +7 -0
  242. package/lib/cli/help.d.ts +1 -0
  243. package/lib/cli/help.js +34 -0
  244. package/lib/cli/help.js.map +7 -0
  245. package/lib/cli/init/getConfig.d.ts +26 -0
  246. package/lib/cli/init/getConfig.js +285 -0
  247. package/lib/cli/init/getConfig.js.map +7 -0
  248. package/lib/cli/init/git.d.ts +7 -0
  249. package/lib/cli/init/git.js +82 -0
  250. package/lib/cli/init/git.js.map +7 -0
  251. package/lib/cli/init/index.d.ts +1 -0
  252. package/lib/cli/init/index.js +146 -0
  253. package/lib/cli/init/index.js.map +7 -0
  254. package/lib/cli/init/prompts.d.ts +45 -0
  255. package/lib/cli/init/prompts.js +97 -0
  256. package/lib/cli/init/prompts.js.map +7 -0
  257. package/lib/cli/init/types.d.ts +28 -0
  258. package/lib/cli/init/types.js +65 -0
  259. package/lib/cli/init/types.js.map +7 -0
  260. package/lib/cli/init/validation.d.ts +8 -0
  261. package/lib/cli/init/validation.js +43 -0
  262. package/lib/cli/init/validation.js.map +7 -0
  263. package/lib/cli/init/writePackageJson.d.ts +13 -0
  264. package/lib/cli/init/writePackageJson.js +58 -0
  265. package/lib/cli/init/writePackageJson.js.map +7 -0
  266. package/lib/cli/lint/annotate/buildkite/eslint.d.ts +2 -0
  267. package/lib/cli/lint/annotate/buildkite/eslint.js +40 -0
  268. package/lib/cli/lint/annotate/buildkite/eslint.js.map +7 -0
  269. package/lib/cli/lint/annotate/buildkite/index.d.ts +4 -0
  270. package/lib/cli/lint/annotate/buildkite/index.js +58 -0
  271. package/lib/cli/lint/annotate/buildkite/index.js.map +7 -0
  272. package/lib/cli/lint/annotate/buildkite/prettier.d.ts +2 -0
  273. package/lib/cli/lint/annotate/buildkite/prettier.js +47 -0
  274. package/lib/cli/lint/annotate/buildkite/prettier.js.map +7 -0
  275. package/lib/cli/lint/annotate/buildkite/tsc.d.ts +2 -0
  276. package/lib/cli/lint/annotate/buildkite/tsc.js +45 -0
  277. package/lib/cli/lint/annotate/buildkite/tsc.js.map +7 -0
  278. package/lib/cli/lint/annotate/github/eslint.d.ts +3 -0
  279. package/lib/cli/lint/annotate/github/eslint.js +45 -0
  280. package/lib/cli/lint/annotate/github/eslint.js.map +7 -0
  281. package/lib/cli/lint/annotate/github/index.d.ts +4 -0
  282. package/lib/cli/lint/annotate/github/index.js +63 -0
  283. package/lib/cli/lint/annotate/github/index.js.map +7 -0
  284. package/lib/cli/lint/annotate/github/prettier.d.ts +3 -0
  285. package/lib/cli/lint/annotate/github/prettier.js +39 -0
  286. package/lib/cli/lint/annotate/github/prettier.js.map +7 -0
  287. package/lib/cli/lint/annotate/github/tsc.d.ts +3 -0
  288. package/lib/cli/lint/annotate/github/tsc.js +63 -0
  289. package/lib/cli/lint/annotate/github/tsc.js.map +7 -0
  290. package/lib/cli/lint/annotate/index.d.ts +4 -0
  291. package/lib/cli/lint/annotate/index.js +36 -0
  292. package/lib/cli/lint/annotate/index.js.map +7 -0
  293. package/lib/cli/lint/autofix.d.ts +10 -0
  294. package/lib/cli/lint/autofix.js +189 -0
  295. package/lib/cli/lint/autofix.js.map +7 -0
  296. package/lib/cli/lint/eslint.d.ts +4 -0
  297. package/lib/cli/lint/eslint.js +55 -0
  298. package/lib/cli/lint/eslint.js.map +7 -0
  299. package/lib/cli/lint/external.d.ts +10 -0
  300. package/lib/cli/lint/external.js +120 -0
  301. package/lib/cli/lint/external.js.map +7 -0
  302. package/lib/cli/lint/index.d.ts +2 -0
  303. package/lib/cli/lint/index.js +49 -0
  304. package/lib/cli/lint/index.js.map +7 -0
  305. package/lib/cli/lint/internal.d.ts +1 -0
  306. package/lib/cli/lint/internal.js +64 -0
  307. package/lib/cli/lint/internal.js.map +7 -0
  308. package/lib/cli/lint/prettier.d.ts +4 -0
  309. package/lib/cli/lint/prettier.js +55 -0
  310. package/lib/cli/lint/prettier.js.map +7 -0
  311. package/lib/cli/lint/tsc.d.ts +2 -0
  312. package/lib/cli/lint/tsc.js +58 -0
  313. package/lib/cli/lint/tsc.js.map +7 -0
  314. package/lib/cli/lint/types.d.ts +34 -0
  315. package/lib/cli/lint/types.js +17 -0
  316. package/lib/cli/lint/types.js.map +7 -0
  317. package/lib/cli/node.d.ts +1 -0
  318. package/lib/cli/node.js +77 -0
  319. package/lib/cli/node.js.map +7 -0
  320. package/lib/cli/release.d.ts +1 -0
  321. package/lib/cli/release.js +32 -0
  322. package/lib/cli/release.js.map +7 -0
  323. package/lib/cli/start.d.ts +1 -0
  324. package/lib/cli/start.js +69 -0
  325. package/lib/cli/start.js.map +7 -0
  326. package/lib/cli/test/index.d.ts +1 -0
  327. package/lib/cli/test/index.js +37 -0
  328. package/lib/cli/test/index.js.map +7 -0
  329. package/lib/cli/test/reporters/github/annotations.d.ts +9 -0
  330. package/lib/cli/test/reporters/github/annotations.js +102 -0
  331. package/lib/cli/test/reporters/github/annotations.js.map +7 -0
  332. package/lib/cli/test/reporters/github/index.d.ts +5 -0
  333. package/lib/cli/test/reporters/github/index.js +77 -0
  334. package/lib/cli/test/reporters/github/index.js.map +7 -0
  335. package/lib/cli/version.d.ts +1 -0
  336. package/lib/cli/version.js +34 -0
  337. package/lib/cli/version.js.map +7 -0
  338. package/lib/enquirer.d.js +2 -0
  339. package/lib/enquirer.d.js.map +7 -0
  340. package/lib/index.d.ts +22 -0
  341. package/lib/index.js +51 -0
  342. package/lib/index.js.map +7 -0
  343. package/lib/skuba.d.ts +11 -0
  344. package/lib/skuba.js +55 -0
  345. package/lib/skuba.js.map +7 -0
  346. package/lib/utils/args.d.ts +43 -0
  347. package/lib/utils/args.js +114 -0
  348. package/lib/utils/args.js.map +7 -0
  349. package/lib/utils/command.d.ts +6 -0
  350. package/lib/utils/command.js +72 -0
  351. package/lib/utils/command.js.map +7 -0
  352. package/lib/utils/copy.d.ts +16 -0
  353. package/lib/utils/copy.js +100 -0
  354. package/lib/utils/copy.js.map +7 -0
  355. package/lib/utils/dir.d.ts +21 -0
  356. package/lib/utils/dir.js +91 -0
  357. package/lib/utils/dir.js.map +7 -0
  358. package/lib/utils/env.d.ts +5 -0
  359. package/lib/utils/env.js +29 -0
  360. package/lib/utils/env.js.map +7 -0
  361. package/lib/utils/error.d.ts +28 -0
  362. package/lib/utils/error.js +72 -0
  363. package/lib/utils/error.js.map +7 -0
  364. package/lib/utils/exec.d.ts +40 -0
  365. package/lib/utils/exec.js +166 -0
  366. package/lib/utils/exec.js.map +7 -0
  367. package/lib/utils/help.d.ts +1 -0
  368. package/lib/utils/help.js +34 -0
  369. package/lib/utils/help.js.map +7 -0
  370. package/lib/utils/logging.d.ts +27 -0
  371. package/lib/utils/logging.js +60 -0
  372. package/lib/utils/logging.js.map +7 -0
  373. package/lib/utils/logo.d.ts +9 -0
  374. package/lib/utils/logo.js +74 -0
  375. package/lib/utils/logo.js.map +7 -0
  376. package/lib/utils/manifest.d.ts +10 -0
  377. package/lib/utils/manifest.js +77 -0
  378. package/lib/utils/manifest.js.map +7 -0
  379. package/lib/utils/port.d.ts +2 -0
  380. package/lib/utils/port.js +48 -0
  381. package/lib/utils/port.js.map +7 -0
  382. package/lib/utils/template.d.ts +37 -0
  383. package/lib/utils/template.js +116 -0
  384. package/lib/utils/template.js.map +7 -0
  385. package/lib/utils/validation.d.ts +6 -0
  386. package/lib/utils/validation.js +44 -0
  387. package/lib/utils/validation.js.map +7 -0
  388. package/lib/utils/version.d.ts +13 -0
  389. package/lib/utils/version.js +89 -0
  390. package/lib/utils/version.js.map +7 -0
  391. package/lib/utils/wait.d.ts +17 -0
  392. package/lib/utils/wait.js +59 -0
  393. package/lib/utils/wait.js.map +7 -0
  394. package/lib/utils/worker.d.ts +20 -0
  395. package/lib/utils/worker.js +63 -0
  396. package/lib/utils/worker.js.map +7 -0
  397. package/lib/wrapper/function-arguments.d.js +2 -0
  398. package/lib/wrapper/function-arguments.d.js.map +7 -0
  399. package/lib/wrapper/functionHandler.d.ts +10 -0
  400. package/lib/wrapper/functionHandler.js +63 -0
  401. package/lib/wrapper/functionHandler.js.map +7 -0
  402. package/lib/wrapper/http.d.ts +20 -0
  403. package/lib/wrapper/http.js +79 -0
  404. package/lib/wrapper/http.js.map +7 -0
  405. package/lib/wrapper/index.d.ts +12 -0
  406. package/lib/wrapper/index.js +16 -0
  407. package/lib/wrapper/index.js.map +7 -0
  408. package/lib/wrapper/main.d.ts +1 -0
  409. package/lib/wrapper/main.js +47 -0
  410. package/lib/wrapper/main.js.map +7 -0
  411. package/lib/wrapper/requestListener.d.ts +11 -0
  412. package/lib/wrapper/requestListener.js +66 -0
  413. package/lib/wrapper/requestListener.js.map +7 -0
  414. package/package.json +167 -0
  415. package/template/base/.github/CODEOWNERS +5 -0
  416. package/template/base/.github/renovate.json5 +3 -0
  417. package/template/base/_.dockerignore +18 -0
  418. package/template/base/_.eslintignore +14 -0
  419. package/template/base/_.eslintrc.js +3 -0
  420. package/template/base/_.gitignore +23 -0
  421. package/template/base/_.prettierignore +20 -0
  422. package/template/base/_.prettierrc.js +1 -0
  423. package/template/base/jest.config.ts +15 -0
  424. package/template/base/jest.setup.ts +3 -0
  425. package/template/base/tsconfig.build.json +5 -0
  426. package/template/base/tsconfig.json +13 -0
  427. package/template/express-rest-api/.buildkite/pipeline.yml +103 -0
  428. package/template/express-rest-api/.env +1 -0
  429. package/template/express-rest-api/.gantry/common.yml +20 -0
  430. package/template/express-rest-api/.gantry/dev.yml +8 -0
  431. package/template/express-rest-api/.gantry/prod.yml +8 -0
  432. package/template/express-rest-api/.nvmrc +1 -0
  433. package/template/express-rest-api/Dockerfile +37 -0
  434. package/template/express-rest-api/Dockerfile.dev-deps +11 -0
  435. package/template/express-rest-api/README.md +106 -0
  436. package/template/express-rest-api/docker-compose.yml +22 -0
  437. package/template/express-rest-api/gantry.apply.yml +122 -0
  438. package/template/express-rest-api/gantry.build.yml +15 -0
  439. package/template/express-rest-api/package.json +30 -0
  440. package/template/express-rest-api/skuba.template.js +49 -0
  441. package/template/express-rest-api/src/api/healthCheck.ts +11 -0
  442. package/template/express-rest-api/src/api/smokeTest.ts +8 -0
  443. package/template/express-rest-api/src/app.test.ts +9 -0
  444. package/template/express-rest-api/src/app.ts +14 -0
  445. package/template/express-rest-api/src/config.ts +57 -0
  446. package/template/express-rest-api/src/framework/logging.ts +17 -0
  447. package/template/express-rest-api/src/listen.ts +23 -0
  448. package/template/express-rest-api/src/register.ts +1 -0
  449. package/template/greeter/.buildkite/pipeline.yml +37 -0
  450. package/template/greeter/.nvmrc +1 -0
  451. package/template/greeter/Dockerfile +11 -0
  452. package/template/greeter/README.md +96 -0
  453. package/template/greeter/docker-compose.yml +22 -0
  454. package/template/greeter/package.json +24 -0
  455. package/template/greeter/skuba.template.js +16 -0
  456. package/template/greeter/src/app.test.ts +5 -0
  457. package/template/greeter/src/app.ts +4 -0
  458. package/template/koa-rest-api/.buildkite/pipeline.yml +103 -0
  459. package/template/koa-rest-api/.env +1 -0
  460. package/template/koa-rest-api/.gantry/common.yml +20 -0
  461. package/template/koa-rest-api/.gantry/dev.yml +11 -0
  462. package/template/koa-rest-api/.gantry/prod.yml +11 -0
  463. package/template/koa-rest-api/.nvmrc +1 -0
  464. package/template/koa-rest-api/Dockerfile +37 -0
  465. package/template/koa-rest-api/Dockerfile.dev-deps +11 -0
  466. package/template/koa-rest-api/README.md +111 -0
  467. package/template/koa-rest-api/docker-compose.yml +22 -0
  468. package/template/koa-rest-api/gantry.apply.yml +129 -0
  469. package/template/koa-rest-api/gantry.build.yml +15 -0
  470. package/template/koa-rest-api/package.json +47 -0
  471. package/template/koa-rest-api/skuba.template.js +49 -0
  472. package/template/koa-rest-api/src/api/healthCheck.ts +13 -0
  473. package/template/koa-rest-api/src/api/index.ts +10 -0
  474. package/template/koa-rest-api/src/api/jobs/getJobs.test.ts +9 -0
  475. package/template/koa-rest-api/src/api/jobs/getJobs.ts +15 -0
  476. package/template/koa-rest-api/src/api/jobs/index.ts +10 -0
  477. package/template/koa-rest-api/src/api/jobs/postJob.test.ts +34 -0
  478. package/template/koa-rest-api/src/api/jobs/postJob.ts +19 -0
  479. package/template/koa-rest-api/src/api/smokeTest.ts +11 -0
  480. package/template/koa-rest-api/src/app.test.ts +21 -0
  481. package/template/koa-rest-api/src/app.ts +8 -0
  482. package/template/koa-rest-api/src/config.ts +57 -0
  483. package/template/koa-rest-api/src/framework/logging.ts +25 -0
  484. package/template/koa-rest-api/src/framework/metrics.ts +11 -0
  485. package/template/koa-rest-api/src/framework/middleware.ts +3 -0
  486. package/template/koa-rest-api/src/framework/server.test.ts +260 -0
  487. package/template/koa-rest-api/src/framework/server.ts +51 -0
  488. package/template/koa-rest-api/src/framework/validation.test.ts +72 -0
  489. package/template/koa-rest-api/src/framework/validation.ts +77 -0
  490. package/template/koa-rest-api/src/listen.ts +24 -0
  491. package/template/koa-rest-api/src/register.ts +1 -0
  492. package/template/koa-rest-api/src/storage/jobs.ts +28 -0
  493. package/template/koa-rest-api/src/testing/logging.ts +16 -0
  494. package/template/koa-rest-api/src/testing/metrics.ts +29 -0
  495. package/template/koa-rest-api/src/testing/server.ts +31 -0
  496. package/template/koa-rest-api/src/testing/types.ts +24 -0
  497. package/template/koa-rest-api/src/tracing.ts +61 -0
  498. package/template/koa-rest-api/src/types/jobs.ts +17 -0
  499. package/template/koa-rest-api/src/types/koa.ts +8 -0
  500. package/template/lambda-sqs-worker/.buildkite/pipeline.yml +96 -0
  501. package/template/lambda-sqs-worker/.env +1 -0
  502. package/template/lambda-sqs-worker/.nvmrc +1 -0
  503. package/template/lambda-sqs-worker/Dockerfile +12 -0
  504. package/template/lambda-sqs-worker/README.md +132 -0
  505. package/template/lambda-sqs-worker/docker-compose.yml +27 -0
  506. package/template/lambda-sqs-worker/package.json +44 -0
  507. package/template/lambda-sqs-worker/serverless.yml +202 -0
  508. package/template/lambda-sqs-worker/skuba.template.js +32 -0
  509. package/template/lambda-sqs-worker/src/app.test.ts +116 -0
  510. package/template/lambda-sqs-worker/src/app.ts +52 -0
  511. package/template/lambda-sqs-worker/src/config.ts +57 -0
  512. package/template/lambda-sqs-worker/src/framework/handler.test.ts +61 -0
  513. package/template/lambda-sqs-worker/src/framework/handler.ts +43 -0
  514. package/template/lambda-sqs-worker/src/framework/logging.ts +27 -0
  515. package/template/lambda-sqs-worker/src/framework/metrics.ts +14 -0
  516. package/template/lambda-sqs-worker/src/framework/validation.test.ts +84 -0
  517. package/template/lambda-sqs-worker/src/framework/validation.ts +10 -0
  518. package/template/lambda-sqs-worker/src/hooks.ts +96 -0
  519. package/template/lambda-sqs-worker/src/mapping/jobScorer.ts +19 -0
  520. package/template/lambda-sqs-worker/src/services/aws.ts +5 -0
  521. package/template/lambda-sqs-worker/src/services/jobScorer.test.ts +44 -0
  522. package/template/lambda-sqs-worker/src/services/jobScorer.ts +56 -0
  523. package/template/lambda-sqs-worker/src/services/pipelineEventSender.test.ts +40 -0
  524. package/template/lambda-sqs-worker/src/services/pipelineEventSender.ts +33 -0
  525. package/template/lambda-sqs-worker/src/testing/handler.ts +13 -0
  526. package/template/lambda-sqs-worker/src/testing/logging.ts +16 -0
  527. package/template/lambda-sqs-worker/src/testing/services.ts +28 -0
  528. package/template/lambda-sqs-worker/src/testing/types.ts +33 -0
  529. package/template/lambda-sqs-worker/src/types/jobScorer.ts +15 -0
  530. package/template/lambda-sqs-worker/src/types/pipelineEvents.ts +21 -0
  531. package/template/lambda-sqs-worker/tsconfig.json +13 -0
  532. package/template/lambda-sqs-worker-cdk/.buildkite/pipeline.yml +93 -0
  533. package/template/lambda-sqs-worker-cdk/.nvmrc +1 -0
  534. package/template/lambda-sqs-worker-cdk/Dockerfile +13 -0
  535. package/template/lambda-sqs-worker-cdk/cdk.json +25 -0
  536. package/template/lambda-sqs-worker-cdk/docker-compose.yml +28 -0
  537. package/template/lambda-sqs-worker-cdk/infra/__snapshots__/appStack.test.ts.snap +723 -0
  538. package/template/lambda-sqs-worker-cdk/infra/appStack.test.ts +34 -0
  539. package/template/lambda-sqs-worker-cdk/infra/appStack.ts +74 -0
  540. package/template/lambda-sqs-worker-cdk/infra/index.ts +14 -0
  541. package/template/lambda-sqs-worker-cdk/package.json +30 -0
  542. package/template/lambda-sqs-worker-cdk/shared/context-types.ts +21 -0
  543. package/template/lambda-sqs-worker-cdk/skuba.template.js +27 -0
  544. package/template/lambda-sqs-worker-cdk/src/app.ts +10 -0
  545. package/template/lambda-sqs-worker-cdk/tsconfig.json +13 -0
  546. package/template/oss-npm-package/.github/workflows/release.yml +35 -0
  547. package/template/oss-npm-package/.github/workflows/validate.yml +31 -0
  548. package/template/oss-npm-package/.nvmrc +1 -0
  549. package/template/oss-npm-package/LICENSE +21 -0
  550. package/template/oss-npm-package/README.md +128 -0
  551. package/template/oss-npm-package/_package.json +41 -0
  552. package/template/oss-npm-package/skuba.template.js +22 -0
  553. package/template/oss-npm-package/src/index.test.ts +5 -0
  554. package/template/oss-npm-package/src/index.ts +7 -0
  555. package/template/oss-npm-package/tsconfig.json +10 -0
  556. package/template/private-npm-package/.buildkite/pipeline.yml +5 -0
  557. package/template/private-npm-package/.nvmrc +1 -0
  558. package/template/private-npm-package/README.md +109 -0
  559. package/template/private-npm-package/_package.json +41 -0
  560. package/template/private-npm-package/skuba.template.js +24 -0
  561. package/template/private-npm-package/src/index.test.ts +5 -0
  562. package/template/private-npm-package/src/index.ts +7 -0
  563. package/template/private-npm-package/tsconfig.json +10 -0
@@ -0,0 +1,52 @@
1
+ import 'skuba-dive/register';
2
+
3
+ import { SQSEvent } from 'aws-lambda';
4
+
5
+ import { createHandler } from 'src/framework/handler';
6
+ import { logger } from 'src/framework/logging';
7
+ import { metricsClient } from 'src/framework/metrics';
8
+ import { validateJson } from 'src/framework/validation';
9
+ import { scoreJobPublishedEvent, scoringService } from 'src/services/jobScorer';
10
+ import { sendPipelineEvent } from 'src/services/pipelineEventSender';
11
+ import { JobPublishedEventSchema } from 'src/types/pipelineEvents';
12
+
13
+ /**
14
+ * Tests connectivity to ensure appropriate access and network configuration.
15
+ */
16
+ const smokeTest = async () => {
17
+ await Promise.all([scoringService.smokeTest(), sendPipelineEvent({}, true)]);
18
+ };
19
+
20
+ export const handler = createHandler<SQSEvent>(async (event) => {
21
+ // Treat an empty object as our smoke test event.
22
+ if (!Object.keys(event).length) {
23
+ logger.info('Received smoke test request');
24
+ return smokeTest();
25
+ }
26
+
27
+ const count = event.Records.length;
28
+
29
+ if (count !== 1) {
30
+ throw Error(`Received ${count} records`);
31
+ }
32
+
33
+ logger.info({ count }, 'Received jobs');
34
+
35
+ metricsClient.distribution('job.received', event.Records.length);
36
+
37
+ const record = event.Records[0];
38
+
39
+ // TODO: this throws an error, which will cause the Lambda function to retry
40
+ // the event and eventually send it to your dead-letter queue. If you don't
41
+ // trust your source to provide consistently well-formed input, consider
42
+ // catching and handling this error in code.
43
+ const publishedJob = validateJson(record.body, JobPublishedEventSchema);
44
+
45
+ const scoredJob = await scoreJobPublishedEvent(publishedJob);
46
+
47
+ const snsMessageId = await sendPipelineEvent(scoredJob);
48
+
49
+ logger.info({ snsMessageId }, 'Scored job');
50
+
51
+ metricsClient.distribution('job.scored', 1);
52
+ });
@@ -0,0 +1,57 @@
1
+ import { Env } from 'skuba-dive';
2
+
3
+ interface Config {
4
+ environment: Environment;
5
+
6
+ logLevel: string;
7
+ metrics: boolean;
8
+ name: string;
9
+ version: string;
10
+
11
+ destinationSnsTopicArn: string;
12
+ }
13
+
14
+ type Environment = (typeof environments)[number];
15
+
16
+ const environments = ['local', 'test', 'dev', 'prod'] as const;
17
+
18
+ const environment = Env.oneOf(environments)('ENVIRONMENT');
19
+
20
+ /* istanbul ignore next: config verification makes more sense in a smoke test */
21
+ const configs: Record<Environment, () => Omit<Config, 'environment'>> = {
22
+ local: () => ({
23
+ logLevel: 'debug',
24
+ metrics: false,
25
+ name: '<%- serviceName %>',
26
+ version: 'local',
27
+
28
+ destinationSnsTopicArn: 'arn:aws:sns:us-east-2:123456789012:destination',
29
+ }),
30
+
31
+ test: () => ({
32
+ ...configs.local(),
33
+
34
+ logLevel: Env.string('LOG_LEVEL', { default: 'silent' }),
35
+ version: 'test',
36
+ }),
37
+
38
+ dev: () => ({
39
+ ...configs.prod(),
40
+
41
+ logLevel: 'debug',
42
+ }),
43
+
44
+ prod: () => ({
45
+ logLevel: 'info',
46
+ metrics: true,
47
+ name: Env.string('SERVICE'),
48
+ version: Env.string('VERSION'),
49
+
50
+ destinationSnsTopicArn: Env.string('DESTINATION_SNS_TOPIC_ARN'),
51
+ }),
52
+ };
53
+
54
+ export const config: Config = {
55
+ ...configs[environment](),
56
+ environment,
57
+ };
@@ -0,0 +1,61 @@
1
+ import { createCtx } from 'src/testing/handler';
2
+ import { logger } from 'src/testing/logging';
3
+ import { chance } from 'src/testing/types';
4
+
5
+ import { createHandler } from './handler';
6
+
7
+ describe('createHandler', () => {
8
+ const ctx = createCtx();
9
+ const input = chance.paragraph();
10
+
11
+ beforeAll(logger.spy);
12
+
13
+ afterEach(logger.clear);
14
+
15
+ it('handles happy path', async () => {
16
+ const output = chance.paragraph();
17
+
18
+ const handler = createHandler((event) => {
19
+ expect(event).toBe(input);
20
+
21
+ logger.info('Handler invoked');
22
+
23
+ return Promise.resolve(output);
24
+ });
25
+
26
+ await expect(handler(input, ctx)).resolves.toBe(output);
27
+
28
+ expect(logger.error).not.toHaveBeenCalled();
29
+
30
+ expect(logger.info.mock.calls).toEqual([
31
+ ['Handler invoked'],
32
+ ['Function succeeded'],
33
+ ]);
34
+ });
35
+
36
+ it('handles async error', async () => {
37
+ const err = Error(chance.sentence());
38
+
39
+ const handler = createHandler(() => Promise.reject(err));
40
+
41
+ await expect(handler(input, ctx)).rejects.toThrow('Function failed');
42
+
43
+ expect(logger.error).toHaveBeenCalledWith({ err }, 'Function failed');
44
+
45
+ expect(logger.info).not.toHaveBeenCalled();
46
+ });
47
+
48
+ it('handles sync error', async () => {
49
+ const err = Error(chance.sentence());
50
+
51
+ const handler = createHandler(() => {
52
+ throw err;
53
+ });
54
+
55
+ await expect(handler(input, ctx)).rejects.toThrow('Function failed');
56
+
57
+ expect(logger.error).toHaveBeenCalledWith({ err }, 'Function failed');
58
+
59
+ expect(logger.info).not.toHaveBeenCalled();
60
+ });
61
+ });
@@ -0,0 +1,43 @@
1
+ import { datadog } from 'datadog-lambda-js';
2
+
3
+ import { config } from 'src/config';
4
+ import { logger, loggerContext } from 'src/framework/logging';
5
+
6
+ interface LambdaContext {
7
+ awsRequestId: string;
8
+ }
9
+
10
+ type Handler<Event, Output> = (
11
+ event: Event,
12
+ ctx: LambdaContext,
13
+ ) => Promise<Output>;
14
+
15
+ /**
16
+ * Conditionally applies the Datadog wrapper to a Lambda handler.
17
+ *
18
+ * This also "fixes" its broken type definitions.
19
+ */
20
+ const withDatadog = <Event, Output = unknown>(
21
+ fn: Handler<Event, Output>,
22
+ ): Handler<Event, Output> =>
23
+ // istanbul ignore next
24
+ config.metrics ? (datadog(fn) as Handler<Event, Output>) : fn;
25
+
26
+ export const createHandler = <Event, Output = unknown>(
27
+ fn: (event: Event) => Promise<Output>,
28
+ ) =>
29
+ withDatadog<Event>((event, { awsRequestId }) =>
30
+ loggerContext.run({ awsRequestId }, async () => {
31
+ try {
32
+ const output = await fn(event);
33
+
34
+ logger.info('Function succeeded');
35
+
36
+ return output;
37
+ } catch (err) {
38
+ logger.error({ err }, 'Function failed');
39
+
40
+ throw new Error('Function failed');
41
+ }
42
+ }),
43
+ );
@@ -0,0 +1,27 @@
1
+ import { AsyncLocalStorage } from 'async_hooks';
2
+
3
+ import createLogger from '@seek/logger';
4
+
5
+ import { config } from 'src/config';
6
+
7
+ interface LoggerContext {
8
+ awsRequestId: string;
9
+ }
10
+
11
+ export const loggerContext = new AsyncLocalStorage<LoggerContext>();
12
+
13
+ export const logger = createLogger({
14
+ base: {
15
+ environment: config.environment,
16
+ version: config.version,
17
+ },
18
+
19
+ level: config.logLevel,
20
+
21
+ mixin: () => ({ ...loggerContext.getStore() }),
22
+
23
+ name: config.name,
24
+
25
+ transport:
26
+ config.environment === 'local' ? { target: 'pino-pretty' } : undefined,
27
+ });
@@ -0,0 +1,14 @@
1
+ import { sendDistributionMetric } from 'datadog-lambda-js';
2
+
3
+ import { config } from 'src/config';
4
+
5
+ const prefix = `${config.name}.`;
6
+
7
+ export const metricsClient = {
8
+ distribution: (
9
+ ...[name, ...rest]: Parameters<typeof sendDistributionMetric>
10
+ ) =>
11
+ config.metrics
12
+ ? sendDistributionMetric(`${prefix}${name}`, ...rest)
13
+ : undefined,
14
+ };
@@ -0,0 +1,84 @@
1
+ import {
2
+ IdDescriptionSchema,
3
+ chance,
4
+ mockIdDescription,
5
+ } from 'src/testing/types';
6
+
7
+ import { validateJson } from './validation';
8
+
9
+ describe('validateJson', () => {
10
+ const idDescription = mockIdDescription();
11
+
12
+ it('permits valid input', () => {
13
+ const input = JSON.stringify(idDescription);
14
+
15
+ expect(validateJson(input, IdDescriptionSchema)).toStrictEqual(
16
+ idDescription,
17
+ );
18
+ });
19
+
20
+ it('filters additional properties', () => {
21
+ const input = JSON.stringify({ ...idDescription, hacker: chance.name() });
22
+
23
+ expect(validateJson(input, IdDescriptionSchema)).toStrictEqual(
24
+ idDescription,
25
+ );
26
+ });
27
+
28
+ it('blocks mistyped prop', () => {
29
+ const input = JSON.stringify({ ...idDescription, id: null });
30
+
31
+ expect(() => validateJson(input, IdDescriptionSchema))
32
+ .toThrowErrorMatchingInlineSnapshot(`
33
+ "[
34
+ {
35
+ "code": "invalid_type",
36
+ "expected": "string",
37
+ "received": "null",
38
+ "path": [
39
+ "id"
40
+ ],
41
+ "message": "Expected string, received null"
42
+ }
43
+ ]"
44
+ `);
45
+ });
46
+
47
+ it('blocks missing prop', () => {
48
+ const input = '{}';
49
+
50
+ expect(() => validateJson(input, IdDescriptionSchema))
51
+ .toThrowErrorMatchingInlineSnapshot(`
52
+ "[
53
+ {
54
+ "code": "invalid_type",
55
+ "expected": "string",
56
+ "received": "undefined",
57
+ "path": [
58
+ "id"
59
+ ],
60
+ "message": "Required"
61
+ },
62
+ {
63
+ "code": "invalid_type",
64
+ "expected": "string",
65
+ "received": "undefined",
66
+ "path": [
67
+ "description"
68
+ ],
69
+ "message": "Required"
70
+ }
71
+ ]"
72
+ `);
73
+ });
74
+
75
+ it('blocks invalid JSON', () => {
76
+ const input = '}';
77
+
78
+ expect(() =>
79
+ validateJson(input, IdDescriptionSchema),
80
+ ).toThrowErrorMatchingInlineSnapshot(
81
+ `"Unexpected token } in JSON at position 0"`,
82
+ );
83
+ });
84
+ });
@@ -0,0 +1,10 @@
1
+ import { z } from 'zod';
2
+
3
+ export const validateJson = <
4
+ Output,
5
+ Def extends z.ZodTypeDef = z.ZodTypeDef,
6
+ Input = Output,
7
+ >(
8
+ input: string,
9
+ schema: z.ZodSchema<Output, Def, Input>,
10
+ ): Output => schema.parse(JSON.parse(input));
@@ -0,0 +1,96 @@
1
+ /* eslint-disable no-console */
2
+ /* istanbul ignore file */
3
+
4
+ // Use minimal dependencies to reduce the chance of crashes on module load.
5
+ import {
6
+ CodeDeployClient,
7
+ PutLifecycleEventHookExecutionStatusCommand,
8
+ } from '@aws-sdk/client-codedeploy';
9
+ import { InvokeCommand, LambdaClient } from '@aws-sdk/client-lambda';
10
+ import { toUtf8 } from '@aws-sdk/util-utf8-node';
11
+
12
+ const codeDeploy = new CodeDeployClient({
13
+ apiVersion: '2014-10-06',
14
+ maxAttempts: 5,
15
+ });
16
+
17
+ const lambda = new LambdaClient({
18
+ apiVersion: '2015-03-31',
19
+ maxAttempts: 5,
20
+ });
21
+
22
+ type Status = 'Succeeded' | 'Failed';
23
+
24
+ /**
25
+ * Synchronously invokes a Lambda function with a smoke test event.
26
+ *
27
+ * Any non-error response is treated as a success.
28
+ */
29
+ const smokeTestLambdaFunction = async (): Promise<Status> => {
30
+ const functionName = process.env.FUNCTION_NAME_TO_INVOKE;
31
+
32
+ if (!functionName) {
33
+ console.error('Missing process.env.FUNCTION_NAME_TO_INVOKE');
34
+ return 'Failed';
35
+ }
36
+
37
+ console.info('Function:', functionName);
38
+
39
+ const response = await lambda.send(
40
+ new InvokeCommand({
41
+ FunctionName: functionName,
42
+ InvocationType: 'RequestResponse',
43
+ // Treat an empty object as our smoke test event.
44
+ Payload: Buffer.from('{}'),
45
+ }),
46
+ );
47
+
48
+ console.info('Version:', response.ExecutedVersion ?? '?');
49
+ console.info('Status', response.StatusCode ?? '?');
50
+
51
+ if (response.FunctionError) {
52
+ console.error('Error:', response.FunctionError);
53
+ if (response.Payload) {
54
+ console.error(toUtf8(response.Payload));
55
+ }
56
+ return 'Failed';
57
+ }
58
+
59
+ return response.StatusCode === 200 ? 'Succeeded' : 'Failed';
60
+ };
61
+
62
+ /**
63
+ * The event supplied to a CodeDeploy lifecycle hook Lambda function.
64
+ *
65
+ * {@link https://docs.aws.amazon.com/codedeploy/latest/userguide/tutorial-ecs-with-hooks-create-hooks.html}
66
+ */
67
+ interface CodeDeployLifecycleHookEvent {
68
+ DeploymentId: string;
69
+ LifecycleEventHookExecutionId: string;
70
+ }
71
+
72
+ /**
73
+ * A handler to smoke test a new Lambda function version before it goes live.
74
+ *
75
+ * This tries to be exception safe so that a status reaches CodeDeploy. If we
76
+ * crash or otherwise fail to report back, the deployment will hang for an hour.
77
+ */
78
+ export const pre = async (
79
+ event: CodeDeployLifecycleHookEvent,
80
+ ): Promise<void> => {
81
+ let status: Status;
82
+ try {
83
+ status = await smokeTestLambdaFunction();
84
+ } catch (err) {
85
+ console.error('Exception:', err);
86
+ status = 'Failed';
87
+ }
88
+
89
+ await codeDeploy.send(
90
+ new PutLifecycleEventHookExecutionStatusCommand({
91
+ deploymentId: event.DeploymentId,
92
+ lifecycleEventHookExecutionId: event.LifecycleEventHookExecutionId,
93
+ status,
94
+ }),
95
+ );
96
+ };
@@ -0,0 +1,19 @@
1
+ import { JobScorerInput, JobScorerOutput } from 'src/types/jobScorer';
2
+ import { JobPublishedEvent, JobScoredEvent } from 'src/types/pipelineEvents';
3
+
4
+ export const jobPublishedEventToScorerInput = (
5
+ record: JobPublishedEvent,
6
+ ): JobScorerInput => ({
7
+ details: record.data.details,
8
+ id: record.entityId,
9
+ });
10
+
11
+ export const jobScorerOutputToScoredEvent = (
12
+ output: JobScorerOutput,
13
+ ): JobScoredEvent => ({
14
+ data: {
15
+ score: output.score,
16
+ },
17
+ entityId: output.id,
18
+ eventType: 'JobScored',
19
+ });
@@ -0,0 +1,5 @@
1
+ import { SNSClient } from '@aws-sdk/client-sns';
2
+
3
+ export const sns = new SNSClient({
4
+ apiVersion: '2010-03-31',
5
+ });
@@ -0,0 +1,44 @@
1
+ import { scoringService } from 'src/testing/services';
2
+ import { chance, mockJobPublishedEvent } from 'src/testing/types';
3
+
4
+ import * as jobScorer from './jobScorer';
5
+
6
+ describe('scoreJobPublishedEvent', () => {
7
+ beforeAll(scoringService.spy);
8
+
9
+ afterEach(scoringService.clear);
10
+
11
+ it('scores an event', async () => {
12
+ const score = chance.floating({ max: 1, min: 0 });
13
+
14
+ scoringService.request.mockResolvedValue(score);
15
+
16
+ await expect(
17
+ jobScorer.scoreJobPublishedEvent(
18
+ mockJobPublishedEvent({ entityId: '1' }),
19
+ ),
20
+ ).resolves.toStrictEqual({
21
+ data: {
22
+ score,
23
+ },
24
+ entityId: '1',
25
+ eventType: 'JobScored',
26
+ });
27
+
28
+ expect(scoringService.request).toHaveBeenCalledTimes(1);
29
+ });
30
+
31
+ it('bubbles up scoring service error', async () => {
32
+ const err = Error(chance.sentence());
33
+
34
+ scoringService.request.mockRejectedValue(err);
35
+
36
+ await expect(
37
+ jobScorer.scoreJobPublishedEvent(
38
+ mockJobPublishedEvent({ entityId: '1' }),
39
+ ),
40
+ ).rejects.toThrow(err);
41
+
42
+ expect(scoringService.request).toHaveBeenCalledTimes(1);
43
+ });
44
+ });
@@ -0,0 +1,56 @@
1
+ import {
2
+ jobPublishedEventToScorerInput,
3
+ jobScorerOutputToScoredEvent,
4
+ } from 'src/mapping/jobScorer';
5
+ import {
6
+ JobScorerInput,
7
+ JobScorerOutput,
8
+ JobScorerOutputSchema,
9
+ } from 'src/types/jobScorer';
10
+ import { JobPublishedEvent, JobScoredEvent } from 'src/types/pipelineEvents';
11
+
12
+ /* istanbul ignore next: simulation of an external service */
13
+ export const scoringService = {
14
+ request: (details: string): Promise<unknown> => {
15
+ // Networking woes
16
+ if (Math.random() < 0.05) {
17
+ const err = Error('could not reach scoring service');
18
+
19
+ return Promise.reject(err);
20
+ }
21
+
22
+ // Unexpected behaviour on certain inputs
23
+ if (details.length % 100 === 0) {
24
+ return Promise.resolve(null);
25
+ }
26
+
27
+ return Promise.resolve(Math.random());
28
+ },
29
+
30
+ smokeTest: async (): Promise<void> => {
31
+ // A connectivity test
32
+ await Promise.resolve();
33
+ },
34
+ };
35
+
36
+ const scoreJob = async ({
37
+ details,
38
+ id,
39
+ }: JobScorerInput): Promise<JobScorerOutput> => {
40
+ const score = await scoringService.request(details);
41
+
42
+ return JobScorerOutputSchema.parse({
43
+ id,
44
+ score,
45
+ });
46
+ };
47
+
48
+ export const scoreJobPublishedEvent = async (
49
+ publishedJob: JobPublishedEvent,
50
+ ): Promise<JobScoredEvent> => {
51
+ const scorerInput = jobPublishedEventToScorerInput(publishedJob);
52
+
53
+ const scorerOutput = await scoreJob(scorerInput);
54
+
55
+ return jobScorerOutputToScoredEvent(scorerOutput);
56
+ };
@@ -0,0 +1,40 @@
1
+ import { PublishCommand } from '@aws-sdk/client-sns';
2
+
3
+ import { sns } from 'src/testing/services';
4
+ import { chance } from 'src/testing/types';
5
+
6
+ import { sendPipelineEvent } from './pipelineEventSender';
7
+
8
+ describe('sendPipelineEvent', () => {
9
+ afterEach(() => {
10
+ jest.clearAllMocks();
11
+ });
12
+
13
+ it('handles happy path', async () => {
14
+ const messageId = chance.guid({ version: 4 });
15
+
16
+ sns.publish.resolves({ MessageId: messageId });
17
+
18
+ await expect(sendPipelineEvent({})).resolves.toBe(messageId);
19
+
20
+ expect(sns.client).toReceiveCommandTimes(PublishCommand, 1);
21
+ });
22
+
23
+ it('bubbles up SNS error', () => {
24
+ const err = Error(chance.sentence());
25
+
26
+ sns.publish.rejects(err);
27
+
28
+ return expect(sendPipelineEvent({})).rejects.toThrow(err);
29
+ });
30
+
31
+ it('throws on missing message ID', () => {
32
+ sns.publish.resolves({});
33
+
34
+ return expect(
35
+ sendPipelineEvent({}),
36
+ ).rejects.toThrowErrorMatchingInlineSnapshot(
37
+ `"SNS did not return a message ID"`,
38
+ );
39
+ });
40
+ });
@@ -0,0 +1,33 @@
1
+ import { PublishCommand } from '@aws-sdk/client-sns';
2
+
3
+ import { config } from 'src/config';
4
+
5
+ import { sns } from './aws';
6
+
7
+ export const sendPipelineEvent = async (
8
+ event: unknown,
9
+ smokeTest: boolean = false,
10
+ ): Promise<string> => {
11
+ const snsResponse = await sns.send(
12
+ new PublishCommand({
13
+ Message: JSON.stringify(event),
14
+ ...(smokeTest && {
15
+ MessageAttributes: {
16
+ // Used for connectivity tests.
17
+ // Subscribers should filter out messages containing this attribute.
18
+ SmokeTest: {
19
+ DataType: 'String',
20
+ StringValue: 'true',
21
+ },
22
+ },
23
+ }),
24
+ TopicArn: config.destinationSnsTopicArn,
25
+ }),
26
+ );
27
+
28
+ if (snsResponse.MessageId === undefined) {
29
+ throw Error('SNS did not return a message ID');
30
+ }
31
+
32
+ return snsResponse.MessageId;
33
+ };
@@ -0,0 +1,13 @@
1
+ import { Context, SQSEvent } from 'aws-lambda';
2
+
3
+ import { chance } from './types';
4
+
5
+ export const createCtx = () =>
6
+ ({
7
+ awsRequestId: chance.guid({ version: 4 }),
8
+ } as Context);
9
+
10
+ export const createSqsEvent = (bodies: string[]) =>
11
+ ({
12
+ Records: bodies.map((body) => ({ body })),
13
+ } as SQSEvent);
@@ -0,0 +1,16 @@
1
+ import * as logging from 'src/framework/logging';
2
+
3
+ export const logger = {
4
+ error: jest.fn(),
5
+ info: jest.fn(),
6
+
7
+ clear: () => {
8
+ logger.error.mockClear();
9
+ logger.info.mockClear();
10
+ },
11
+
12
+ spy: () => {
13
+ jest.spyOn(logging.logger, 'error').mockImplementation(logger.error);
14
+ jest.spyOn(logging.logger, 'info').mockImplementation(logger.info);
15
+ },
16
+ };