From 1b619f4be2982219afc12302cb623e8442222ca8 Mon Sep 17 00:00:00 2001 From: isoppp Date: Tue, 24 Sep 2024 00:44:05 +0900 Subject: [PATCH 1/4] init fastify `fastify generate api-fastify --lang=typescript` --- apps/api-fastify/.gitignore | 65 ++ apps/api-fastify/README.md | 23 + apps/api-fastify/package.json | 35 + apps/api-fastify/src/app.ts | 37 + apps/api-fastify/src/plugins/README.md | 16 + apps/api-fastify/src/plugins/sensible.ts | 11 + apps/api-fastify/src/plugins/support.ts | 20 + apps/api-fastify/src/routes/README.md | 30 + apps/api-fastify/src/routes/example/index.ts | 9 + apps/api-fastify/src/routes/root.ts | 9 + apps/api-fastify/test/helper.ts | 37 + apps/api-fastify/test/plugins/support.test.ts | 13 + apps/api-fastify/test/routes/example.test.ts | 13 + apps/api-fastify/test/routes/root.test.ts | 12 + apps/api-fastify/test/tsconfig.json | 8 + apps/api-fastify/tsconfig.json | 8 + pnpm-lock.yaml | 797 +++++++++++++++++- 17 files changed, 1098 insertions(+), 45 deletions(-) create mode 100644 apps/api-fastify/.gitignore create mode 100644 apps/api-fastify/README.md create mode 100644 apps/api-fastify/package.json create mode 100644 apps/api-fastify/src/app.ts create mode 100644 apps/api-fastify/src/plugins/README.md create mode 100644 apps/api-fastify/src/plugins/sensible.ts create mode 100644 apps/api-fastify/src/plugins/support.ts create mode 100644 apps/api-fastify/src/routes/README.md create mode 100644 apps/api-fastify/src/routes/example/index.ts create mode 100644 apps/api-fastify/src/routes/root.ts create mode 100644 apps/api-fastify/test/helper.ts create mode 100644 apps/api-fastify/test/plugins/support.test.ts create mode 100644 apps/api-fastify/test/routes/example.test.ts create mode 100644 apps/api-fastify/test/routes/root.test.ts create mode 100644 apps/api-fastify/test/tsconfig.json create mode 100644 apps/api-fastify/tsconfig.json diff --git a/apps/api-fastify/.gitignore b/apps/api-fastify/.gitignore new file mode 100644 index 0000000..f4cefe8 --- /dev/null +++ b/apps/api-fastify/.gitignore @@ -0,0 +1,65 @@ +# Logs +logs +*.log +npm-debug.log* + +# Runtime data +pids +*.pid +*.seed + +# Directory for instrumented libs generated by jscoverage/JSCover +lib-cov + +# Coverage directory used by tools like istanbul +coverage + +# nyc test coverage +.nyc_output + +# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files) +.grunt + +# node-waf configuration +.lock-wscript + +# Compiled binary addons (http://nodejs.org/api/addons.html) +build/Release + +# Dependency directories +node_modules +jspm_packages + +# Optional npm cache directory +.npm + +# Optional REPL history +.node_repl_history + +# 0x +profile-* + +# mac files +.DS_Store + +# vim swap files +*.swp + +# webstorm +.idea + +# vscode +.vscode +*code-workspace + +# clinic +profile* +*clinic* +*flamegraph* + +# generated code +examples/typescript-server.js +test/types/index.js + +# compiled app +dist diff --git a/apps/api-fastify/README.md b/apps/api-fastify/README.md new file mode 100644 index 0000000..2d5e417 --- /dev/null +++ b/apps/api-fastify/README.md @@ -0,0 +1,23 @@ +# Getting Started with [Fastify-CLI](https://www.npmjs.com/package/fastify-cli) +This project was bootstrapped with Fastify-CLI. + +## Available Scripts + +In the project directory, you can run: + +### `npm run dev` + +To start the app in dev mode.\ +Open [http://localhost:3000](http://localhost:3000) to view it in the browser. + +### `npm start` + +For production mode + +### `npm run test` + +Run the test cases. + +## Learn More + +To learn Fastify, check out the [Fastify documentation](https://fastify.dev/docs/latest/). diff --git a/apps/api-fastify/package.json b/apps/api-fastify/package.json new file mode 100644 index 0000000..3ed2cdf --- /dev/null +++ b/apps/api-fastify/package.json @@ -0,0 +1,35 @@ +{ + "name": "api-fastify", + "version": "1.0.0", + "description": "This project was bootstrapped with Fastify-CLI.", + "main": "app.ts", + "directories": { + "test": "test" + }, + "scripts": { + "test": "npm run build:ts && tsc -p test/tsconfig.json && c8 node --test -r ts-node/register \"test/**/*.ts\"", + "start": "npm run build:ts && fastify start -l info dist/app.js", + "build:ts": "tsc", + "watch:ts": "tsc -w", + "dev": "npm run build:ts && concurrently -k -p \"[{name}]\" -n \"TypeScript,App\" -c \"yellow.bold,cyan.bold\" \"npm:watch:ts\" \"npm:dev:start\"", + "dev:start": "fastify start --ignore-watch=.ts$ -w -l info -P dist/app.js" + }, + "keywords": [], + "author": "", + "license": "ISC", + "dependencies": { + "fastify": "^5.0.0", + "fastify-plugin": "^5.0.0", + "@fastify/autoload": "^6.0.0", + "@fastify/sensible": "^6.0.0", + "fastify-cli": "^7.0.1" + }, + "devDependencies": { + "@types/node": "^22.1.0", + "c8": "^10.1.2", + "ts-node": "^10.4.0", + "concurrently": "^9.0.0", + "fastify-tsconfig": "^2.0.0", + "typescript": "^5.2.2" + } +} \ No newline at end of file diff --git a/apps/api-fastify/src/app.ts b/apps/api-fastify/src/app.ts new file mode 100644 index 0000000..be51f92 --- /dev/null +++ b/apps/api-fastify/src/app.ts @@ -0,0 +1,37 @@ +import { join } from 'path'; +import AutoLoad, {AutoloadPluginOptions} from '@fastify/autoload'; +import { FastifyPluginAsync, FastifyServerOptions } from 'fastify'; + +export interface AppOptions extends FastifyServerOptions, Partial { + +} +// Pass --options via CLI arguments in command to enable these options. +const options: AppOptions = { +} + +const app: FastifyPluginAsync = async ( + fastify, + opts +): Promise => { + // Place here your custom code! + + // Do not touch the following lines + + // This loads all plugins defined in plugins + // those should be support plugins that are reused + // through your application + void fastify.register(AutoLoad, { + dir: join(__dirname, 'plugins'), + options: opts + }) + + // This loads all plugins defined in routes + // define your routes in one of these + void fastify.register(AutoLoad, { + dir: join(__dirname, 'routes'), + options: opts + }) +}; + +export default app; +export { app, options } diff --git a/apps/api-fastify/src/plugins/README.md b/apps/api-fastify/src/plugins/README.md new file mode 100644 index 0000000..1e61ee5 --- /dev/null +++ b/apps/api-fastify/src/plugins/README.md @@ -0,0 +1,16 @@ +# Plugins Folder + +Plugins define behavior that is common to all the routes in your +application. Authentication, caching, templates, and all the other cross +cutting concerns should be handled by plugins placed in this folder. + +Files in this folder are typically defined through the +[`fastify-plugin`](https://github.com/fastify/fastify-plugin) module, +making them non-encapsulated. They can define decorators and set hooks +that will then be used in the rest of your application. + +Check out: + +* [The hitchhiker's guide to plugins](https://fastify.dev/docs/latest/Guides/Plugins-Guide/) +* [Fastify decorators](https://fastify.dev/docs/latest/Reference/Decorators/). +* [Fastify lifecycle](https://fastify.dev/docs/latest/Reference/Lifecycle/). diff --git a/apps/api-fastify/src/plugins/sensible.ts b/apps/api-fastify/src/plugins/sensible.ts new file mode 100644 index 0000000..5be2c8a --- /dev/null +++ b/apps/api-fastify/src/plugins/sensible.ts @@ -0,0 +1,11 @@ +import fp from 'fastify-plugin' +import sensible, { FastifySensibleOptions } from '@fastify/sensible' + +/** + * This plugins adds some utilities to handle http errors + * + * @see https://github.com/fastify/fastify-sensible + */ +export default fp(async (fastify) => { + fastify.register(sensible) +}) diff --git a/apps/api-fastify/src/plugins/support.ts b/apps/api-fastify/src/plugins/support.ts new file mode 100644 index 0000000..94bae4f --- /dev/null +++ b/apps/api-fastify/src/plugins/support.ts @@ -0,0 +1,20 @@ +import fp from 'fastify-plugin' + +export interface SupportPluginOptions { + // Specify Support plugin options here +} + +// The use of fastify-plugin is required to be able +// to export the decorators to the outer scope +export default fp(async (fastify, opts) => { + fastify.decorate('someSupport', function () { + return 'hugs' + }) +}) + +// When using .decorate you have to specify added properties for Typescript +declare module 'fastify' { + export interface FastifyInstance { + someSupport(): string; + } +} diff --git a/apps/api-fastify/src/routes/README.md b/apps/api-fastify/src/routes/README.md new file mode 100644 index 0000000..084748f --- /dev/null +++ b/apps/api-fastify/src/routes/README.md @@ -0,0 +1,30 @@ +# Routes Folder + +Routes define the pathways within your application. +Fastify's structure supports the modular monolith approach, where your +application is organized into distinct, self-contained modules. +This facilitates easier scaling and future transition to a microservice architecture. +In the future you might want +to independently deploy some of those. + +In this folder you should define all the routes that define the endpoints +of your web application. +Each service is a [Fastify +plugin](https://fastify.dev/docs/latest/Reference/Plugins/), it is +encapsulated (it can have its own independent plugins) and it is +typically stored in a file; be careful to group your routes logically, +e.g. all `/users` routes in a `users.js` file. We have added +a `root.js` file for you with a '/' root added. + +If a single file becomes too large, create a folder and add a `index.js` file there: +this file must be a Fastify plugin, and it will be loaded automatically +by the application. You can now add as many files as you want inside that folder. +In this way you can create complex routes within a single monolith, +and eventually extract them. + +If you need to share functionality between routes, place that +functionality into the `plugins` folder, and share it via +[decorators](https://fastify.dev/docs/latest/Reference/Decorators/). + +If you're a bit confused about using `async/await` to write routes, you would +better take a look at [Promise resolution](https://fastify.dev/docs/latest/Reference/Routes/#promise-resolution) for more details. diff --git a/apps/api-fastify/src/routes/example/index.ts b/apps/api-fastify/src/routes/example/index.ts new file mode 100644 index 0000000..819c5e7 --- /dev/null +++ b/apps/api-fastify/src/routes/example/index.ts @@ -0,0 +1,9 @@ +import { FastifyPluginAsync } from "fastify" + +const example: FastifyPluginAsync = async (fastify, opts): Promise => { + fastify.get('/', async function (request, reply) { + return 'this is an example' + }) +} + +export default example; diff --git a/apps/api-fastify/src/routes/root.ts b/apps/api-fastify/src/routes/root.ts new file mode 100644 index 0000000..2a1b334 --- /dev/null +++ b/apps/api-fastify/src/routes/root.ts @@ -0,0 +1,9 @@ +import { FastifyPluginAsync } from 'fastify' + +const root: FastifyPluginAsync = async (fastify, opts): Promise => { + fastify.get('/', async function (request, reply) { + return { root: true } + }) +} + +export default root; diff --git a/apps/api-fastify/test/helper.ts b/apps/api-fastify/test/helper.ts new file mode 100644 index 0000000..7045177 --- /dev/null +++ b/apps/api-fastify/test/helper.ts @@ -0,0 +1,37 @@ +// This file contains code that we reuse between our tests. +const helper = require('fastify-cli/helper.js') +import * as path from 'path' +import * as test from 'node:test' + +export type TestContext = { + after: typeof test.after +}; + +const AppPath = path.join(__dirname, '..', 'src', 'app.ts') + +// Fill in this config with all the configurations +// needed for testing the application +async function config () { + return {} +} + +// Automatically build and tear down our instance +async function build (t: TestContext) { + // you can set all the options supported by the fastify CLI command + const argv = [AppPath] + + // fastify-plugin ensures that all decorators + // are exposed for testing purposes, this is + // different from the production setup + const app = await helper.build(argv, await config()) + + // Tear down our app after we are done + t.after(() => void app.close()) + + return app +} + +export { + config, + build +} diff --git a/apps/api-fastify/test/plugins/support.test.ts b/apps/api-fastify/test/plugins/support.test.ts new file mode 100644 index 0000000..d3b5c46 --- /dev/null +++ b/apps/api-fastify/test/plugins/support.test.ts @@ -0,0 +1,13 @@ +import { test } from 'node:test' +import * as assert from 'node:assert' + +import Fastify from 'fastify' +import Support from '../../src/plugins/support' + +test('support works standalone', async (t) => { + const fastify = Fastify() + void fastify.register(Support) + await fastify.ready() + + assert.equal(fastify.someSupport(), 'hugs') +}) diff --git a/apps/api-fastify/test/routes/example.test.ts b/apps/api-fastify/test/routes/example.test.ts new file mode 100644 index 0000000..0c35dfe --- /dev/null +++ b/apps/api-fastify/test/routes/example.test.ts @@ -0,0 +1,13 @@ +import { test } from 'node:test' +import * as assert from 'node:assert' +import { build } from '../helper' + +test('example is loaded', async (t) => { + const app = await build(t) + + const res = await app.inject({ + url: '/example' + }) + + assert.equal(res.payload, 'this is an example') +}) diff --git a/apps/api-fastify/test/routes/root.test.ts b/apps/api-fastify/test/routes/root.test.ts new file mode 100644 index 0000000..17554ec --- /dev/null +++ b/apps/api-fastify/test/routes/root.test.ts @@ -0,0 +1,12 @@ +import { test } from 'node:test' +import * as assert from 'node:assert' +import { build } from '../helper' + +test('default root route', async (t) => { + const app = await build(t) + + const res = await app.inject({ + url: '/' + }) + assert.deepStrictEqual(JSON.parse(res.payload), { root: true }) +}) diff --git a/apps/api-fastify/test/tsconfig.json b/apps/api-fastify/test/tsconfig.json new file mode 100644 index 0000000..384d171 --- /dev/null +++ b/apps/api-fastify/test/tsconfig.json @@ -0,0 +1,8 @@ +{ + "extends": "../tsconfig.json", + "compilerOptions": { + "baseUrl": ".", + "noEmit": true + }, + "include": ["../src/**/*.ts", "**/*.ts"] +} diff --git a/apps/api-fastify/tsconfig.json b/apps/api-fastify/tsconfig.json new file mode 100644 index 0000000..50dd099 --- /dev/null +++ b/apps/api-fastify/tsconfig.json @@ -0,0 +1,8 @@ +{ + "extends": "fastify-tsconfig", + "compilerOptions": { + "outDir": "dist", + "sourceMap": true + }, + "include": ["src/**/*.ts"] +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index b909f63..3a1a487 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -18,6 +18,43 @@ importers: specifier: 5.6.2 version: 5.6.2 + apps/api-fastify: + dependencies: + '@fastify/autoload': + specifier: ^6.0.0 + version: 6.0.1 + '@fastify/sensible': + specifier: ^6.0.0 + version: 6.0.1 + fastify: + specifier: ^5.0.0 + version: 5.0.0 + fastify-cli: + specifier: ^7.0.1 + version: 7.0.1 + fastify-plugin: + specifier: ^5.0.0 + version: 5.0.1 + devDependencies: + '@types/node': + specifier: ^22.1.0 + version: 22.5.5 + c8: + specifier: ^10.1.2 + version: 10.1.2 + concurrently: + specifier: ^9.0.0 + version: 9.0.1 + fastify-tsconfig: + specifier: ^2.0.0 + version: 2.0.0 + ts-node: + specifier: ^10.4.0 + version: 10.9.2(@types/node@22.5.5)(typescript@5.6.2) + typescript: + specifier: ^5.2.2 + version: 5.6.2 + apps/app: dependencies: '@google-cloud/error-reporting': @@ -949,6 +986,9 @@ packages: '@base2/pretty-print-object@1.0.1': resolution: {integrity: sha512-4iri8i1AqYHJE2DstZYkyEprg6Pq6sKx3xn5FpySk9sNhH7qN2LLlHJCfDTZRILNwQNPD7mATWM0TBui7uC1pA==} + '@bcoe/v8-coverage@0.2.3': + resolution: {integrity: sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==} + '@biomejs/biome@1.8.3': resolution: {integrity: sha512-/uUV3MV+vyAczO+vKrPdOW0Iaet7UnJMU4bNMinggGJTAnBPjCoLEYcyYtYHNnUNYlv4xZMH6hVIQCAozq8d5w==} engines: {node: '>=14.21.3'} @@ -1603,6 +1643,27 @@ packages: resolution: {integrity: sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@fastify/ajv-compiler@4.0.1': + resolution: {integrity: sha512-DxrBdgsjNLP0YM6W5Hd6/Fmj43S8zMKiFJYgi+Ri3htTGAowPVG/tG1wpnWLMjufEnehRivUCKZ1pLDIoZdTuw==} + + '@fastify/autoload@6.0.1': + resolution: {integrity: sha512-NVVu/g9yUFkv4nNxsDHhNKz37jINXlPxwNYbA4Hu8pnho6sDVeKo24pt+dnax1SeUapxxs/yulXukbUai6Aebw==} + + '@fastify/deepmerge@2.0.0': + resolution: {integrity: sha512-fsaybTGDyQ5KpPsplQqb9yKdCf2x/pbNpMNk8Tvp3rRz7lVcupKysH4b2ELMN2P4Hak1+UqTYdTj/u4FNV2p0g==} + + '@fastify/error@4.0.0': + resolution: {integrity: sha512-OO/SA8As24JtT1usTUTKgGH7uLvhfwZPwlptRi2Dp5P4KKmJI3gvsZ8MIHnNwDs4sLf/aai5LzTyl66xr7qMxA==} + + '@fastify/fast-json-stringify-compiler@5.0.1': + resolution: {integrity: sha512-f2d3JExJgFE3UbdFcpPwqNUEoHWmt8pAKf8f+9YuLESdefA0WgqxeT6DrGL4Yrf/9ihXNSKOqpjEmurV405meA==} + + '@fastify/merge-json-schemas@0.1.1': + resolution: {integrity: sha512-fERDVz7topgNjtXsJTTW1JKLy0rhuLRcquYqNR9rF7OcVpCa2OVW49ZPDIhaRRCaUuvVxI+N416xUoF76HNSXA==} + + '@fastify/sensible@6.0.1': + resolution: {integrity: sha512-D0rN0kMeZKP23f4w9MoCI9e4+n7vASFAsGoRNn9bondSbplLeIfR2HcjCbyElAM04jGrPRLi/edyThEPOyC9cQ==} + '@google-cloud/common@4.0.3': resolution: {integrity: sha512-fUoMo5b8iAKbrYpneIRV3z95AlxVJPrjpevxs4SKoclngWZvTXBSGpNisF5+x5m+oNGve7jfB1e6vNBZBUs7Fw==} engines: {node: '>=12.0.0'} @@ -1719,6 +1780,10 @@ packages: resolution: {integrity: sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA==} engines: {node: '>=12'} + '@istanbuljs/schema@0.1.3': + resolution: {integrity: sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==} + engines: {node: '>=8'} + '@jest/schemas@29.6.3': resolution: {integrity: sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA==} engines: {node: ^14.15.0 || ^16.10.0 || >=18.0.0} @@ -1759,6 +1824,10 @@ packages: '@jspm/core@2.0.1': resolution: {integrity: sha512-Lg3PnLp0QXpxwLIAuuJboLeRaIhrgJjeuh797QADg3xz8wGLugQOS5DpsE8A6i6Adgzf+bacllkKZG3J0tGfDw==} + '@lukeed/ms@2.0.2': + resolution: {integrity: sha512-9I2Zn6+NJLfaGoz9jN3lpwDgAYvfGeNYdbAIjJOqzs4Tpc+VU3Jqq4IofSUBKajiDS8k9fZIg18/z13mpk1bsA==} + engines: {node: '>=8'} + '@mdx-js/mdx@2.3.0': resolution: {integrity: sha512-jLuwRlz8DQfQNiUCJR50Y09CGPq3fLtmtUQfVrj79E0JWu3dvsVcxVIcfhR5h0iXu+/z++zDrYeiJqifRynJkA==} @@ -2867,6 +2936,9 @@ packages: '@types/http-errors@2.0.4': resolution: {integrity: sha512-D0CFMMtydbJAegzOyHjtiKPLlvnm3iTZyZRSZoLq2mRhDdmLfIWOCYPfQJ4cu2erKghU++QvjcUjp/5h7hESpA==} + '@types/istanbul-lib-coverage@2.0.6': + resolution: {integrity: sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w==} + '@types/jsonfile@6.1.4': resolution: {integrity: sha512-D5qGUYwjvnNNextdU59/+fI+spnwtTFmyQP0h+PfIOSkNfpU6AOICUOkm4i0OnSk+NyjdPJrxCDro0sJsWlRpQ==} @@ -2903,9 +2975,6 @@ packages: '@types/node@18.19.50': resolution: {integrity: sha512-xonK+NRrMBRtkL1hVCc3G+uXtjh1Al4opBLjqVmipe5ZAaBYWW6cNAiBVZ1BvmkBhep698rP3UM3aRAdSALuhg==} - '@types/node@20.16.5': - resolution: {integrity: sha512-VwYCweNo3ERajwy0IUlqqcyZ8/A7Zwa9ZP3MnENWcB11AejO+tLy3pu850goUW2FC/IJMdZUfKpX/yxL1gymCA==} - '@types/node@22.5.5': resolution: {integrity: sha512-Xjs4y5UPO/CLdzpgR6GirZJx36yScjh73+2NlLlkFRSoQN8B0DpfXPdZGnvVmLRLOsqDpOfTNv7D9trgGhmOIA==} @@ -3096,6 +3165,9 @@ packages: resolution: {integrity: sha512-h8lQ8tacZYnR3vNQTgibj+tODHI5/+l06Au2Pcriv/Gmet0eaj4TwWH41sO9wnHDiQsEj19q0drzdWdeAHtweg==} engines: {node: '>=6.5'} + abstract-logging@2.0.1: + resolution: {integrity: sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA==} + accepts@1.3.8: resolution: {integrity: sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==} engines: {node: '>= 0.6'} @@ -3145,9 +3217,20 @@ packages: resolution: {integrity: sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==} engines: {node: '>=8'} + ajv-formats@3.0.1: + resolution: {integrity: sha512-8iUql50EUR+uUcdRQ3HDqa6EVyo3docL8g5WJ3FNcWmu62IbkGUue/pEyLBW8VGKKucTPgqeks4fIU1DA4yowQ==} + peerDependencies: + ajv: ^8.0.0 + peerDependenciesMeta: + ajv: + optional: true + ajv@6.12.6: resolution: {integrity: sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==} + ajv@8.17.1: + resolution: {integrity: sha512-B/gBuNg5SiMTrPkC+A2+cW0RszwxYmn6VYxB/inlBStS5nx6xHIt/ehKRhIMhqusl7a8LjQoZnjCs5vhwxOQ1g==} + ansi-align@3.0.1: resolution: {integrity: sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w==} @@ -3244,6 +3327,10 @@ packages: asynckit@0.4.0: resolution: {integrity: sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==} + atomic-sleep@1.0.0: + resolution: {integrity: sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ==} + engines: {node: '>=8.0.0'} + atomically@2.0.3: resolution: {integrity: sha512-kU6FmrwZ3Lx7/7y3hPS5QnbJfaohcIul5fGqf7ok+4KklIEk9tJ0C2IQPdacSbVUWv6zVHXEBWoWd6NrVMT7Cw==} @@ -3258,6 +3345,9 @@ packages: resolution: {integrity: sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==} engines: {node: '>= 0.4'} + avvio@9.0.0: + resolution: {integrity: sha512-UbYrOXgE/I+knFG+3kJr9AgC7uNo8DG+FGGODpH9Bj1O1kL/QDjBXnTem9leD3VdQKtaHjV3O85DQ7hHh4IIHw==} + babel-core@7.0.0-bridge.0: resolution: {integrity: sha512-poPX9mZH/5CSanm50Q+1toVci6pv5KSRv/5TWCwtzQS5XEwn40BcCrgIeMFWP9CKKIniKXNxoIOnOq4VVlGXhg==} peerDependencies: @@ -3343,6 +3433,9 @@ packages: buffer@5.7.1: resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==} + buffer@6.0.3: + resolution: {integrity: sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==} + busboy@1.6.0: resolution: {integrity: sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==} engines: {node: '>=10.16.0'} @@ -3355,6 +3448,16 @@ packages: resolution: {integrity: sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==} engines: {node: '>= 0.8'} + c8@10.1.2: + resolution: {integrity: sha512-Qr6rj76eSshu5CgRYvktW0uM0CFY0yi4Fd5D0duDXO6sYinyopmftUiJVuzBQxQcwQLor7JWDVRP+dUfCmzgJw==} + engines: {node: '>=18'} + hasBin: true + peerDependencies: + monocart-coverage-reports: ^2 + peerDependenciesMeta: + monocart-coverage-reports: + optional: true + cac@6.7.14: resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==} engines: {node: '>=8'} @@ -3442,6 +3545,10 @@ packages: resolution: {integrity: sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==} engines: {node: '>= 8.10.0'} + chokidar@4.0.1: + resolution: {integrity: sha512-n8enUVCED/KVRQlab1hr3MVpcVMvxtZjmEa956u+4YijlmQED223XMSYj2tLuKvr4jcCTzNNMpQDUer72MMmzA==} + engines: {node: '>= 14.16.0'} + chownr@1.1.4: resolution: {integrity: sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==} @@ -3527,6 +3634,9 @@ packages: color@3.2.1: resolution: {integrity: sha512-aBl7dZI9ENN6fUGC7mWpMTPNHmWUSNan9tuWN6ahh5ZLNk9baLJOnSMlrQkHcrfFgz2/RigjUVAjdx36VcemKA==} + colorette@2.0.20: + resolution: {integrity: sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==} + colorspace@1.1.4: resolution: {integrity: sha512-BgvKJiuVu1igBUF2kEjRCZXol6wiiGbY5ipL/oVPwm0BL9sIpMIzM8IK7vwuxIIzOXMV3Ey5w+vxhm0rR/TN8w==} @@ -3553,6 +3663,9 @@ packages: resolution: {integrity: sha512-U7VdrJFnJgo4xjrHpTzu0yrHPGImdsmD95ZlgYSEajAn2JKzDhDTPG9kBTefmObL2w/ngeZnilk+OV9CG3d7UA==} engines: {node: '>= 6'} + commist@3.2.0: + resolution: {integrity: sha512-4PIMoPniho+LqXmpS5d3NuGYncG6XWlkBSVGiWycL22dd42OYdUGil2CWuzklaJoNxyxUSpO4MKIBU94viWNAw==} + commondir@1.0.1: resolution: {integrity: sha512-W9pAhw0ja1Edb5GVdIF1mjZw/ASI0AlShXM83UUGe2DVr5TdAPEA1OA8m/g8zWp9x6On7gqufY+FatDbC3MDQg==} @@ -3567,6 +3680,11 @@ packages: concat-map@0.0.1: resolution: {integrity: sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==} + concurrently@9.0.1: + resolution: {integrity: sha512-wYKvCd/f54sTXJMSfV6Ln/B8UrfLBKOYa+lzc6CHay3Qek+LorVSBdMVfyewFhRbH0Rbabsk4D+3PL/VjQ5gzg==} + engines: {node: '>=18'} + hasBin: true + confbox@0.1.7: resolution: {integrity: sha512-uJcB/FKZtBMCJpK8MQji6bJHgu1tixKPxRLeGkNzBoOZzpnZUJm0jm2/sBDWcuBx1dYgxV4JU+g5hmNxCyAmdA==} @@ -3673,6 +3791,9 @@ packages: date-fns@3.6.0: resolution: {integrity: sha512-fRHTG8g/Gif+kSh50gaGEdToemgfj74aRX3swtiouboip5JDLAyDE9F11nHMIcvOaXeOC6D7SpNhi7uFyB7Uww==} + dateformat@4.6.3: + resolution: {integrity: sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA==} + dayjs@1.11.13: resolution: {integrity: sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==} @@ -4095,6 +4216,10 @@ packages: resolution: {integrity: sha512-sPNTqiMokAvV048P2c9+foqVJzk49o6d4e0D/sq5jog3pw+4kBgyR0gaM1FM7Mx6Kzd9dztesh9oYz1LWWOpzw==} engines: {node: '>=10'} + events@3.3.0: + resolution: {integrity: sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==} + engines: {node: '>=0.8.x'} + execa@5.1.1: resolution: {integrity: sha512-8uSpZZocAZRBAPIEINJj3Lo9HyGitllczc27Eh5YYojjMFMn8yHMDMaUHE2Jqfq05D/wucwI4JGURyXt1vchyg==} engines: {node: '>=10'} @@ -4128,6 +4253,12 @@ packages: resolution: {integrity: sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==} engines: {node: '>=4'} + fast-copy@3.0.2: + resolution: {integrity: sha512-dl0O9Vhju8IrcLndv2eU4ldt1ftXMqqfgN4H1cpmGV7P6jeB9FwpN9a2c8DPGE1Ys88rNUJVYDHq73CGAGOPfQ==} + + fast-decode-uri-component@1.0.1: + resolution: {integrity: sha512-WKgKWg5eUxvRZGwW8FvfbaH7AXSh2cL+3j5fMGzUMCxWBJ3dV3a7Wz8y2f/uQ0e3B6WmodD3oS54jTQ9HVTIIg==} + fast-deep-equal@2.0.1: resolution: {integrity: sha512-bCK/2Z4zLidyB4ReuIsvALH6w31YfAQDmXMqMx6FyfHqvBxtjC0eRumeSu4Bs3XtXwpyIywtSTrVT99BxY1f9w==} @@ -4141,12 +4272,45 @@ packages: fast-json-stable-stringify@2.1.0: resolution: {integrity: sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==} + fast-json-stringify@6.0.0: + resolution: {integrity: sha512-FGMKZwniMTgZh7zQp9b6XnBVxUmKVahQLQeRQHqwYmPDqDhcEKZ3BaQsxelFFI5PY7nN71OEeiL47/zUWcYe1A==} + fast-levenshtein@2.0.6: resolution: {integrity: sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw==} + fast-querystring@1.1.2: + resolution: {integrity: sha512-g6KuKWmFXc0fID8WWH0jit4g0AGBoJhCkJMb1RmbsSEUNvQ+ZC8D6CUZ+GtF8nMzSPXnhiePyyqqipzNNEnHjg==} + + fast-redact@3.5.0: + resolution: {integrity: sha512-dwsoQlS7h9hMeYUq1W++23NDcBLV4KqONnITDV9DjfS3q1SgDGVrBdvvTLUotWtPSD7asWDV9/CmsZPy8Hf70A==} + engines: {node: '>=6'} + + fast-safe-stringify@2.1.1: + resolution: {integrity: sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA==} + fast-text-encoding@1.0.6: resolution: {integrity: sha512-VhXlQgj9ioXCqGstD37E/HBeqEGV/qOD/kmbVG8h5xKBYvM1L3lR1Zn4555cQ8GkYbJa8aJSipLPndE1k6zK2w==} + fast-uri@2.4.0: + resolution: {integrity: sha512-ypuAmmMKInk5q7XcepxlnUWDLWv4GFtaJqAzWKqn62IpQ3pejtr5dTVbt3vwqVaMKmkNR55sTT+CqUKIaT21BA==} + + fast-uri@3.0.1: + resolution: {integrity: sha512-MWipKbbYiYI0UC7cl8m/i/IWTqfC8YXsqjzybjddLsFjStroQzsHXkc73JutMvBiXmOvapk+axIl79ig5t55Bw==} + + fastify-cli@7.0.1: + resolution: {integrity: sha512-2+ghTcw78K0LAoYoOl3l8ZyOunALjumuy9g2Kz0RdnqEnIDaHfEKOJvyo7Q707lh130RRzWSUmdtpO0dCAfp3A==} + hasBin: true + + fastify-plugin@5.0.1: + resolution: {integrity: sha512-HCxs+YnRaWzCl+cWRYFnHmeRFyR5GVnJTAaCJQiYzQSDwK9MgJdyAsuL3nh0EWRCYMgQ5MeziymvmAhUHYHDUQ==} + + fastify-tsconfig@2.0.0: + resolution: {integrity: sha512-pvYwdtbZUJr/aTD7ZE0rGlvtYpx7IThHKVLBoqCKmT3FJpwm23XA2+PDmq8ZzfqqG4ajpyrHd5bkIixcIFjPhQ==} + engines: {node: '>=18.0.0'} + + fastify@5.0.0: + resolution: {integrity: sha512-Qe4dU+zGOzg7vXjw4EvcuyIbNnMwTmcuOhlOrOJsgwzvjEZmsM/IeHulgJk+r46STjdJS/ZJbxO8N70ODXDMEQ==} + fastq@1.17.1: resolution: {integrity: sha512-sRVD3lWVIXWg6By68ZN7vho9a1pQcN/WBFaAAsDDFzlJjvoGx0P8z7V1t72grFJfJhu3YPZBuu25f7Kaw2jN1w==} @@ -4191,6 +4355,10 @@ packages: resolution: {integrity: sha512-wXZV5emFEjrridIgED11OoUKLxiYjAcqot/NJdAkOhlJ+vGzwhOAfcG5OX1jP+S0PcjEn8bdMJv+g2jwQ3Onig==} engines: {node: '>=8'} + find-my-way@9.0.1: + resolution: {integrity: sha512-/5NN/R0pFWuff16TMajeKt2JyiW+/OE8nOO8vo1DwZTxLaIURb7lcBYPIgRPh61yCNh9l8voeKwcrkUzmB00vw==} + engines: {node: '>=14'} + find-up@3.0.0: resolution: {integrity: sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==} engines: {node: '>=6'} @@ -4311,6 +4479,10 @@ packages: generic-names@4.0.0: resolution: {integrity: sha512-ySFolZQfw9FoDb3ed9d80Cm9f0+r7qj+HJkWjeD9RBfpxEVTlVhol+gvaQB/78WbwYfbnNh8nWHHBSlg072y6A==} + generify@4.2.0: + resolution: {integrity: sha512-b4cVhbPfbgbCZtK0dcUc1lASitXGEAIqukV5DDAyWm25fomWnV+C+a1yXvqikcRZXHN2j0pSDyj3cTfzq8pC7Q==} + hasBin: true + gensync@1.0.0-beta.2: resolution: {integrity: sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg==} engines: {node: '>=6.9.0'} @@ -4388,6 +4560,10 @@ packages: engines: {node: '>=16 || 14 >=14.17'} hasBin: true + glob@10.4.5: + resolution: {integrity: sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg==} + hasBin: true + glob@11.0.0: resolution: {integrity: sha512-9UiX/Bl6J2yaBbxKoEBRm4Cipxgok8kQYcOPEhScPwebu2I0HoQOuYdIO6S3hLuWoZgpDpwQZMzTFxgpkyT76g==} engines: {node: 20 || >=22} @@ -4397,6 +4573,11 @@ packages: resolution: {integrity: sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==} deprecated: Glob versions prior to v9 are no longer supported + glob@8.1.0: + resolution: {integrity: sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ==} + engines: {node: '>=12'} + deprecated: Glob versions prior to v9 are no longer supported + global-directory@4.0.1: resolution: {integrity: sha512-wHTUcDUoZ1H5/0iVqEudYW4/kAlN5cZ3j/bXn0Dpbizl9iaUVeWSHqiOjsgk6OW2bkLclbBjzewBz6weQ1zA2Q==} engines: {node: '>=18'} @@ -4531,6 +4712,12 @@ packages: resolution: {integrity: sha512-g+HZqgfbpXdCkme/Cd/mZkV0aV3BZZZSugecH03kl38m/Kmdx8jKjBikpDj2cr+Iynv4KpYEviojNdTJActJAg==} engines: {node: '>=16.0.0'} + help-me@4.2.0: + resolution: {integrity: sha512-TAOnTB8Tz5Dw8penUuzHVrKNKlCIbwwbHnXraNJxPwf8LRtE2HlM84RYuezMFcwOJmoYOCWVDyJ8TQGxn9PgxA==} + + help-me@5.0.0: + resolution: {integrity: sha512-7xgomUX6ADmcYzFik0HzAxh/73YlKR9bmFzf51CZwR+b6YtzU2m0u49hQCqV6SvlqIqsaxovfwdvbnsw3b/zpg==} + hosted-git-info@2.8.9: resolution: {integrity: sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==} @@ -4541,6 +4728,9 @@ packages: html-entities@2.5.2: resolution: {integrity: sha512-K//PSRMQk4FZ78Kyau+mZurHn3FH0Vwr+H36eE0rPbeYkRRi9YxceYPhuN60UwWorxyKHhqoAJl2OFKa4BVtaA==} + html-escaper@2.0.2: + resolution: {integrity: sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==} + html-tags@3.3.1: resolution: {integrity: sha512-ztqyC3kLto0e9WbNp0aeP+M3kTt+nbaIveGmUxAtZa+8iFgKLUOD4YKM5j+f3QD89bra7UeumolZHKuOXnTmeQ==} engines: {node: '>=8'} @@ -4719,6 +4909,11 @@ packages: is-deflate@1.0.0: resolution: {integrity: sha512-YDoFpuZWu1VRXlsnlYMzKyVRITXj7Ej/V9gXQ2/pAe7X1J7M/RNOqaIYi6qUn+B7nGyB9pDXrv02dsB58d2ZAQ==} + is-docker@2.2.1: + resolution: {integrity: sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==} + engines: {node: '>=8'} + hasBin: true + is-extglob@2.1.1: resolution: {integrity: sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==} engines: {node: '>=0.10.0'} @@ -4878,6 +5073,10 @@ packages: isarray@2.0.5: resolution: {integrity: sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==} + isbinaryfile@4.0.10: + resolution: {integrity: sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw==} + engines: {node: '>= 8.0.0'} + isbot@5.1.17: resolution: {integrity: sha512-/wch8pRKZE+aoVhRX/hYPY1C7dMCeeMyhkQLNLNlYAbGQn9bkvMB8fOUXNnk5I0m4vDYbBJ9ciVtkr9zfBJ7qA==} engines: {node: '>=18'} @@ -4889,10 +5088,25 @@ packages: resolution: {integrity: sha512-WhB9zCku7EGTj/HQQRz5aUQEUeoQZH2bWcltRErOpymJ4boYE6wL9Tbr23krRPSZ+C5zqNSrSw+Cc7sZZ4b7vg==} engines: {node: '>=0.10.0'} + istanbul-lib-coverage@3.2.2: + resolution: {integrity: sha512-O8dpsF+r0WV/8MNRKfnmrtCWhuKjxrq2w+jpzBL5UZKTi2LeVWnWOmWRxFlesJONmc+wLAGvKQZEOanko0LFTg==} + engines: {node: '>=8'} + + istanbul-lib-report@3.0.1: + resolution: {integrity: sha512-GCfE1mtsHGOELCU8e/Z7YWzpmybrx/+dSTfLrvY8qRmaY6zXTKWn6WQIjaAFw069icm6GVMNkgu0NzI4iPZUNw==} + engines: {node: '>=10'} + + istanbul-reports@3.1.7: + resolution: {integrity: sha512-BewmUXImeuRk2YY0PVbxgKAysvhRPUQE0h5QRM++nVWyubKGV0l8qQ5op8+B2DOmwSe63Jivj0BjkPQVf8fP5g==} + engines: {node: '>=8'} + jackspeak@2.3.6: resolution: {integrity: sha512-N3yCS/NegsOBokc8GAdM8UcmfsKiSS8cipheD/nivzr700H+nsMOxJjQnvwOcRYVuFkdH0wGUvW2WbXGmrZGbQ==} engines: {node: '>=14'} + jackspeak@3.4.3: + resolution: {integrity: sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw==} + jackspeak@4.0.1: resolution: {integrity: sha512-cub8rahkh0Q/bw1+GxP7aeSe29hHHn2V4m29nnDlvCdlgU+3UGxkZp7Z53jLUdpX3jdTO0nJZUDl3xvbWc2Xog==} engines: {node: 20 || >=22} @@ -4904,6 +5118,10 @@ packages: resolution: {integrity: sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w==} hasBin: true + joycon@3.1.1: + resolution: {integrity: sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==} + engines: {node: '>=10'} + js-beautify@1.15.1: resolution: {integrity: sha512-ESjNzSlt/sWE8sciZH8kBF8BPlwXPwhR6pWKAw8bw4Bwj+iZcnKW6ONWUutJ7eObuBZQpiIb8S7OYspWrKt7rA==} engines: {node: '>=14'} @@ -4960,9 +5178,15 @@ packages: resolution: {integrity: sha512-fi0NG4bPjCHunUJffmLd0gxssIgkNmArMvis4iNah6Owg1MCJjWhEcDLmsK6iGkJq3tHwbDkTlce70/tmXN4cQ==} engines: {node: ^14.17.0 || ^16.13.0 || >=18.0.0} + json-schema-ref-resolver@1.0.1: + resolution: {integrity: sha512-EJAj1pgHc1hxF6vo2Z3s69fMjO1INq6eGHXZ8Z6wCQeldCuwxGK9Sxf4/cScGn3FZubCVUehfWtcDM/PLteCQw==} + json-schema-traverse@0.4.1: resolution: {integrity: sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==} + json-schema-traverse@1.0.0: + resolution: {integrity: sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==} + json-stable-stringify-without-jsonify@1.0.1: resolution: {integrity: sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw==} @@ -5071,6 +5295,9 @@ packages: resolution: {integrity: sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ==} engines: {node: '>= 0.8.0'} + light-my-request@6.0.0: + resolution: {integrity: sha512-kFkFXrmKCL0EEeOmJybMH5amWFd+AFvlvMlvFTRxCUwbhfapZqDmeLMPoWihntnYY6JpoQDE9k+vOzObF1fDqg==} + lilconfig@2.1.0: resolution: {integrity: sha512-utWOt/GHzuUxnLKxB6dk81RoOeoNeHgbrXiuGk4yyF5qlRz+iIVWu56E2fqGHFrXz0QNUhLB/8nKqvRH66JKGQ==} engines: {node: '>=10'} @@ -5191,9 +5418,16 @@ packages: resolution: {integrity: sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==} engines: {node: '>=8'} + make-dir@4.0.0: + resolution: {integrity: sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==} + engines: {node: '>=10'} + make-error@1.3.6: resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} + makeerror@1.0.12: + resolution: {integrity: sha512-JmqCvUhmt43madlpFzG4BQzG2Z3m6tvQDNKdClZnO3VbIudJYmxsT0FNJMeiB2+JTSlTQTSbU8QdesVmwJcmLg==} + map-or-similar@1.5.0: resolution: {integrity: sha512-0aF7ZmVon1igznGI4VS30yugpduQW3y3GkcgGJOp7d8x8QrizhigUxjI/m2UojsXXto+jLAH3KSz+xOJTiORjg==} @@ -5408,6 +5642,10 @@ packages: minimatch@3.1.2: resolution: {integrity: sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==} + minimatch@5.1.6: + resolution: {integrity: sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==} + engines: {node: '>=10'} + minimatch@9.0.1: resolution: {integrity: sha512-0jWhJpD/MdhPXwPuiRkCbfYfSKp2qnn2eOc279qI7f+osl/l+prKSrvhg157zSYvx/1nmgn2NqdT6k2Z7zSH9w==} engines: {node: '>=16 || 14 >=14.17'} @@ -5651,6 +5889,10 @@ packages: ohash@1.1.4: resolution: {integrity: sha512-FlDryZAahJmEF3VR3w1KogSEdWX3WhA5GPakFx4J81kEAiHyLMpdLLElS8n8dfNadMgAne/MywcvmogzscVt4g==} + on-exit-leak-free@2.1.2: + resolution: {integrity: sha512-0eJJY6hXLGf1udHwfNftBqH+g73EU4B504nZeKpz1sYRKafAghwxEJunB2O7rDZkL4PGfsMVnTXZ2EjibbqcsA==} + engines: {node: '>=14.0.0'} + on-finished@2.3.0: resolution: {integrity: sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==} engines: {node: '>= 0.8'} @@ -5861,6 +6103,20 @@ packages: resolution: {integrity: sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==} engines: {node: '>=6'} + pino-abstract-transport@1.2.0: + resolution: {integrity: sha512-Guhh8EZfPCfH+PMXAb6rKOjGQEoy0xlAIn+irODG5kgfYV+BQ0rGYYWTIel3P5mmyXqkYkPmdIkywsn6QKUR1Q==} + + pino-pretty@11.2.2: + resolution: {integrity: sha512-2FnyGir8nAJAqD3srROdrF1J5BIcMT4nwj7hHSc60El6Uxlym00UbCCd8pYIterstVBFlMyF1yFV8XdGIPbj4A==} + hasBin: true + + pino-std-serializers@7.0.0: + resolution: {integrity: sha512-e906FRY0+tV27iq4juKzSYPbUj2do2X2JX4EzSca1631EB2QJQUqGbDuERal7LCtOpxl6x3+nvo9NPZcmjkiFA==} + + pino@9.4.0: + resolution: {integrity: sha512-nbkQb5+9YPhQRz/BeQmrWpEknAaqjpAqRK8NwJpmrX/JHu7JuZC5G1CeAwJDJfGes4h+YihC6in3Q2nGb+Y09w==} + hasBin: true + pirates@4.0.6: resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==} engines: {node: '>= 6'} @@ -5876,6 +6132,10 @@ packages: pkg-types@1.2.0: resolution: {integrity: sha512-+ifYuSSqOQ8CqP4MbZA5hDpb97n3E8SVWdJe+Wms9kj745lmd3b7EZJiqvmLwAlmRfjrI7Hi5z3kdBJ93lFNPA==} + pkg-up@3.1.0: + resolution: {integrity: sha512-nDywThFk1i4BQK4twPQ6TA4RT8bDY96yeuCVBWL3ePARCiEKDRSrNGbFIgUJpLp+XeIR65v8ra7WuJOFUBtkMA==} + engines: {node: '>=8'} + playwright-core@1.47.0: resolution: {integrity: sha512-1DyHT8OqkcfCkYUD9zzUTfg7EfTd+6a8MkD/NWOvjo0u/SCNd5YmY/lJwFvUZOxJbWNds+ei7ic2+R/cRz/PDg==} engines: {node: '>=18'} @@ -6024,6 +6284,9 @@ packages: process-nextick-args@2.0.1: resolution: {integrity: sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==} + process-warning@4.0.0: + resolution: {integrity: sha512-/MyYDxttz7DfGMMHiysAsFE4qF+pQYAA8ziO/3NcRVrQ5fSk+Mns4QZA/oRPFzvcqNoVJXQNWNAsdwBXLUkQKw==} + process@0.11.10: resolution: {integrity: sha512-cdGef/drWFoydD1JsMzuFf8100nZl+GT+yacc2bEced5f9Rjk4z+WtFUTBu9PhOi9j/jfmBPu0mMEY4wIdAF8A==} engines: {node: '>= 0.6.0'} @@ -6105,6 +6368,9 @@ packages: queue-microtask@1.2.3: resolution: {integrity: sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==} + quick-format-unescaped@4.0.4: + resolution: {integrity: sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg==} + range-parser@1.2.1: resolution: {integrity: sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==} engines: {node: '>= 0.6'} @@ -6217,10 +6483,22 @@ packages: resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==} engines: {node: '>= 6'} + readable-stream@4.5.2: + resolution: {integrity: sha512-yjavECdqeZ3GLXNgRXgeQEdz9fvDDkNKyHnbHRFtOr7/LcfgBcmct7t/ET+HaCTqfh06OzoAxrkN/IfjJBVe+g==} + engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0} + readdirp@3.6.0: resolution: {integrity: sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==} engines: {node: '>=8.10.0'} + readdirp@4.0.1: + resolution: {integrity: sha512-GkMg9uOTpIWWKbSsgwb5fA4EavTR+SG/PMPoAY8hkhHfEEY0/vqljY+XHqtDf2cr2IJtoNRDbrrEpZUiZCkYRw==} + engines: {node: '>= 14.16.0'} + + real-require@0.2.0: + resolution: {integrity: sha512-57frrGM/OCTLqLOAh0mhVA9VBMHd+9U7Zb2THMGdBUoZVOtGbJzjxsYGDJ3A9AYYCP4hn6y1TVbaOfzWtm5GFg==} + engines: {node: '>= 12.13.0'} + recast@0.23.9: resolution: {integrity: sha512-Hx/BGIbwj+Des3+xy5uAtAbdCyqK9y9wbBcDFDYanLS9JnMqf7OeF87HQwUimE87OEc72mr6tkKUKMBBL+hF9Q==} engines: {node: '>= 4'} @@ -6288,6 +6566,10 @@ packages: resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==} engines: {node: '>=0.10.0'} + require-from-string@2.0.2: + resolution: {integrity: sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==} + engines: {node: '>=0.10.0'} + require-in-the-middle@7.4.0: resolution: {integrity: sha512-X34iHADNbNDfr6OTStIAHWSAvvKQRYgLO6duASaVf7J2VA3lvmNYboAHOuLC2huav1IwgZJtyEcJCKVzFxOSMQ==} engines: {node: '>=8.6.0'} @@ -6302,6 +6584,10 @@ packages: resolution: {integrity: sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==} engines: {node: '>=4'} + resolve-from@5.0.0: + resolution: {integrity: sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==} + engines: {node: '>=8'} + resolve-pkg-maps@1.0.0: resolution: {integrity: sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==} @@ -6317,6 +6603,10 @@ packages: resolution: {integrity: sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==} engines: {node: '>=8'} + ret@0.5.0: + resolution: {integrity: sha512-I1XxrZSQ+oErkRR4jYbAyEEu2I0avBvvMM5JN+6EBprOGRCs63ENqZ3vjavq8fBw2+62G5LF5XelKwuJpcvcxw==} + engines: {node: '>=10'} + retry-request@5.0.2: resolution: {integrity: sha512-wfI3pk7EE80lCIXprqh7ym48IHYdwmAAzESdbU8Q9l7pnRCk9LEhpbOTNKjz6FARLm/Bl5m+4F0ABxOkYUujSQ==} engines: {node: '>=12'} @@ -6333,6 +6623,9 @@ packages: resolution: {integrity: sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==} engines: {iojs: '>=1.0.0', node: '>=0.10.0'} + rfdc@1.4.1: + resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==} + rimraf@2.6.3: resolution: {integrity: sha512-mwqeW5XsA2qAejG46gYdENaxXjx9onRNCfn7L0duuP4hCuTIi/QO7PDK07KJfp1d+izWPrzEJDcSqBa0OZQriA==} deprecated: Rimraf versions prior to v4 are no longer supported @@ -6375,6 +6668,9 @@ packages: resolution: {integrity: sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==} engines: {node: '>= 0.4'} + safe-regex2@4.0.0: + resolution: {integrity: sha512-Hvjfv25jPDVr3U+4LDzBuZPPOymELG3PYcSk5hcevooo1yxxamQL/bHs/GrEPGmMoMEwRrHVGiCA1pXi97B8Ew==} + safe-stable-stringify@2.5.0: resolution: {integrity: sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==} engines: {node: '>=10'} @@ -6389,6 +6685,9 @@ packages: scheduler@0.23.2: resolution: {integrity: sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==} + secure-json-parse@2.7.0: + resolution: {integrity: sha512-6aU+Rwsezw7VR8/nyvKTx8QpWH9FrcYiXXlqC4z5d5XQBDRqtbfsRjnwGyqbi3gddNtWHuEk9OANUotL26qKUw==} + selderee@0.11.0: resolution: {integrity: sha512-5TF+l7p4+OsnP8BCCvSyZiSPc4x4//p5uPwK8TCnVPJYRmU2aYKMpOXvw8zM5a5JvuuCGN1jmsMwuU2W02ukfA==} @@ -6504,6 +6803,9 @@ packages: resolution: {integrity: sha512-DmeAkF6cwM9jSfmp6Dr/5/mfMwb5Z5qRrSXLpo3Fq5SqyU8CMF15jIN4ZhfSwu35ksM1qmHZDQ/DK5XTccSTvA==} engines: {node: '>=10.2.0'} + sonic-boom@4.1.0: + resolution: {integrity: sha512-NGipjjRicyJJ03rPiZCJYjwlsuP2d1/5QUviozRXC7S3WdVWNK5e3Ojieb9CCyfhq2UC+3+SRd9nG3I2lPRvUw==} + source-map-js@1.2.1: resolution: {integrity: sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==} engines: {node: '>=0.10.0'} @@ -6534,6 +6836,13 @@ packages: spdx-license-ids@3.0.20: resolution: {integrity: sha512-jg25NiDV/1fLtSgEgyvVyDunvaNHbuwF9lfNV17gSmPFAlYzdfNBlLtLzXTevwkPj7DhGbmN9VnmJIgLnhvaBw==} + split2@3.2.2: + resolution: {integrity: sha512-9NThjpgZnifTkJpzTZ7Eue85S49QwpNhZTq6GRJwObb6jnLFNGB7Qm73V5HewTROPyxD0C29xqmaI68bQtV+hg==} + + split2@4.2.0: + resolution: {integrity: sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==} + engines: {node: '>= 10.x'} + sprintf-js@1.0.3: resolution: {integrity: sha512-D9cPgkvLlV3t3IzL0D0YLvGA9Ahk4PcvVwUbN0dSGr1aP0Nrt4AEnTUbuGvquEC0mA64Gqt1fzirlRs5ibXx8g==} @@ -6696,6 +7005,10 @@ packages: resolution: {integrity: sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==} engines: {node: '>=8'} + supports-color@8.1.1: + resolution: {integrity: sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==} + engines: {node: '>=10'} + supports-preserve-symlinks-flag@1.0.0: resolution: {integrity: sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==} engines: {node: '>= 0.4'} @@ -6751,6 +7064,10 @@ packages: resolution: {integrity: sha512-rcdty1xZ2/BkWa4ANjWRp4JGpda2quksXIHgn5TMjNBPZfwzJIgR68DKfSYiTL+CZWowDX/sbOo5ME/FRURvYQ==} engines: {node: '>=18'} + test-exclude@7.0.1: + resolution: {integrity: sha512-pFYqmTw68LXVjeWJMST4+borgQP2AyMNbg1BpZh9LbyhUeNkeaPF9gzfPGUAnSMV3qPYdWUwDIjjCLiSDOl7vg==} + engines: {node: '>=18'} + text-hex@1.0.0: resolution: {integrity: sha512-uuVGNWzgJ4yhRaNSiubPY7OjISw4sw4E5Uv0wbjp+OzcbmVU/rsT8ujgcXJhn9ypzsgr5vlzpPqP+MBBKcGvbg==} @@ -6764,6 +7081,9 @@ packages: thenify@3.3.1: resolution: {integrity: sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==} + thread-stream@3.1.0: + resolution: {integrity: sha512-OqyPZ9u96VohAyMfJykzmivOrY2wfMSf3C5TtFJVgN+Hm6aj+voFhlK+kZEIv2FBh1X6Xp3DlnCOfEQ3B2J86A==} + through2@2.0.5: resolution: {integrity: sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==} @@ -6793,6 +7113,9 @@ packages: resolution: {integrity: sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==} engines: {node: '>=0.6.0'} + tmpl@1.0.5: + resolution: {integrity: sha512-3f0uOEAQwIqGuWW2MVzYg8fV/QNnc/IpuJNG837rLuczAaLVHslWHZQj4IGiEl5Hs3kkbhwL9Ab7Hrsmuj+Smw==} + to-fast-properties@2.0.0: resolution: {integrity: sha512-/OaKK0xYrs3DmxRYqL/yDc+FxFUVYhDlXMhRmv3z915w2HF1tnN1omB354j8VUGO/hbRzyD6Y3sA7v7GS/ceog==} engines: {node: '>=4'} @@ -6801,6 +7124,10 @@ packages: resolution: {integrity: sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==} engines: {node: '>=8.0'} + toad-cache@3.7.0: + resolution: {integrity: sha512-/m8M+2BJUpoJdgAHoG+baCwBT+tf2VraSfkBgl0Y00qIWt41DJ8R5B8nsEw0I58YwF5IZH6z24/2TobDKnqSWw==} + engines: {node: '>=12'} + toidentifier@1.0.1: resolution: {integrity: sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==} engines: {node: '>=0.6'} @@ -6819,6 +7146,10 @@ packages: tr46@0.0.3: resolution: {integrity: sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==} + tree-kill@1.2.2: + resolution: {integrity: sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==} + hasBin: true + trim-lines@3.0.1: resolution: {integrity: sha512-kRj8B+YHZCc9kQYdWfJB2/oUl9rA99qbowYYBtr4ui4mZyAQ2JpvVBd/6U2YloATfqBhBTSMhTpgBHtU0Mf3Rg==} @@ -7133,6 +7464,10 @@ packages: v8-compile-cache-lib@3.0.1: resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==} + v8-to-istanbul@9.3.0: + resolution: {integrity: sha512-kiGUalWN+rgBJ/1OHZsBtU4rXZOfj/7rKQxULKlIzwzQSvMJUUNgPwJEEh7gU6xEVxC0ahoOBvN2YI8GH6FNgA==} + engines: {node: '>=10.12.0'} + valibot@0.41.0: resolution: {integrity: sha512-igDBb8CTYr8YTQlOKgaN9nSS0Be7z+WRuaeYqGf3Cjz3aKmSnqEmYnkfVjzIuumGqfHpa3fLIvMEAfhrpqN8ng==} peerDependencies: @@ -7247,6 +7582,9 @@ packages: resolution: {integrity: sha512-3eBwRyEln6E1MSzcxcVpQIhRG8Q1jLvEqRmCZqS3dsfXEDR/AhOF4d+jHg1qvDCpYaVRZjENPQyrVxAkQqxPgQ==} engines: {node: '>=6.0.0'} + walker@1.0.8: + resolution: {integrity: sha512-ts/8E8l5b7kY0vlWLewOkDXMmPdLcVV4GmOQLyxuSswIJsweeFZtAsMF7k1Nszz+TYBQrlYRmzOnr398y1JemQ==} + wcwidth@1.0.1: resolution: {integrity: sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg==} @@ -8308,6 +8646,8 @@ snapshots: '@base2/pretty-print-object@1.0.1': {} + '@bcoe/v8-coverage@0.2.3': {} + '@biomejs/biome@1.8.3': optionalDependencies: '@biomejs/cli-darwin-arm64': 1.8.3 @@ -8361,7 +8701,6 @@ snapshots: '@cspotcode/source-map-support@0.8.1': dependencies: '@jridgewell/trace-mapping': 0.3.9 - optional: true '@dabh/diagnostics@2.0.3': dependencies: @@ -8680,6 +9019,36 @@ snapshots: '@eslint/object-schema@2.1.4': {} + '@fastify/ajv-compiler@4.0.1': + dependencies: + ajv: 8.17.1 + ajv-formats: 3.0.1(ajv@8.17.1) + fast-uri: 3.0.1 + + '@fastify/autoload@6.0.1': {} + + '@fastify/deepmerge@2.0.0': {} + + '@fastify/error@4.0.0': {} + + '@fastify/fast-json-stringify-compiler@5.0.1': + dependencies: + fast-json-stringify: 6.0.0 + + '@fastify/merge-json-schemas@0.1.1': + dependencies: + fast-deep-equal: 3.1.3 + + '@fastify/sensible@6.0.1': + dependencies: + '@lukeed/ms': 2.0.2 + dequal: 2.0.3 + fastify-plugin: 5.0.1 + forwarded: 0.2.0 + http-errors: 2.0.0 + type-is: 1.6.18 + vary: 1.1.2 + '@google-cloud/common@4.0.3': dependencies: '@google-cloud/projectify': 3.0.0 @@ -8863,6 +9232,8 @@ snapshots: wrap-ansi: 8.1.0 wrap-ansi-cjs: wrap-ansi@7.0.0 + '@istanbuljs/schema@0.1.3': {} + '@jest/schemas@29.6.3': dependencies: '@sinclair/typebox': 0.27.8 @@ -8898,12 +9269,13 @@ snapshots: dependencies: '@jridgewell/resolve-uri': 3.1.2 '@jridgewell/sourcemap-codec': 1.5.0 - optional: true '@js-sdsl/ordered-map@4.4.2': {} '@jspm/core@2.0.1': {} + '@lukeed/ms@2.0.2': {} + '@mdx-js/mdx@2.3.0': dependencies: '@types/estree-jsx': 1.0.5 @@ -10168,17 +10540,13 @@ snapshots: '@trpc/server@11.0.0-rc.502': {} - '@tsconfig/node10@1.0.11': - optional: true + '@tsconfig/node10@1.0.11': {} - '@tsconfig/node12@1.0.11': - optional: true + '@tsconfig/node12@1.0.11': {} - '@tsconfig/node14@1.0.3': - optional: true + '@tsconfig/node14@1.0.3': {} - '@tsconfig/node16@1.0.4': - optional: true + '@tsconfig/node16@1.0.4': {} '@types/acorn@4.0.6': dependencies: @@ -10210,7 +10578,7 @@ snapshots: '@types/body-parser@1.19.5': dependencies: '@types/connect': 3.4.38 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/caseless@0.12.5': {} @@ -10220,7 +10588,7 @@ snapshots: '@types/connect@3.4.38': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/cookie@0.4.1': {} @@ -10228,11 +10596,11 @@ snapshots: '@types/cors@2.8.17': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/cross-spawn@6.0.6': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/debug@4.1.12': dependencies: @@ -10256,7 +10624,7 @@ snapshots: '@types/express-serve-static-core@4.19.5': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/qs': 6.9.16 '@types/range-parser': 1.2.7 '@types/send': 0.17.4 @@ -10273,12 +10641,12 @@ snapshots: '@types/fs-extra@11.0.4': dependencies: '@types/jsonfile': 6.1.4 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/glob@7.2.0': dependencies: '@types/minimatch': 5.1.2 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/hast@2.3.10': dependencies: @@ -10290,9 +10658,11 @@ snapshots: '@types/http-errors@2.0.4': {} + '@types/istanbul-lib-coverage@2.0.6': {} + '@types/jsonfile@6.1.4': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/lodash@4.17.7': {} @@ -10314,7 +10684,7 @@ snapshots: '@types/morgan@1.9.9': dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/ms@0.7.34': {} @@ -10326,10 +10696,6 @@ snapshots: dependencies: undici-types: 5.26.5 - '@types/node@20.16.5': - dependencies: - undici-types: 6.19.8 - '@types/node@22.5.5': dependencies: undici-types: 6.19.8 @@ -10352,7 +10718,7 @@ snapshots: '@types/request@2.48.12': dependencies: '@types/caseless': 0.12.5 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/tough-cookie': 4.0.5 form-data: 2.5.1 @@ -10363,12 +10729,12 @@ snapshots: '@types/send@0.17.4': dependencies: '@types/mime': 1.3.5 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/serve-static@1.15.7': dependencies: '@types/http-errors': 2.0.4 - '@types/node': 20.16.5 + '@types/node': 22.5.5 '@types/send': 0.17.4 '@types/shimmer@1.2.0': {} @@ -10603,6 +10969,8 @@ snapshots: dependencies: event-target-shim: 5.0.1 + abstract-logging@2.0.1: {} + accepts@1.3.8: dependencies: mime-types: 2.1.35 @@ -10629,7 +10997,6 @@ snapshots: acorn-walk@8.3.4: dependencies: acorn: 8.12.1 - optional: true acorn@7.4.1: {} @@ -10652,6 +11019,10 @@ snapshots: clean-stack: 2.2.0 indent-string: 4.0.0 + ajv-formats@3.0.1(ajv@8.17.1): + optionalDependencies: + ajv: 8.17.1 + ajv@6.12.6: dependencies: fast-deep-equal: 3.1.3 @@ -10659,6 +11030,13 @@ snapshots: json-schema-traverse: 0.4.1 uri-js: 4.4.1 + ajv@8.17.1: + dependencies: + fast-deep-equal: 3.1.3 + fast-uri: 3.0.1 + json-schema-traverse: 1.0.0 + require-from-string: 2.0.2 + ansi-align@3.0.1: dependencies: string-width: 4.2.3 @@ -10692,8 +11070,7 @@ snapshots: normalize-path: 3.0.0 picomatch: 2.3.1 - arg@4.1.3: - optional: true + arg@4.1.3: {} arg@5.0.2: {} @@ -10745,6 +11122,8 @@ snapshots: asynckit@0.4.0: {} + atomic-sleep@1.0.0: {} + atomically@2.0.3: dependencies: stubborn-fs: 1.2.5 @@ -10764,6 +11143,11 @@ snapshots: dependencies: possible-typed-array-names: 1.0.0 + avvio@9.0.0: + dependencies: + '@fastify/error': 4.0.0 + fastq: 1.17.1 + babel-core@7.0.0-bridge.0(@babel/core@7.25.2): dependencies: '@babel/core': 7.25.2 @@ -10877,6 +11261,11 @@ snapshots: base64-js: 1.5.1 ieee754: 1.2.1 + buffer@6.0.3: + dependencies: + base64-js: 1.5.1 + ieee754: 1.2.1 + busboy@1.6.0: dependencies: streamsearch: 1.1.0 @@ -10885,6 +11274,20 @@ snapshots: bytes@3.1.2: {} + c8@10.1.2: + dependencies: + '@bcoe/v8-coverage': 0.2.3 + '@istanbuljs/schema': 0.1.3 + find-up: 5.0.0 + foreground-child: 3.3.0 + istanbul-lib-coverage: 3.2.2 + istanbul-lib-report: 3.0.1 + istanbul-reports: 3.1.7 + test-exclude: 7.0.1 + v8-to-istanbul: 9.3.0 + yargs: 17.7.2 + yargs-parser: 21.1.1 + cac@6.7.14: {} cacache@17.1.4: @@ -10988,6 +11391,10 @@ snapshots: optionalDependencies: fsevents: 2.3.3 + chokidar@4.0.1: + dependencies: + readdirp: 4.0.1 + chownr@1.1.4: {} chownr@2.0.0: {} @@ -11063,6 +11470,8 @@ snapshots: color-convert: 1.9.3 color-string: 1.9.1 + colorette@2.0.20: {} + colorspace@1.1.4: dependencies: color: 3.2.1 @@ -11082,6 +11491,8 @@ snapshots: commander@6.2.1: {} + commist@3.2.0: {} + commondir@1.0.1: {} compressible@2.0.18: @@ -11102,6 +11513,16 @@ snapshots: concat-map@0.0.1: {} + concurrently@9.0.1: + dependencies: + chalk: 4.1.2 + lodash: 4.17.21 + rxjs: 7.8.1 + shell-quote: 1.8.1 + supports-color: 8.1.1 + tree-kill: 1.2.2 + yargs: 17.7.2 + confbox@0.1.7: {} config-chain@1.1.13: @@ -11149,8 +11570,7 @@ snapshots: object-assign: 4.1.1 vary: 1.1.2 - create-require@1.1.1: - optional: true + create-require@1.1.1: {} cross-spawn@6.0.5: dependencies: @@ -11200,6 +11620,8 @@ snapshots: date-fns@3.6.0: {} + dateformat@4.6.3: {} + dayjs@1.11.13: {} debounce@2.0.0: {} @@ -11285,8 +11707,7 @@ snapshots: diff-sequences@29.6.3: {} - diff@4.0.2: - optional: true + diff@4.0.2: {} diff@5.2.0: {} @@ -11394,7 +11815,7 @@ snapshots: dependencies: '@types/cookie': 0.4.1 '@types/cors': 2.8.17 - '@types/node': 20.16.5 + '@types/node': 22.5.5 accepts: 1.3.8 base64id: 2.0.0 cookie: 0.4.2 @@ -11759,7 +12180,7 @@ snapshots: eval@0.1.8: dependencies: - '@types/node': 20.16.5 + '@types/node': 22.5.5 require-like: 0.1.2 event-target-shim@5.0.1: {} @@ -11768,6 +12189,8 @@ snapshots: dependencies: uuid: 8.3.2 + events@3.3.0: {} + execa@5.1.1: dependencies: cross-spawn: 7.0.3 @@ -11857,6 +12280,10 @@ snapshots: iconv-lite: 0.4.24 tmp: 0.0.33 + fast-copy@3.0.2: {} + + fast-decode-uri-component@1.0.1: {} + fast-deep-equal@2.0.1: {} fast-deep-equal@3.1.3: {} @@ -11871,10 +12298,73 @@ snapshots: fast-json-stable-stringify@2.1.0: {} + fast-json-stringify@6.0.0: + dependencies: + '@fastify/merge-json-schemas': 0.1.1 + ajv: 8.17.1 + ajv-formats: 3.0.1(ajv@8.17.1) + fast-deep-equal: 3.1.3 + fast-uri: 2.4.0 + json-schema-ref-resolver: 1.0.1 + rfdc: 1.4.1 + fast-levenshtein@2.0.6: {} + fast-querystring@1.1.2: + dependencies: + fast-decode-uri-component: 1.0.1 + + fast-redact@3.5.0: {} + + fast-safe-stringify@2.1.1: {} + fast-text-encoding@1.0.6: {} + fast-uri@2.4.0: {} + + fast-uri@3.0.1: {} + + fastify-cli@7.0.1: + dependencies: + '@fastify/deepmerge': 2.0.0 + chalk: 4.1.2 + chokidar: 4.0.1 + close-with-grace: 2.1.0 + commist: 3.2.0 + dotenv: 16.4.5 + fastify: 5.0.0 + fastify-plugin: 5.0.1 + generify: 4.2.0 + help-me: 4.2.0 + is-docker: 2.2.1 + pino-pretty: 11.2.2 + pkg-up: 3.1.0 + resolve-from: 5.0.0 + semver: 7.6.3 + yargs-parser: 21.1.1 + + fastify-plugin@5.0.1: {} + + fastify-tsconfig@2.0.0: {} + + fastify@5.0.0: + dependencies: + '@fastify/ajv-compiler': 4.0.1 + '@fastify/error': 4.0.0 + '@fastify/fast-json-stringify-compiler': 5.0.1 + abstract-logging: 2.0.1 + avvio: 9.0.0 + fast-json-stringify: 6.0.0 + find-my-way: 9.0.1 + light-my-request: 6.0.0 + pino: 9.4.0 + process-warning: 4.0.0 + proxy-addr: 2.0.7 + rfdc: 1.4.1 + secure-json-parse: 2.7.0 + semver: 7.6.3 + toad-cache: 3.7.0 + fastq@1.17.1: dependencies: reusify: 1.0.4 @@ -11932,6 +12422,12 @@ snapshots: make-dir: 3.1.0 pkg-dir: 4.2.0 + find-my-way@9.0.1: + dependencies: + fast-deep-equal: 3.1.3 + fast-querystring: 1.1.2 + safe-regex2: 4.0.0 + find-up@3.0.0: dependencies: locate-path: 3.0.0 @@ -12068,6 +12564,13 @@ snapshots: dependencies: loader-utils: 3.3.1 + generify@4.2.0: + dependencies: + isbinaryfile: 4.0.10 + pump: 3.0.2 + split2: 3.2.2 + walker: 1.0.8 + gensync@1.0.0-beta.2: {} get-caller-file@2.0.5: {} @@ -12149,6 +12652,15 @@ snapshots: minipass: 7.1.2 path-scurry: 1.11.1 + glob@10.4.5: + dependencies: + foreground-child: 3.3.0 + jackspeak: 3.4.3 + minimatch: 9.0.5 + minipass: 7.1.2 + package-json-from-dist: 1.0.0 + path-scurry: 1.11.1 + glob@11.0.0: dependencies: foreground-child: 3.3.0 @@ -12167,6 +12679,14 @@ snapshots: once: 1.4.0 path-is-absolute: 1.0.1 + glob@8.1.0: + dependencies: + fs.realpath: 1.0.0 + inflight: 1.0.6 + inherits: 2.0.4 + minimatch: 5.1.6 + once: 1.4.0 + global-directory@4.0.1: dependencies: ini: 4.1.1 @@ -12363,6 +12883,13 @@ snapshots: helmet@7.1.0: {} + help-me@4.2.0: + dependencies: + glob: 8.1.0 + readable-stream: 3.6.2 + + help-me@5.0.0: {} + hosted-git-info@2.8.9: {} hosted-git-info@6.1.1: @@ -12371,6 +12898,8 @@ snapshots: html-entities@2.5.2: {} + html-escaper@2.0.2: {} + html-tags@3.3.1: {} html-to-text@9.0.5: @@ -12564,6 +13093,8 @@ snapshots: is-deflate@1.0.0: {} + is-docker@2.2.1: {} + is-extglob@2.1.1: {} is-fullwidth-code-point@3.0.0: {} @@ -12675,18 +13206,39 @@ snapshots: isarray@2.0.5: {} + isbinaryfile@4.0.10: {} + isbot@5.1.17: {} isexe@2.0.0: {} isobject@3.0.1: {} + istanbul-lib-coverage@3.2.2: {} + + istanbul-lib-report@3.0.1: + dependencies: + istanbul-lib-coverage: 3.2.2 + make-dir: 4.0.0 + supports-color: 7.2.0 + + istanbul-reports@3.1.7: + dependencies: + html-escaper: 2.0.2 + istanbul-lib-report: 3.0.1 + jackspeak@2.3.6: dependencies: '@isaacs/cliui': 8.0.2 optionalDependencies: '@pkgjs/parseargs': 0.11.0 + jackspeak@3.4.3: + dependencies: + '@isaacs/cliui': 8.0.2 + optionalDependencies: + '@pkgjs/parseargs': 0.11.0 + jackspeak@4.0.1: dependencies: '@isaacs/cliui': 8.0.2 @@ -12697,6 +13249,8 @@ snapshots: jiti@1.21.6: {} + joycon@3.1.1: {} + js-beautify@1.15.1: dependencies: config-chain: 1.1.13 @@ -12761,8 +13315,14 @@ snapshots: json-parse-even-better-errors@3.0.2: {} + json-schema-ref-resolver@1.0.1: + dependencies: + fast-deep-equal: 3.1.3 + json-schema-traverse@0.4.1: {} + json-schema-traverse@1.0.0: {} + json-stable-stringify-without-jsonify@1.0.1: {} json5@2.2.3: {} @@ -12854,6 +13414,12 @@ snapshots: prelude-ls: 1.2.1 type-check: 0.4.0 + light-my-request@6.0.0: + dependencies: + cookie: 0.6.0 + process-warning: 4.0.0 + set-cookie-parser: 2.7.0 + lilconfig@2.1.0: {} lilconfig@3.1.2: {} @@ -12969,8 +13535,15 @@ snapshots: dependencies: semver: 6.3.1 - make-error@1.3.6: - optional: true + make-dir@4.0.0: + dependencies: + semver: 7.6.3 + + make-error@1.3.6: {} + + makeerror@1.0.12: + dependencies: + tmpl: 1.0.5 map-or-similar@1.5.0: {} @@ -13360,6 +13933,10 @@ snapshots: dependencies: brace-expansion: 1.1.11 + minimatch@5.1.6: + dependencies: + brace-expansion: 2.0.1 + minimatch@9.0.1: dependencies: brace-expansion: 2.0.1 @@ -13612,6 +14189,8 @@ snapshots: ohash@1.1.4: {} + on-exit-leak-free@2.1.2: {} + on-finished@2.3.0: dependencies: ee-first: 1.1.1 @@ -13809,6 +14388,44 @@ snapshots: pify@4.0.1: {} + pino-abstract-transport@1.2.0: + dependencies: + readable-stream: 4.5.2 + split2: 4.2.0 + + pino-pretty@11.2.2: + dependencies: + colorette: 2.0.20 + dateformat: 4.6.3 + fast-copy: 3.0.2 + fast-safe-stringify: 2.1.1 + help-me: 5.0.0 + joycon: 3.1.1 + minimist: 1.2.8 + on-exit-leak-free: 2.1.2 + pino-abstract-transport: 1.2.0 + pump: 3.0.2 + readable-stream: 4.5.2 + secure-json-parse: 2.7.0 + sonic-boom: 4.1.0 + strip-json-comments: 3.1.1 + + pino-std-serializers@7.0.0: {} + + pino@9.4.0: + dependencies: + atomic-sleep: 1.0.0 + fast-redact: 3.5.0 + on-exit-leak-free: 2.1.2 + pino-abstract-transport: 1.2.0 + pino-std-serializers: 7.0.0 + process-warning: 4.0.0 + quick-format-unescaped: 4.0.4 + real-require: 0.2.0 + safe-stable-stringify: 2.5.0 + sonic-boom: 4.1.0 + thread-stream: 3.1.0 + pirates@4.0.6: {} pkg-dir@3.0.0: @@ -13825,6 +14442,10 @@ snapshots: mlly: 1.7.1 pathe: 1.1.2 + pkg-up@3.1.0: + dependencies: + find-up: 3.0.0 + playwright-core@1.47.0: {} playwright@1.47.0: @@ -13962,6 +14583,8 @@ snapshots: process-nextick-args@2.0.1: {} + process-warning@4.0.0: {} + process@0.11.10: {} promise-inflight@1.0.1: {} @@ -14002,7 +14625,7 @@ snapshots: '@protobufjs/path': 1.1.2 '@protobufjs/pool': 1.1.0 '@protobufjs/utf8': 1.1.0 - '@types/node': 20.16.5 + '@types/node': 22.5.5 long: 5.2.3 proxy-addr@2.0.7: @@ -14054,6 +14677,8 @@ snapshots: queue-microtask@1.2.3: {} + quick-format-unescaped@4.0.4: {} + range-parser@1.2.1: {} raw-body@2.5.2: @@ -14207,10 +14832,22 @@ snapshots: string_decoder: 1.3.0 util-deprecate: 1.0.2 + readable-stream@4.5.2: + dependencies: + abort-controller: 3.0.0 + buffer: 6.0.3 + events: 3.3.0 + process: 0.11.10 + string_decoder: 1.3.0 + readdirp@3.6.0: dependencies: picomatch: 2.3.1 + readdirp@4.0.1: {} + + real-require@0.2.0: {} + recast@0.23.9: dependencies: ast-types: 0.16.1 @@ -14319,6 +14956,8 @@ snapshots: require-directory@2.1.1: {} + require-from-string@2.0.2: {} + require-in-the-middle@7.4.0: dependencies: debug: 4.3.7 @@ -14333,6 +14972,8 @@ snapshots: resolve-from@4.0.0: {} + resolve-from@5.0.0: {} + resolve-pkg-maps@1.0.0: {} resolve.exports@2.0.2: {} @@ -14348,6 +14989,8 @@ snapshots: onetime: 5.1.2 signal-exit: 3.0.7 + ret@0.5.0: {} + retry-request@5.0.2: dependencies: debug: 4.3.7 @@ -14368,6 +15011,8 @@ snapshots: reusify@1.0.4: {} + rfdc@1.4.1: {} + rimraf@2.6.3: dependencies: glob: 7.2.3 @@ -14427,6 +15072,10 @@ snapshots: es-errors: 1.3.0 is-regex: 1.1.4 + safe-regex2@4.0.0: + dependencies: + ret: 0.5.0 + safe-stable-stringify@2.5.0: {} safer-buffer@2.1.2: {} @@ -14469,6 +15118,8 @@ snapshots: dependencies: loose-envify: 1.4.0 + secure-json-parse@2.7.0: {} + selderee@0.11.0: dependencies: parseley: 0.12.1 @@ -14630,6 +15281,10 @@ snapshots: - supports-color - utf-8-validate + sonic-boom@4.1.0: + dependencies: + atomic-sleep: 1.0.0 + source-map-js@1.2.1: {} source-map-support@0.5.21: @@ -14657,6 +15312,12 @@ snapshots: spdx-license-ids@3.0.20: {} + split2@3.2.2: + dependencies: + readable-stream: 3.6.2 + + split2@4.2.0: {} + sprintf-js@1.0.3: {} ssri@10.0.6: @@ -14847,6 +15508,10 @@ snapshots: dependencies: has-flag: 4.0.0 + supports-color@8.1.1: + dependencies: + has-flag: 4.0.0 + supports-preserve-symlinks-flag@1.0.0: {} tailwind-merge@2.5.2: {} @@ -14947,6 +15612,12 @@ snapshots: terminal-size@4.0.0: {} + test-exclude@7.0.1: + dependencies: + '@istanbuljs/schema': 0.1.3 + glob: 10.4.5 + minimatch: 9.0.5 + text-hex@1.0.0: {} text-table@0.2.0: {} @@ -14959,6 +15630,10 @@ snapshots: dependencies: any-promise: 1.3.0 + thread-stream@3.1.0: + dependencies: + real-require: 0.2.0 + through2@2.0.5: dependencies: readable-stream: 2.3.8 @@ -14980,12 +15655,16 @@ snapshots: dependencies: os-tmpdir: 1.0.2 + tmpl@1.0.5: {} + to-fast-properties@2.0.0: {} to-regex-range@5.0.1: dependencies: is-number: 7.0.0 + toad-cache@3.7.0: {} + toidentifier@1.0.1: {} toml@3.0.0: {} @@ -15001,6 +15680,8 @@ snapshots: tr46@0.0.3: {} + tree-kill@1.2.2: {} + trim-lines@3.0.1: {} triple-beam@1.4.1: {} @@ -15034,6 +15715,24 @@ snapshots: yn: 3.1.1 optional: true + ts-node@10.9.2(@types/node@22.5.5)(typescript@5.6.2): + dependencies: + '@cspotcode/source-map-support': 0.8.1 + '@tsconfig/node10': 1.0.11 + '@tsconfig/node12': 1.0.11 + '@tsconfig/node14': 1.0.3 + '@tsconfig/node16': 1.0.4 + '@types/node': 22.5.5 + acorn: 8.12.1 + acorn-walk: 8.3.4 + arg: 4.1.3 + create-require: 1.1.1 + diff: 4.0.2 + make-error: 1.3.6 + typescript: 5.6.2 + v8-compile-cache-lib: 3.0.1 + yn: 3.1.1 + tsconfck@3.1.3(typescript@5.5.4): optionalDependencies: typescript: 5.5.4 @@ -15308,8 +16007,13 @@ snapshots: kleur: 4.1.5 sade: 1.8.1 - v8-compile-cache-lib@3.0.1: - optional: true + v8-compile-cache-lib@3.0.1: {} + + v8-to-istanbul@9.3.0: + dependencies: + '@jridgewell/trace-mapping': 0.3.25 + '@types/istanbul-lib-coverage': 2.0.6 + convert-source-map: 2.0.0 valibot@0.41.0(typescript@5.5.4): optionalDependencies: @@ -15436,6 +16140,10 @@ snapshots: walkdir@0.4.1: {} + walker@1.0.8: + dependencies: + makeerror: 1.0.12 + wcwidth@1.0.1: dependencies: defaults: 1.0.4 @@ -15587,8 +16295,7 @@ snapshots: y18n: 5.0.8 yargs-parser: 21.1.1 - yn@3.1.1: - optional: true + yn@3.1.1: {} yocto-queue@0.1.0: {} From 51f547a0af0cd660074ee9ad439c3f5cbce2a46e Mon Sep 17 00:00:00 2001 From: isoppp Date: Tue, 24 Sep 2024 01:17:30 +0900 Subject: [PATCH 2/4] add middlewares and logger, OTEL --- apps/api-fastify/package.json | 21 +++- apps/api-fastify/src/app.ts | 8 +- apps/api-fastify/src/lib/logger.ts | 18 +++ apps/api-fastify/src/lib/open-telemetry.ts | 26 +++++ apps/api-fastify/src/plugins/compress.ts | 10 ++ apps/api-fastify/src/plugins/http-redirect.ts | 15 +++ apps/api-fastify/src/plugins/rate-limit.ts | 8 ++ pnpm-lock.yaml | 105 ++++++++++++++++++ 8 files changed, 201 insertions(+), 10 deletions(-) create mode 100644 apps/api-fastify/src/lib/logger.ts create mode 100644 apps/api-fastify/src/lib/open-telemetry.ts create mode 100644 apps/api-fastify/src/plugins/compress.ts create mode 100644 apps/api-fastify/src/plugins/http-redirect.ts create mode 100644 apps/api-fastify/src/plugins/rate-limit.ts diff --git a/apps/api-fastify/package.json b/apps/api-fastify/package.json index 3ed2cdf..6581f90 100644 --- a/apps/api-fastify/package.json +++ b/apps/api-fastify/package.json @@ -18,18 +18,31 @@ "author": "", "license": "ISC", "dependencies": { - "fastify": "^5.0.0", - "fastify-plugin": "^5.0.0", "@fastify/autoload": "^6.0.0", + "@fastify/compress": "8.0.1", + "@fastify/helmet": "12.0.1", + "@fastify/rate-limit": "10.0.1", "@fastify/sensible": "^6.0.0", - "fastify-cli": "^7.0.1" + "@google-cloud/logging-winston": "6.0.0", + "@google-cloud/opentelemetry-cloud-trace-exporter": "2.3.0", + "@opentelemetry/instrumentation-fastify": "0.39.0", + "@opentelemetry/instrumentation-http": "0.53.0", + "@opentelemetry/instrumentation-pino": "0.42.0", + "@opentelemetry/instrumentation-winston": "0.40.0", + "@opentelemetry/sdk-node": "0.53.0", + "@prisma/instrumentation": "5.19.1", + "fastify": "^5.0.0", + "fastify-cli": "^7.0.1", + "fastify-plugin": "^5.0.0", + "helmet": "7.1.0", + "winston": "3.14.2" }, "devDependencies": { "@types/node": "^22.1.0", "c8": "^10.1.2", - "ts-node": "^10.4.0", "concurrently": "^9.0.0", "fastify-tsconfig": "^2.0.0", + "ts-node": "^10.4.0", "typescript": "^5.2.2" } } \ No newline at end of file diff --git a/apps/api-fastify/src/app.ts b/apps/api-fastify/src/app.ts index be51f92..a21aa58 100644 --- a/apps/api-fastify/src/app.ts +++ b/apps/api-fastify/src/app.ts @@ -1,6 +1,7 @@ import { join } from 'path'; import AutoLoad, {AutoloadPluginOptions} from '@fastify/autoload'; import { FastifyPluginAsync, FastifyServerOptions } from 'fastify'; +import { initOpenTelemetry } from './lib/open-telemetry' export interface AppOptions extends FastifyServerOptions, Partial { @@ -14,19 +15,14 @@ const app: FastifyPluginAsync = async ( opts ): Promise => { // Place here your custom code! + initOpenTelemetry() // Do not touch the following lines - - // This loads all plugins defined in plugins - // those should be support plugins that are reused - // through your application void fastify.register(AutoLoad, { dir: join(__dirname, 'plugins'), options: opts }) - // This loads all plugins defined in routes - // define your routes in one of these void fastify.register(AutoLoad, { dir: join(__dirname, 'routes'), options: opts diff --git a/apps/api-fastify/src/lib/logger.ts b/apps/api-fastify/src/lib/logger.ts new file mode 100644 index 0000000..4b65171 --- /dev/null +++ b/apps/api-fastify/src/lib/logger.ts @@ -0,0 +1,18 @@ +import winston from 'winston' + +// Imports the Google Cloud client library for Winston +import { LoggingWinston } from '@google-cloud/logging-winston' + +export const logger = winston.createLogger({ + level: 'info', + transports: [ + new winston.transports.Console({ + format: winston.format.combine(winston.format.colorize(), winston.format.cli()), + }), + process.env.GOOGLE_APPLICATION_CREDENTIALS + ? new LoggingWinston({ + keyFile: process.env.GOOGLE_APPLICATION_CREDENTIALS, + }) + : null, + ].filter((v) => !!v), +}) diff --git a/apps/api-fastify/src/lib/open-telemetry.ts b/apps/api-fastify/src/lib/open-telemetry.ts new file mode 100644 index 0000000..e92a696 --- /dev/null +++ b/apps/api-fastify/src/lib/open-telemetry.ts @@ -0,0 +1,26 @@ +import { TraceExporter } from '@google-cloud/opentelemetry-cloud-trace-exporter' +import { FastifyInstrumentation } from '@opentelemetry/instrumentation-fastify' +import { PinoInstrumentation } from '@opentelemetry/instrumentation-pino' +import { HttpInstrumentation } from '@opentelemetry/instrumentation-http' +import { WinstonInstrumentation } from '@opentelemetry/instrumentation-winston' +import { NodeSDK } from '@opentelemetry/sdk-node' +import p from '@prisma/instrumentation' + +export const initOpenTelemetry = () => { + if (process.env.GOOGLE_APPLICATION_CREDENTIALS) { + const sdk = new NodeSDK({ + traceExporter: new TraceExporter({ + keyFile: process.env.GOOGLE_APPLICATION_CREDENTIALS, + }), + instrumentations: [ + new FastifyInstrumentation(), + new HttpInstrumentation(), + new PinoInstrumentation(), + new WinstonInstrumentation(), + new p.PrismaInstrumentation(), + ], + }) + + sdk.start() + } +} diff --git a/apps/api-fastify/src/plugins/compress.ts b/apps/api-fastify/src/plugins/compress.ts new file mode 100644 index 0000000..0c2ccfb --- /dev/null +++ b/apps/api-fastify/src/plugins/compress.ts @@ -0,0 +1,10 @@ +// plugins/compress.ts +import fp from 'fastify-plugin' +import fastifyCompress from '@fastify/compress' +import { FastifyPluginAsync } from 'fastify' + +const compressPlugin: FastifyPluginAsync = async (fastify, opts) => { + fastify.register(fastifyCompress, opts) +} + +export default fp(compressPlugin) diff --git a/apps/api-fastify/src/plugins/http-redirect.ts b/apps/api-fastify/src/plugins/http-redirect.ts new file mode 100644 index 0000000..898786f --- /dev/null +++ b/apps/api-fastify/src/plugins/http-redirect.ts @@ -0,0 +1,15 @@ +// plugins/httpsRedirect.ts +import fp from 'fastify-plugin' +import { FastifyPluginAsync } from 'fastify' + +const httpsRedirectPlugin: FastifyPluginAsync = async (fastify) => { + fastify.addHook('onRequest', (request, reply, done) => { + if (request.headers['x-forwarded-proto'] === 'http') { + reply.redirect(`https://${request.hostname}${request.url}`) + } else { + done() + } + }) +} + +export default fp(httpsRedirectPlugin) diff --git a/apps/api-fastify/src/plugins/rate-limit.ts b/apps/api-fastify/src/plugins/rate-limit.ts new file mode 100644 index 0000000..5bf69af --- /dev/null +++ b/apps/api-fastify/src/plugins/rate-limit.ts @@ -0,0 +1,8 @@ +import fp from 'fastify-plugin' +import { FastifyPluginAsync } from 'fastify' + +const rateLimitPlugin: FastifyPluginAsync = async (fastify, opts) => { +// TODO +} + +export default fp(rateLimitPlugin) diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 3a1a487..f2615cd 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -23,9 +23,42 @@ importers: '@fastify/autoload': specifier: ^6.0.0 version: 6.0.1 + '@fastify/compress': + specifier: 8.0.1 + version: 8.0.1 + '@fastify/helmet': + specifier: 12.0.1 + version: 12.0.1 + '@fastify/rate-limit': + specifier: 10.0.1 + version: 10.0.1 '@fastify/sensible': specifier: ^6.0.0 version: 6.0.1 + '@google-cloud/logging-winston': + specifier: 6.0.0 + version: 6.0.0(winston@3.14.2) + '@google-cloud/opentelemetry-cloud-trace-exporter': + specifier: 2.3.0 + version: 2.3.0(@opentelemetry/api@1.9.0)(@opentelemetry/core@1.26.0(@opentelemetry/api@1.9.0))(@opentelemetry/resources@1.26.0(@opentelemetry/api@1.9.0))(@opentelemetry/sdk-trace-base@1.26.0(@opentelemetry/api@1.9.0)) + '@opentelemetry/instrumentation-fastify': + specifier: 0.39.0 + version: 0.39.0(@opentelemetry/api@1.9.0) + '@opentelemetry/instrumentation-http': + specifier: 0.53.0 + version: 0.53.0(@opentelemetry/api@1.9.0) + '@opentelemetry/instrumentation-pino': + specifier: 0.42.0 + version: 0.42.0(@opentelemetry/api@1.9.0) + '@opentelemetry/instrumentation-winston': + specifier: 0.40.0 + version: 0.40.0(@opentelemetry/api@1.9.0) + '@opentelemetry/sdk-node': + specifier: 0.53.0 + version: 0.53.0(@opentelemetry/api@1.9.0) + '@prisma/instrumentation': + specifier: 5.19.1 + version: 5.19.1 fastify: specifier: ^5.0.0 version: 5.0.0 @@ -35,6 +68,12 @@ importers: fastify-plugin: specifier: ^5.0.0 version: 5.0.1 + helmet: + specifier: 7.1.0 + version: 7.1.0 + winston: + specifier: 3.14.2 + version: 3.14.2 devDependencies: '@types/node': specifier: ^22.1.0 @@ -1643,12 +1682,18 @@ packages: resolution: {integrity: sha512-BsWiH1yFGjXXS2yvrf5LyuoSIIbPrGUWob917o+BTKuZ7qJdxX8aJLRxs1fS9n6r7vESrq1OUqb68dANcFXuQQ==} engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0} + '@fastify/accept-negotiator@2.0.0': + resolution: {integrity: sha512-/Sce/kBzuTxIq5tJh85nVNOq9wKD8s+viIgX0fFMDBdw95gnpf53qmF1oBgJym3cPFliWUuSloVg/1w/rH0FcQ==} + '@fastify/ajv-compiler@4.0.1': resolution: {integrity: sha512-DxrBdgsjNLP0YM6W5Hd6/Fmj43S8zMKiFJYgi+Ri3htTGAowPVG/tG1wpnWLMjufEnehRivUCKZ1pLDIoZdTuw==} '@fastify/autoload@6.0.1': resolution: {integrity: sha512-NVVu/g9yUFkv4nNxsDHhNKz37jINXlPxwNYbA4Hu8pnho6sDVeKo24pt+dnax1SeUapxxs/yulXukbUai6Aebw==} + '@fastify/compress@8.0.1': + resolution: {integrity: sha512-yWNfKhvL4orfN45LKCHCo8Fcsbj1kdNgwyShw2xpdHfzPf4A3MESmgSfUm3TCKQwgqDdrPnLfy1E+3I/DVP+BQ==} + '@fastify/deepmerge@2.0.0': resolution: {integrity: sha512-fsaybTGDyQ5KpPsplQqb9yKdCf2x/pbNpMNk8Tvp3rRz7lVcupKysH4b2ELMN2P4Hak1+UqTYdTj/u4FNV2p0g==} @@ -1658,9 +1703,15 @@ packages: '@fastify/fast-json-stringify-compiler@5.0.1': resolution: {integrity: sha512-f2d3JExJgFE3UbdFcpPwqNUEoHWmt8pAKf8f+9YuLESdefA0WgqxeT6DrGL4Yrf/9ihXNSKOqpjEmurV405meA==} + '@fastify/helmet@12.0.1': + resolution: {integrity: sha512-kkjBcedWwdflRThovGuvN9jB2QQLytBqArCFPdMIb7o2Fp0l/H3xxYi/6x/SSRuH/FFt9qpTGIfJz2bfnMrLqA==} + '@fastify/merge-json-schemas@0.1.1': resolution: {integrity: sha512-fERDVz7topgNjtXsJTTW1JKLy0rhuLRcquYqNR9rF7OcVpCa2OVW49ZPDIhaRRCaUuvVxI+N416xUoF76HNSXA==} + '@fastify/rate-limit@10.0.1': + resolution: {integrity: sha512-wLGtemfKepoe9yUyHB0zcTvCWy7H3qAScDCxny8NE+mqGNg9L1HhkWK8bIlMDCLOLfng1i7+GTikYeZvM2HFWQ==} + '@fastify/sensible@6.0.1': resolution: {integrity: sha512-D0rN0kMeZKP23f4w9MoCI9e4+n7vASFAsGoRNn9bondSbplLeIfR2HcjCbyElAM04jGrPRLi/edyThEPOyC9cQ==} @@ -2010,12 +2061,24 @@ packages: peerDependencies: '@opentelemetry/api': ^1.3.0 + '@opentelemetry/instrumentation-fastify@0.39.0': + resolution: {integrity: sha512-SS9uSlKcsWZabhBp2szErkeuuBDgxOUlllwkS92dVaWRnMmwysPhcEgHKB8rUe3BHg/GnZC1eo1hbTZv4YhfoA==} + engines: {node: '>=14'} + peerDependencies: + '@opentelemetry/api': ^1.3.0 + '@opentelemetry/instrumentation-http@0.53.0': resolution: {integrity: sha512-H74ErMeDuZfj7KgYCTOFGWF5W9AfaPnqLQQxeFq85+D29wwV2yqHbz2IKLYpkOh7EI6QwDEl7rZCIxjJLyc/CQ==} engines: {node: '>=14'} peerDependencies: '@opentelemetry/api': ^1.3.0 + '@opentelemetry/instrumentation-pino@0.42.0': + resolution: {integrity: sha512-SoX6FzucBfTuFNMZjdurJhcYWq2ve8/LkhmyVLUW31HpIB45RF1JNum0u4MkGisosDmXlK4njomcgUovShI+WA==} + engines: {node: '>=14'} + peerDependencies: + '@opentelemetry/api': ^1.3.0 + '@opentelemetry/instrumentation-winston@0.40.0': resolution: {integrity: sha512-eMk2tKl86YJ8/yHvtDbyhrE35/R0InhO9zuHTflPx8T0+IvKVUhPV71MsJr32sImftqeOww92QHt4Jd+a5db4g==} engines: {node: '>=14'} @@ -9019,6 +9082,8 @@ snapshots: '@eslint/object-schema@2.1.4': {} + '@fastify/accept-negotiator@2.0.0': {} + '@fastify/ajv-compiler@4.0.1': dependencies: ajv: 8.17.1 @@ -9027,6 +9092,17 @@ snapshots: '@fastify/autoload@6.0.1': {} + '@fastify/compress@8.0.1': + dependencies: + '@fastify/accept-negotiator': 2.0.0 + fastify-plugin: 5.0.1 + mime-db: 1.53.0 + minipass: 7.1.2 + peek-stream: 1.1.3 + pump: 3.0.2 + pumpify: 2.0.1 + readable-stream: 4.5.2 + '@fastify/deepmerge@2.0.0': {} '@fastify/error@4.0.0': {} @@ -9035,10 +9111,21 @@ snapshots: dependencies: fast-json-stringify: 6.0.0 + '@fastify/helmet@12.0.1': + dependencies: + fastify-plugin: 5.0.1 + helmet: 7.1.0 + '@fastify/merge-json-schemas@0.1.1': dependencies: fast-deep-equal: 3.1.3 + '@fastify/rate-limit@10.0.1': + dependencies: + '@lukeed/ms': 2.0.2 + fastify-plugin: 5.0.1 + toad-cache: 3.7.0 + '@fastify/sensible@6.0.1': dependencies: '@lukeed/ms': 2.0.2 @@ -9491,6 +9578,15 @@ snapshots: transitivePeerDependencies: - supports-color + '@opentelemetry/instrumentation-fastify@0.39.0(@opentelemetry/api@1.9.0)': + dependencies: + '@opentelemetry/api': 1.9.0 + '@opentelemetry/core': 1.26.0(@opentelemetry/api@1.9.0) + '@opentelemetry/instrumentation': 0.53.0(@opentelemetry/api@1.9.0) + '@opentelemetry/semantic-conventions': 1.27.0 + transitivePeerDependencies: + - supports-color + '@opentelemetry/instrumentation-http@0.53.0(@opentelemetry/api@1.9.0)': dependencies: '@opentelemetry/api': 1.9.0 @@ -9501,6 +9597,15 @@ snapshots: transitivePeerDependencies: - supports-color + '@opentelemetry/instrumentation-pino@0.42.0(@opentelemetry/api@1.9.0)': + dependencies: + '@opentelemetry/api': 1.9.0 + '@opentelemetry/api-logs': 0.53.0 + '@opentelemetry/core': 1.26.0(@opentelemetry/api@1.9.0) + '@opentelemetry/instrumentation': 0.53.0(@opentelemetry/api@1.9.0) + transitivePeerDependencies: + - supports-color + '@opentelemetry/instrumentation-winston@0.40.0(@opentelemetry/api@1.9.0)': dependencies: '@opentelemetry/api': 1.9.0 From 034c8c65df4280c16996baabb585ae3949761860 Mon Sep 17 00:00:00 2001 From: isoppp Date: Tue, 24 Sep 2024 01:26:33 +0900 Subject: [PATCH 3/4] lint-fix with biome --- apps/api-fastify/biome.jsonc | 43 +++++++++++++++++++ apps/api-fastify/package.json | 5 ++- apps/api-fastify/src/app.ts | 26 +++++------ apps/api-fastify/src/lib/open-telemetry.ts | 2 +- apps/api-fastify/src/plugins/compress.ts | 4 +- apps/api-fastify/src/plugins/http-redirect.ts | 2 +- apps/api-fastify/src/plugins/rate-limit.ts | 6 +-- apps/api-fastify/src/plugins/sensible.ts | 2 +- apps/api-fastify/src/plugins/support.ts | 20 --------- apps/api-fastify/src/routes/example/index.ts | 10 ++--- apps/api-fastify/src/routes/root.ts | 10 ++--- apps/api-fastify/test/helper.ts | 15 +++---- apps/api-fastify/test/plugins/support.test.ts | 4 +- apps/api-fastify/test/routes/example.test.ts | 4 +- apps/api-fastify/test/routes/root.test.ts | 4 +- package.json | 1 + pnpm-lock.yaml | 3 ++ 17 files changed, 88 insertions(+), 73 deletions(-) create mode 100644 apps/api-fastify/biome.jsonc delete mode 100644 apps/api-fastify/src/plugins/support.ts diff --git a/apps/api-fastify/biome.jsonc b/apps/api-fastify/biome.jsonc new file mode 100644 index 0000000..310f1fc --- /dev/null +++ b/apps/api-fastify/biome.jsonc @@ -0,0 +1,43 @@ +{ + "$schema": "https://biomejs.dev/schemas/1.8.3/schema.json", + "files": { + "ignore": ["dist"] + }, + "json": { + "parser": { + "allowComments": true + } + }, + "formatter": { + "enabled": true, + "formatWithErrors": false, + "indentStyle": "space", + "indentWidth": 2, + "lineEnding": "lf", + "lineWidth": 120, + "attributePosition": "auto" + }, + "javascript": { + "formatter": { + "semicolons": "asNeeded", + "quoteStyle": "single", + "jsxQuoteStyle": "single" + } + }, + "organizeImports": { + "enabled": true + }, + "linter": { + "enabled": true, + "rules": { + "recommended": true, + "correctness": { + "noUnusedImports": "error", + "noUnusedVariables": { + "level": "error", + "fix": "none" + } + } + } + } +} diff --git a/apps/api-fastify/package.json b/apps/api-fastify/package.json index 6581f90..9ad1a4c 100644 --- a/apps/api-fastify/package.json +++ b/apps/api-fastify/package.json @@ -12,7 +12,8 @@ "build:ts": "tsc", "watch:ts": "tsc -w", "dev": "npm run build:ts && concurrently -k -p \"[{name}]\" -n \"TypeScript,App\" -c \"yellow.bold,cyan.bold\" \"npm:watch:ts\" \"npm:dev:start\"", - "dev:start": "fastify start --ignore-watch=.ts$ -w -l info -P dist/app.js" + "dev:start": "fastify start --ignore-watch=.ts$ -w -l info -P dist/app.js", + "lint-fix": "pnpm biome check --fix --unsafe" }, "keywords": [], "author": "", @@ -45,4 +46,4 @@ "ts-node": "^10.4.0", "typescript": "^5.2.2" } -} \ No newline at end of file +} diff --git a/apps/api-fastify/src/app.ts b/apps/api-fastify/src/app.ts index a21aa58..e4154ba 100644 --- a/apps/api-fastify/src/app.ts +++ b/apps/api-fastify/src/app.ts @@ -1,33 +1,27 @@ -import { join } from 'path'; -import AutoLoad, {AutoloadPluginOptions} from '@fastify/autoload'; -import { FastifyPluginAsync, FastifyServerOptions } from 'fastify'; +import { join } from 'node:path' +import AutoLoad, { type AutoloadPluginOptions } from '@fastify/autoload' +import type { FastifyPluginAsync, FastifyServerOptions } from 'fastify' import { initOpenTelemetry } from './lib/open-telemetry' -export interface AppOptions extends FastifyServerOptions, Partial { - -} +export interface AppOptions extends FastifyServerOptions, Partial {} // Pass --options via CLI arguments in command to enable these options. -const options: AppOptions = { -} +const options: AppOptions = {} -const app: FastifyPluginAsync = async ( - fastify, - opts -): Promise => { +const app: FastifyPluginAsync = async (fastify, opts): Promise => { // Place here your custom code! initOpenTelemetry() // Do not touch the following lines void fastify.register(AutoLoad, { dir: join(__dirname, 'plugins'), - options: opts + options: opts, }) void fastify.register(AutoLoad, { dir: join(__dirname, 'routes'), - options: opts + options: opts, }) -}; +} -export default app; +export default app export { app, options } diff --git a/apps/api-fastify/src/lib/open-telemetry.ts b/apps/api-fastify/src/lib/open-telemetry.ts index e92a696..9576442 100644 --- a/apps/api-fastify/src/lib/open-telemetry.ts +++ b/apps/api-fastify/src/lib/open-telemetry.ts @@ -1,7 +1,7 @@ import { TraceExporter } from '@google-cloud/opentelemetry-cloud-trace-exporter' import { FastifyInstrumentation } from '@opentelemetry/instrumentation-fastify' -import { PinoInstrumentation } from '@opentelemetry/instrumentation-pino' import { HttpInstrumentation } from '@opentelemetry/instrumentation-http' +import { PinoInstrumentation } from '@opentelemetry/instrumentation-pino' import { WinstonInstrumentation } from '@opentelemetry/instrumentation-winston' import { NodeSDK } from '@opentelemetry/sdk-node' import p from '@prisma/instrumentation' diff --git a/apps/api-fastify/src/plugins/compress.ts b/apps/api-fastify/src/plugins/compress.ts index 0c2ccfb..027999c 100644 --- a/apps/api-fastify/src/plugins/compress.ts +++ b/apps/api-fastify/src/plugins/compress.ts @@ -1,7 +1,7 @@ +import fastifyCompress from '@fastify/compress' +import type { FastifyPluginAsync } from 'fastify' // plugins/compress.ts import fp from 'fastify-plugin' -import fastifyCompress from '@fastify/compress' -import { FastifyPluginAsync } from 'fastify' const compressPlugin: FastifyPluginAsync = async (fastify, opts) => { fastify.register(fastifyCompress, opts) diff --git a/apps/api-fastify/src/plugins/http-redirect.ts b/apps/api-fastify/src/plugins/http-redirect.ts index 898786f..5a95eed 100644 --- a/apps/api-fastify/src/plugins/http-redirect.ts +++ b/apps/api-fastify/src/plugins/http-redirect.ts @@ -1,6 +1,6 @@ +import type { FastifyPluginAsync } from 'fastify' // plugins/httpsRedirect.ts import fp from 'fastify-plugin' -import { FastifyPluginAsync } from 'fastify' const httpsRedirectPlugin: FastifyPluginAsync = async (fastify) => { fastify.addHook('onRequest', (request, reply, done) => { diff --git a/apps/api-fastify/src/plugins/rate-limit.ts b/apps/api-fastify/src/plugins/rate-limit.ts index 5bf69af..99881ec 100644 --- a/apps/api-fastify/src/plugins/rate-limit.ts +++ b/apps/api-fastify/src/plugins/rate-limit.ts @@ -1,8 +1,8 @@ +import type { FastifyPluginAsync } from 'fastify' import fp from 'fastify-plugin' -import { FastifyPluginAsync } from 'fastify' -const rateLimitPlugin: FastifyPluginAsync = async (fastify, opts) => { -// TODO +const rateLimitPlugin: FastifyPluginAsync = async () => { + // TODO } export default fp(rateLimitPlugin) diff --git a/apps/api-fastify/src/plugins/sensible.ts b/apps/api-fastify/src/plugins/sensible.ts index 5be2c8a..a53651b 100644 --- a/apps/api-fastify/src/plugins/sensible.ts +++ b/apps/api-fastify/src/plugins/sensible.ts @@ -1,5 +1,5 @@ +import sensible, { type FastifySensibleOptions } from '@fastify/sensible' import fp from 'fastify-plugin' -import sensible, { FastifySensibleOptions } from '@fastify/sensible' /** * This plugins adds some utilities to handle http errors diff --git a/apps/api-fastify/src/plugins/support.ts b/apps/api-fastify/src/plugins/support.ts deleted file mode 100644 index 94bae4f..0000000 --- a/apps/api-fastify/src/plugins/support.ts +++ /dev/null @@ -1,20 +0,0 @@ -import fp from 'fastify-plugin' - -export interface SupportPluginOptions { - // Specify Support plugin options here -} - -// The use of fastify-plugin is required to be able -// to export the decorators to the outer scope -export default fp(async (fastify, opts) => { - fastify.decorate('someSupport', function () { - return 'hugs' - }) -}) - -// When using .decorate you have to specify added properties for Typescript -declare module 'fastify' { - export interface FastifyInstance { - someSupport(): string; - } -} diff --git a/apps/api-fastify/src/routes/example/index.ts b/apps/api-fastify/src/routes/example/index.ts index 819c5e7..804961f 100644 --- a/apps/api-fastify/src/routes/example/index.ts +++ b/apps/api-fastify/src/routes/example/index.ts @@ -1,9 +1,7 @@ -import { FastifyPluginAsync } from "fastify" +import type { FastifyPluginAsync } from 'fastify' -const example: FastifyPluginAsync = async (fastify, opts): Promise => { - fastify.get('/', async function (request, reply) { - return 'this is an example' - }) +const example: FastifyPluginAsync = async (fastify): Promise => { + fastify.get('/', async () => 'this is an example') } -export default example; +export default example diff --git a/apps/api-fastify/src/routes/root.ts b/apps/api-fastify/src/routes/root.ts index 2a1b334..4377a3a 100644 --- a/apps/api-fastify/src/routes/root.ts +++ b/apps/api-fastify/src/routes/root.ts @@ -1,9 +1,7 @@ -import { FastifyPluginAsync } from 'fastify' +import type { FastifyPluginAsync } from 'fastify' -const root: FastifyPluginAsync = async (fastify, opts): Promise => { - fastify.get('/', async function (request, reply) { - return { root: true } - }) +const root: FastifyPluginAsync = async (fastify): Promise => { + fastify.get('/', async () => ({ root: true })) } -export default root; +export default root diff --git a/apps/api-fastify/test/helper.ts b/apps/api-fastify/test/helper.ts index 7045177..cab4ee8 100644 --- a/apps/api-fastify/test/helper.ts +++ b/apps/api-fastify/test/helper.ts @@ -1,22 +1,22 @@ // This file contains code that we reuse between our tests. const helper = require('fastify-cli/helper.js') -import * as path from 'path' -import * as test from 'node:test' +import * as path from 'node:path' +import type * as test from 'node:test' export type TestContext = { after: typeof test.after -}; +} const AppPath = path.join(__dirname, '..', 'src', 'app.ts') // Fill in this config with all the configurations // needed for testing the application -async function config () { +async function config() { return {} } // Automatically build and tear down our instance -async function build (t: TestContext) { +async function build(t: TestContext) { // you can set all the options supported by the fastify CLI command const argv = [AppPath] @@ -31,7 +31,4 @@ async function build (t: TestContext) { return app } -export { - config, - build -} +export { config, build } diff --git a/apps/api-fastify/test/plugins/support.test.ts b/apps/api-fastify/test/plugins/support.test.ts index d3b5c46..1e8a2c6 100644 --- a/apps/api-fastify/test/plugins/support.test.ts +++ b/apps/api-fastify/test/plugins/support.test.ts @@ -1,10 +1,10 @@ -import { test } from 'node:test' import * as assert from 'node:assert' +import { test } from 'node:test' import Fastify from 'fastify' import Support from '../../src/plugins/support' -test('support works standalone', async (t) => { +test('support works standalone', async () => { const fastify = Fastify() void fastify.register(Support) await fastify.ready() diff --git a/apps/api-fastify/test/routes/example.test.ts b/apps/api-fastify/test/routes/example.test.ts index 0c35dfe..2aa7ea2 100644 --- a/apps/api-fastify/test/routes/example.test.ts +++ b/apps/api-fastify/test/routes/example.test.ts @@ -1,12 +1,12 @@ -import { test } from 'node:test' import * as assert from 'node:assert' +import { test } from 'node:test' import { build } from '../helper' test('example is loaded', async (t) => { const app = await build(t) const res = await app.inject({ - url: '/example' + url: '/example', }) assert.equal(res.payload, 'this is an example') diff --git a/apps/api-fastify/test/routes/root.test.ts b/apps/api-fastify/test/routes/root.test.ts index 17554ec..57f10d2 100644 --- a/apps/api-fastify/test/routes/root.test.ts +++ b/apps/api-fastify/test/routes/root.test.ts @@ -1,12 +1,12 @@ -import { test } from 'node:test' import * as assert from 'node:assert' +import { test } from 'node:test' import { build } from '../helper' test('default root route', async (t) => { const app = await build(t) const res = await app.inject({ - url: '/' + url: '/', }) assert.deepStrictEqual(JSON.parse(res.payload), { root: true }) }) diff --git a/package.json b/package.json index e8da4ff..b2f30e5 100644 --- a/package.json +++ b/package.json @@ -10,6 +10,7 @@ "format": "prettier --write \"**/*.{ts,tsx,md}\"" }, "devDependencies": { + "@biomejs/biome": "1.8.3", "prettier": "3.3.3", "turbo": "2.1.2", "typescript": "5.6.2" diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index f2615cd..2bc0ebd 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -8,6 +8,9 @@ importers: .: devDependencies: + '@biomejs/biome': + specifier: 1.8.3 + version: 1.8.3 prettier: specifier: 3.3.3 version: 3.3.3 From f04781f2406a5936dc2a9dfbb4a71281e70a727a Mon Sep 17 00:00:00 2001 From: isoppp Date: Tue, 24 Sep 2024 02:57:57 +0900 Subject: [PATCH 4/4] adjusted logger --- apps/api-fastify/package.json | 9 ++-- apps/api-fastify/src/app.ts | 39 ++++++++++++--- apps/api-fastify/src/lib/logger.ts | 18 ++++++- .../api-fastify/src/plugins/request-logger.ts | 50 +++++++++++++++++++ apps/api-fastify/tsconfig.json | 15 +++++- pnpm-lock.yaml | 21 ++++++++ 6 files changed, 140 insertions(+), 12 deletions(-) create mode 100644 apps/api-fastify/src/plugins/request-logger.ts diff --git a/apps/api-fastify/package.json b/apps/api-fastify/package.json index 9ad1a4c..5db1143 100644 --- a/apps/api-fastify/package.json +++ b/apps/api-fastify/package.json @@ -6,13 +6,13 @@ "directories": { "test": "test" }, + "type": "module", "scripts": { "test": "npm run build:ts && tsc -p test/tsconfig.json && c8 node --test -r ts-node/register \"test/**/*.ts\"", - "start": "npm run build:ts && fastify start -l info dist/app.js", + "start": "tsx src/app.ts", "build:ts": "tsc", "watch:ts": "tsc -w", - "dev": "npm run build:ts && concurrently -k -p \"[{name}]\" -n \"TypeScript,App\" -c \"yellow.bold,cyan.bold\" \"npm:watch:ts\" \"npm:dev:start\"", - "dev:start": "fastify start --ignore-watch=.ts$ -w -l info -P dist/app.js", + "dev": "tsx --watch src/app.ts", "lint-fix": "pnpm biome check --fix --unsafe" }, "keywords": [], @@ -32,10 +32,13 @@ "@opentelemetry/instrumentation-winston": "0.40.0", "@opentelemetry/sdk-node": "0.53.0", "@prisma/instrumentation": "5.19.1", + "chalk": "5.3.0", "fastify": "^5.0.0", "fastify-cli": "^7.0.1", "fastify-plugin": "^5.0.0", + "fastify-winston": "1.0.8", "helmet": "7.1.0", + "tsx": "4.19.0", "winston": "3.14.2" }, "devDependencies": { diff --git a/apps/api-fastify/src/app.ts b/apps/api-fastify/src/app.ts index e4154ba..6ed3b61 100644 --- a/apps/api-fastify/src/app.ts +++ b/apps/api-fastify/src/app.ts @@ -1,11 +1,21 @@ -import { join } from 'node:path' -import AutoLoad, { type AutoloadPluginOptions } from '@fastify/autoload' +import { logger } from '@/lib/logger' +import type { AutoloadPluginOptions } from '@fastify/autoload' +import chalk from 'chalk' import type { FastifyPluginAsync, FastifyServerOptions } from 'fastify' +export interface AppOptions extends FastifyServerOptions, Partial {} +import { dirname, join } from 'node:path' +import { fileURLToPath } from 'node:url' +import AutoLoad from '@fastify/autoload' +import Fastify from 'fastify' import { initOpenTelemetry } from './lib/open-telemetry' -export interface AppOptions extends FastifyServerOptions, Partial {} +const __filename = fileURLToPath(import.meta.url) +const __dirname = dirname(__filename) + // Pass --options via CLI arguments in command to enable these options. -const options: AppOptions = {} +const options: AppOptions = { + logger: false, +} const app: FastifyPluginAsync = async (fastify, opts): Promise => { // Place here your custom code! @@ -23,5 +33,22 @@ const app: FastifyPluginAsync = async (fastify, opts): Promise }) } -export default app -export { app, options } +// アプリケーションを起動する関数 +const start = async () => { + const fastify = Fastify(options) + await fastify.register(app) + try { + fastify.listen({ port: 3000, host: '127.0.0.1' }, (err, address) => { + if (err) { + logger.error(err) + } else { + logger.info(chalk.blueBright(`Server listening on ${address}`)) + } + }) + } catch (err) { + logger.error(err) + process.exit(1) + } +} + +await start() diff --git a/apps/api-fastify/src/lib/logger.ts b/apps/api-fastify/src/lib/logger.ts index 4b65171..7572c11 100644 --- a/apps/api-fastify/src/lib/logger.ts +++ b/apps/api-fastify/src/lib/logger.ts @@ -3,11 +3,27 @@ import winston from 'winston' // Imports the Google Cloud client library for Winston import { LoggingWinston } from '@google-cloud/logging-winston' +const pinoLikeFormat = winston.format.printf(({ level, message, timestamp, ...metadata }) => { + const pid = process.pid + + const formattedMessage = `[${timestamp}] ${level.toUpperCase()} (${pid}): ${message}` + // if (Object.keys(metadata).length > 0) { + // formattedMessage += JSON.stringify(metadata) + // } + return formattedMessage +}) + export const logger = winston.createLogger({ level: 'info', transports: [ new winston.transports.Console({ - format: winston.format.combine(winston.format.colorize(), winston.format.cli()), + format: winston.format.combine( + winston.format.timestamp({ + format: 'HH:mm:ss.SSS', + }), + winston.format.splat(), + pinoLikeFormat, + ), }), process.env.GOOGLE_APPLICATION_CREDENTIALS ? new LoggingWinston({ diff --git a/apps/api-fastify/src/plugins/request-logger.ts b/apps/api-fastify/src/plugins/request-logger.ts new file mode 100644 index 0000000..e100c04 --- /dev/null +++ b/apps/api-fastify/src/plugins/request-logger.ts @@ -0,0 +1,50 @@ +import { logger } from '@/lib/logger' +import chalk from 'chalk' +import type { FastifyPluginAsync } from 'fastify' +import fp from 'fastify-plugin' + +const getStatusColor = (status: number) => { + if (status >= 500) return chalk.red(status) + if (status >= 400) return chalk.red(status) + if (status >= 300) return chalk.yellow(status) + if (status >= 200) return chalk.blue(status) + return status +} + +const getDurationInMilliseconds = (start: [number, number]) => { + const diff = process.hrtime(start) + return diff[0] * 1000 + diff[1] / 1e6 +} +// プラグインの定義 +const requestLogger: FastifyPluginAsync = async (fastify) => { + // Fastifyにロガーインスタンスを追加 + // fastify.decorate('logger', logger) + let start: [number, number] = [0, 0] + + // ログをリクエスト時とレスポンス時に出力する + fastify.addHook('onRequest', (request, _, done) => { + start = process.hrtime() + logger.info(`<-- ${request.id} ${request.method} ${request.url}`, { + ip: request.ip, + requestId: request.id, + hostname: request.hostname, + }) + done() + }) + + fastify.addHook('onResponse', (request, reply, done) => { + const durationInMilliseconds = getDurationInMilliseconds(start) + + logger.info( + `--> ${request.id} ${request.method} ${getStatusColor(reply.statusCode)} ${request.url} ${durationInMilliseconds}ms`, + { + ip: request.ip, + requestId: request.id, + ms: durationInMilliseconds, + }, + ) + done() + }) +} + +export default fp(requestLogger) diff --git a/apps/api-fastify/tsconfig.json b/apps/api-fastify/tsconfig.json index 50dd099..8938a1f 100644 --- a/apps/api-fastify/tsconfig.json +++ b/apps/api-fastify/tsconfig.json @@ -1,8 +1,19 @@ { "extends": "fastify-tsconfig", "compilerOptions": { - "outDir": "dist", - "sourceMap": true + "module": "ESNext", + "moduleResolution": "Bundler", + "target": "ES2022", + "sourceMap": true, + "strict": true, + "noUncheckedIndexedAccess": true, // added + "allowJs": true, + "skipLibCheck": true, + "forceConsistentCasingInFileNames": true, + "baseUrl": ".", + "paths": { + "@/*": ["./src/*"] + } }, "include": ["src/**/*.ts"] } diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 2bc0ebd..5131049 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -62,6 +62,9 @@ importers: '@prisma/instrumentation': specifier: 5.19.1 version: 5.19.1 + chalk: + specifier: 5.3.0 + version: 5.3.0 fastify: specifier: ^5.0.0 version: 5.0.0 @@ -71,9 +74,15 @@ importers: fastify-plugin: specifier: ^5.0.0 version: 5.0.1 + fastify-winston: + specifier: 1.0.8 + version: 1.0.8(fastify@5.0.0)(winston@3.14.2) helmet: specifier: 7.1.0 version: 7.1.0 + tsx: + specifier: 4.19.0 + version: 4.19.0 winston: specifier: 3.14.2 version: 3.14.2 @@ -4374,6 +4383,13 @@ packages: resolution: {integrity: sha512-pvYwdtbZUJr/aTD7ZE0rGlvtYpx7IThHKVLBoqCKmT3FJpwm23XA2+PDmq8ZzfqqG4ajpyrHd5bkIixcIFjPhQ==} engines: {node: '>=18.0.0'} + fastify-winston@1.0.8: + resolution: {integrity: sha512-bgrHZ3rkr/+No1xaItEkPfIpahYSp4lnVzxkAkcF6SjcLtou3XMZlCEt0l27P/YWob9Wxd75PYBuaN26im7GZw==} + engines: {node: '>=16'} + peerDependencies: + fastify: '4' + winston: '3' + fastify@5.0.0: resolution: {integrity: sha512-Qe4dU+zGOzg7vXjw4EvcuyIbNnMwTmcuOhlOrOJsgwzvjEZmsM/IeHulgJk+r46STjdJS/ZJbxO8N70ODXDMEQ==} @@ -12455,6 +12471,11 @@ snapshots: fastify-tsconfig@2.0.0: {} + fastify-winston@1.0.8(fastify@5.0.0)(winston@3.14.2): + dependencies: + fastify: 5.0.0 + winston: 3.14.2 + fastify@5.0.0: dependencies: '@fastify/ajv-compiler': 4.0.1