From 04f7685a6f06e497fdafce2930b62e90384bf866 Mon Sep 17 00:00:00 2001 From: Nathan Flurry Date: Wed, 13 Mar 2024 05:52:50 +0000 Subject: [PATCH] feat: impl rate limit (#63) --- .github/workflows/test-all.yml | 4 +- .../auth/scripts/auth_email_passwordless.ts | 4 +- .../auth/scripts/verify_email_passwordless.ts | 2 +- modules/currency/scripts/deposit.ts | 2 +- modules/currency/scripts/get_balance.ts | 2 +- .../currency/scripts/get_balance_by_token.ts | 2 +- modules/currency/scripts/set_balance.ts | 2 +- modules/currency/scripts/withdraw.ts | 2 +- modules/currency/tests/e2e.ts | 6 +- modules/currency/tests/errors.ts | 30 +- modules/friends/scripts/accept_request.ts | 2 +- modules/friends/scripts/decline_request.ts | 2 +- modules/friends/scripts/list_friends.ts | 2 +- .../scripts/list_incoming_friend_requests.ts | 2 +- .../scripts/list_outgoing_friend_requests.ts | 2 +- modules/friends/scripts/remove_friend.ts | 2 +- modules/friends/scripts/send_request.ts | 2 +- modules/friends/tests/e2e.ts | 16 +- .../20240313051342_init/migration.sql | 9 + .../db/migrations/migration_lock.toml | 3 + modules/rate_limit/db/schema.prisma | 13 + modules/rate_limit/module.yaml | 4 +- modules/rate_limit/scripts/throttle.ts | 73 ++- modules/rate_limit/scripts/throttle_public.ts | 48 ++ modules/rate_limit/tests/e2e.ts | 39 ++ modules/users/module.yaml | 3 +- modules/users/scripts/create_user.ts | 2 +- modules/users/scripts/get_user.ts | 2 +- modules/users/scripts/validate_user_token.ts | 2 + modules/users/tests/create_user_token.ts | 11 - modules/users/tests/e2e.ts | 9 +- tests/basic/deno.lock | 596 +++++++++++++++++- 32 files changed, 824 insertions(+), 76 deletions(-) create mode 100644 modules/rate_limit/db/migrations/20240313051342_init/migration.sql create mode 100644 modules/rate_limit/db/migrations/migration_lock.toml create mode 100644 modules/rate_limit/db/schema.prisma create mode 100644 modules/rate_limit/scripts/throttle_public.ts create mode 100644 modules/rate_limit/tests/e2e.ts delete mode 100644 modules/users/tests/create_user_token.ts diff --git a/.github/workflows/test-all.yml b/.github/workflows/test-all.yml index 3199ee46..5e71c142 100644 --- a/.github/workflows/test-all.yml +++ b/.github/workflows/test-all.yml @@ -3,7 +3,7 @@ on: - push env: - CURRENT_WORKING_ENGINE_COMMIT: 920ac277dd0f5378cf8c3cc1ae414340dc6d76a6 + CURRENT_WORKING_ENGINE_COMMIT: cbdd1e1b364b1906347b3ef541a4a5057651650f jobs: build: @@ -40,4 +40,4 @@ jobs: # Run tests on all modules in the registry - name: Run Tests for all modules - run: cd ./opengb-registry/tests/basic && opengb test + run: cd ./opengb-registry/tests/basic && opengb test --strict-schemas --force-deploy-migrations diff --git a/modules/auth/scripts/auth_email_passwordless.ts b/modules/auth/scripts/auth_email_passwordless.ts index a1d475c5..820e9702 100644 --- a/modules/auth/scripts/auth_email_passwordless.ts +++ b/modules/auth/scripts/auth_email_passwordless.ts @@ -15,9 +15,9 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - if (!ctx.userConfig.email) throw new RuntimeError("PROVIDER_DISABLED"); + await ctx.modules.rateLimit.throttlePublic({}); - await ctx.modules.rateLimit.throttle({}); + if (!ctx.userConfig.email) throw new RuntimeError("PROVIDER_DISABLED"); // Fetch existing user if session token is provided let userId: string | undefined; diff --git a/modules/auth/scripts/verify_email_passwordless.ts b/modules/auth/scripts/verify_email_passwordless.ts index 199cf056..0757d6ea 100644 --- a/modules/auth/scripts/verify_email_passwordless.ts +++ b/modules/auth/scripts/verify_email_passwordless.ts @@ -18,7 +18,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({}); + await ctx.modules.rateLimit.throttlePublic({}); const code = req.code.toUpperCase(); diff --git a/modules/currency/scripts/deposit.ts b/modules/currency/scripts/deposit.ts index 8878153f..78f265e3 100644 --- a/modules/currency/scripts/deposit.ts +++ b/modules/currency/scripts/deposit.ts @@ -16,7 +16,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 25 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 25 }); if (req.amount < 0 || !Number.isFinite(req.amount)) { throw new RuntimeError("INVALID_AMOUNT"); diff --git a/modules/currency/scripts/get_balance.ts b/modules/currency/scripts/get_balance.ts index cdd02c01..fd93be6e 100644 --- a/modules/currency/scripts/get_balance.ts +++ b/modules/currency/scripts/get_balance.ts @@ -14,7 +14,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 25 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 25 }); return { balance: await getBalance(ctx.db, req.userId), diff --git a/modules/currency/scripts/get_balance_by_token.ts b/modules/currency/scripts/get_balance_by_token.ts index cd3512f0..f0151056 100644 --- a/modules/currency/scripts/get_balance_by_token.ts +++ b/modules/currency/scripts/get_balance_by_token.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 25 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 25 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/currency/scripts/set_balance.ts b/modules/currency/scripts/set_balance.ts index 0380af8c..f2129578 100644 --- a/modules/currency/scripts/set_balance.ts +++ b/modules/currency/scripts/set_balance.ts @@ -14,7 +14,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 25 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 25 }); if (req.balance < 0 || !Number.isFinite(req.balance)) { throw new RuntimeError("INVALID_AMOUNT"); diff --git a/modules/currency/scripts/withdraw.ts b/modules/currency/scripts/withdraw.ts index ad1a90b5..87b76bc2 100644 --- a/modules/currency/scripts/withdraw.ts +++ b/modules/currency/scripts/withdraw.ts @@ -16,7 +16,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 25 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 25 }); if (req.amount < 0 || !Number.isFinite(req.amount)) { throw new RuntimeError("INVALID_AMOUNT"); diff --git a/modules/currency/tests/e2e.ts b/modules/currency/tests/e2e.ts index df72e24a..31adcf7f 100644 --- a/modules/currency/tests/e2e.ts +++ b/modules/currency/tests/e2e.ts @@ -5,9 +5,11 @@ import { faker } from "https://deno.land/x/deno_faker@v1.0.3/mod.ts"; test( "e2e transaction", async (ctx: TestContext) => { - const { user: user, token: token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, + }); + const { token } = await ctx.modules.users.createUserToken({ + userId: user.id, }); const { updatedBalance } = await ctx.modules.currency.deposit({ diff --git a/modules/currency/tests/errors.ts b/modules/currency/tests/errors.ts index 5ab36f8c..be8fe689 100644 --- a/modules/currency/tests/errors.ts +++ b/modules/currency/tests/errors.ts @@ -19,9 +19,8 @@ test( test( "withdraw more than balance", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const { updatedBalance: _ } = await ctx.modules.currency.deposit({ @@ -39,9 +38,8 @@ test( test( "withdraw negative amount", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -54,9 +52,8 @@ test( test( "withdraw Infinity", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -72,9 +69,8 @@ test( test( "withdraw NaN", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -87,9 +83,8 @@ test( test( "deposit Infinity", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -102,9 +97,8 @@ test( test( "deposit NaN", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -117,9 +111,8 @@ test( test( "deposit negative amount", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -132,9 +125,8 @@ test( test( "set balance to negative", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -147,9 +139,8 @@ test( test( "set balance to NaN", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { @@ -163,9 +154,8 @@ test( test( "set balance to infinity", async (ctx: TestContext) => { - const { user: user, token: _token } = await ctx.modules.users.register({ + const { user: user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); const error = await assertRejects(async () => { diff --git a/modules/friends/scripts/accept_request.ts b/modules/friends/scripts/accept_request.ts index 9d661357..307e938c 100644 --- a/modules/friends/scripts/accept_request.ts +++ b/modules/friends/scripts/accept_request.ts @@ -11,7 +11,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 50 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 50 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/decline_request.ts b/modules/friends/scripts/decline_request.ts index 59c29f18..0f5f9a29 100644 --- a/modules/friends/scripts/decline_request.ts +++ b/modules/friends/scripts/decline_request.ts @@ -14,7 +14,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 50 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 50 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/list_friends.ts b/modules/friends/scripts/list_friends.ts index 0ff589e5..2e62bfd2 100644 --- a/modules/friends/scripts/list_friends.ts +++ b/modules/friends/scripts/list_friends.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 50 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 50 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/list_incoming_friend_requests.ts b/modules/friends/scripts/list_incoming_friend_requests.ts index a8cb0ed2..30a6ceb6 100644 --- a/modules/friends/scripts/list_incoming_friend_requests.ts +++ b/modules/friends/scripts/list_incoming_friend_requests.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 50 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 50 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/list_outgoing_friend_requests.ts b/modules/friends/scripts/list_outgoing_friend_requests.ts index 952e5ee9..6a9d2335 100644 --- a/modules/friends/scripts/list_outgoing_friend_requests.ts +++ b/modules/friends/scripts/list_outgoing_friend_requests.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({}); + await ctx.modules.rateLimit.throttlePublic({}); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/remove_friend.ts b/modules/friends/scripts/remove_friend.ts index 34e6f1a0..807597d0 100644 --- a/modules/friends/scripts/remove_friend.ts +++ b/modules/friends/scripts/remove_friend.ts @@ -11,7 +11,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 50 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 50 }); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/scripts/send_request.ts b/modules/friends/scripts/send_request.ts index 80f326a1..5bf24e16 100644 --- a/modules/friends/scripts/send_request.ts +++ b/modules/friends/scripts/send_request.ts @@ -14,7 +14,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({}); + await ctx.modules.rateLimit.throttlePublic({}); const { userId } = await ctx.modules.users.validateUserToken({ userToken: req.userToken, diff --git a/modules/friends/tests/e2e.ts b/modules/friends/tests/e2e.ts index 4f0dbd08..46386b09 100644 --- a/modules/friends/tests/e2e.ts +++ b/modules/friends/tests/e2e.ts @@ -3,15 +3,15 @@ import { assertEquals } from "https://deno.land/std@0.217.0/assert/mod.ts"; import { faker } from "https://deno.land/x/deno_faker@v1.0.3/mod.ts"; test("e2e accept", async (ctx: TestContext) => { - const { user: _userA, token: tokenA } = await ctx.modules.users.register({ + const { user: userA } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); + const { token: tokenA } = await ctx.modules.users.createUserToken({ userId: userA.id }); - const { user: userB, token: tokenB } = await ctx.modules.users.register({ + const { user: userB } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); + const { token: tokenB } = await ctx.modules.users.createUserToken({ userId: userB.id }); const { friendRequest } = await ctx.modules.friends.sendRequest({ userToken: tokenA.token, @@ -57,15 +57,15 @@ test("e2e accept", async (ctx: TestContext) => { }); test("e2e reject", async (ctx: TestContext) => { - const { user: _userA, token: tokenA } = await ctx.modules.users.register({ + const { user: userA } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); + const { token: tokenA } = await ctx.modules.users.createUserToken({ userId: userA.id }); - const { user: userB, token: tokenB } = await ctx.modules.users.register({ + const { user: userB } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); + const { token: tokenB } = await ctx.modules.users.createUserToken({ userId: userB.id }); const { friendRequest } = await ctx.modules.friends.sendRequest({ userToken: tokenA.token, diff --git a/modules/rate_limit/db/migrations/20240313051342_init/migration.sql b/modules/rate_limit/db/migrations/20240313051342_init/migration.sql new file mode 100644 index 00000000..84cd423a --- /dev/null +++ b/modules/rate_limit/db/migrations/20240313051342_init/migration.sql @@ -0,0 +1,9 @@ +-- CreateTable +CREATE UNLOGGED TABLE "TokenBuckets" ( + "type" TEXT NOT NULL, + "key" TEXT NOT NULL, + "tokens" BIGINT NOT NULL, + "lastRefill" TIMESTAMP(3) NOT NULL, + + CONSTRAINT "TokenBuckets_pkey" PRIMARY KEY ("type","key") +); diff --git a/modules/rate_limit/db/migrations/migration_lock.toml b/modules/rate_limit/db/migrations/migration_lock.toml new file mode 100644 index 00000000..fbffa92c --- /dev/null +++ b/modules/rate_limit/db/migrations/migration_lock.toml @@ -0,0 +1,3 @@ +# Please do not edit this file manually +# It should be added in your version-control system (i.e. Git) +provider = "postgresql" \ No newline at end of file diff --git a/modules/rate_limit/db/schema.prisma b/modules/rate_limit/db/schema.prisma new file mode 100644 index 00000000..1199d381 --- /dev/null +++ b/modules/rate_limit/db/schema.prisma @@ -0,0 +1,13 @@ +datasource db { + provider = "postgresql" + url = env("DATABASE_URL") +} + +model TokenBuckets { + type String + key String + tokens BigInt + lastRefill DateTime + + @@id([type, key]) +} diff --git a/modules/rate_limit/module.yaml b/modules/rate_limit/module.yaml index ab79bfd6..9e39d183 100644 --- a/modules/rate_limit/module.yaml +++ b/modules/rate_limit/module.yaml @@ -3,4 +3,6 @@ authors: - NathanFlurry scripts: throttle: {} -errors: {} + throttle_public: {} +errors: + RATE_LIMIT_EXCEEDED: {} diff --git a/modules/rate_limit/scripts/throttle.ts b/modules/rate_limit/scripts/throttle.ts index 49cafa58..153e2276 100644 --- a/modules/rate_limit/scripts/throttle.ts +++ b/modules/rate_limit/scripts/throttle.ts @@ -1,31 +1,84 @@ +import { RuntimeError } from "../../auth/_gen/mod.ts"; import { ScriptContext } from "../_gen/scripts/throttle.ts"; export interface Request { /** - * The preset to use for rate limiting. + * The type of entity to rate limit by. For example, "ip" or "user". */ - preset?: string; + type: string; + + /** + * The key to rate limit by. For example, the IP address or user ID. + */ + key: string; /** * Number of requests in `period` before rate limiting. - * @default 20 */ - requests?: number; + requests: number; /** - * How frequently to reset the request counter. - * @default 300 + * How frequently to reset the request counter, in seconds. */ - period?: number; + period: number; } export type Response = Record; export async function run( - _ctx: ScriptContext, - _req: Request, + ctx: ScriptContext, + req: Request, ): Promise { - // TODO: + interface TokenBucket { + tokens: number; + lastRefill: Date; + } + + // Update the token bucket + // + // `TokenBucket` is an unlogged table which are significantly faster to + // write to than regular tables, but are not durable. This is important + // because this script will be called on every request. + const rows = await ctx.db.$queryRaw` + WITH updated_bucket AS ( + UPDATE "TokenBuckets" b + SET + "tokens" = CASE + -- Reset the bucket and consume 1 token + WHEN now() > b."lastRefill" + make_interval(secs => ${req.period}) THEN ${ + req.requests - 1 + } + -- Consume 1 token + ELSE b.tokens - 1 + END, + "lastRefill" = CASE + WHEN now() > b."lastRefill" + make_interval(secs => ${req.period}) THEN now() + ELSE b."lastRefill" + END + WHERE b."type" = ${req.type} AND b."key" = ${req.key} + RETURNING b."tokens", b."lastRefill" + ), + inserted AS ( + INSERT INTO "TokenBuckets" ("type", "key", "tokens", "lastRefill") + SELECT ${req.type}, ${req.key}, ${req.requests - 1}, now() + WHERE NOT EXISTS (SELECT 1 FROM updated_bucket) + RETURNING "tokens", "lastRefill" + ) + SELECT * FROM updated_bucket + UNION ALL + SELECT * FROM inserted; + `; + const { tokens, lastRefill } = rows[0]; + + // If the bucket is empty, throw an error + if (tokens < 0) { + throw new RuntimeError("RATE_LIMIT_EXCEEDED", { + meta: { + retryAfter: new Date(lastRefill.getTime() + req.period * 1000) + .toUTCString(), + }, + }); + } return {}; } diff --git a/modules/rate_limit/scripts/throttle_public.ts b/modules/rate_limit/scripts/throttle_public.ts new file mode 100644 index 00000000..b2a1528e --- /dev/null +++ b/modules/rate_limit/scripts/throttle_public.ts @@ -0,0 +1,48 @@ +import { ScriptContext } from "../_gen/scripts/throttle_public.ts"; + +export interface Request { + /** + * Number of requests in `period` before rate limiting. + * @default 20 + */ + requests?: number; + + /** + * How frequently to reset the request counter. + * @default 300 + */ + period?: number; +} + +export type Response = Record; + +export async function run( + ctx: ScriptContext, + req: Request, +): Promise { + const requests = req.requests || 20; + const period = req.period || 300; + + // Find the IP address of the client + let key: string | undefined; + for (const entry of ctx.trace.entries) { + if ("httpRequest" in entry.type) { + key = entry.type.httpRequest.remoteAddress; + break; + } + } + + // If no IP address, this request is not coming from a client + if (!key) { + return {}; + } + + await ctx.modules.rateLimit.throttle({ + type: "ip", + key, + requests: req.requests ?? 20, + period: req.period ?? 300, + }); + + return {}; +} diff --git a/modules/rate_limit/tests/e2e.ts b/modules/rate_limit/tests/e2e.ts new file mode 100644 index 00000000..a1473caa --- /dev/null +++ b/modules/rate_limit/tests/e2e.ts @@ -0,0 +1,39 @@ +import { RuntimeError, test, TestContext } from "../_gen/test.ts"; +import { + assertEquals, + assertRejects, +} from "https://deno.land/std@0.208.0/assert/mod.ts"; +import { delay } from "https://deno.land/std@0.214.0/async/delay.ts"; + +test("e2e", async (ctx: TestContext) => { + const testId = crypto.randomUUID(); + + const requests = 5; + const period = 2; + + const makeRequest = async () => { + await ctx.modules.rateLimit.throttle({ + type: "test", + key: testId, + requests, + period, + }); + }; + + // Exhause rate limit + for (let i = 0; i < 5; i++) { + await makeRequest(); + } + + // Should be rate limited for all future requests + for (let i = 0; i < 3; i++) { + const error = await assertRejects(() => makeRequest(), RuntimeError); + assertEquals("RATE_LIMIT_EXCEEDED", error.code); + } + + // Wait for the rate limit to reset + await delay(period * 1000); + + // Should be able to make requests again + await makeRequest(); +}); diff --git a/modules/users/module.yaml b/modules/users/module.yaml index beb2228b..e051c4fd 100644 --- a/modules/users/module.yaml +++ b/modules/users/module.yaml @@ -8,7 +8,8 @@ scripts: get_user: public: true create_user: {} - validate_user_token: {} + validate_user_token: + public: true create_user_token: {} errors: TOKEN_NOT_USER_TOKEN: {} diff --git a/modules/users/scripts/create_user.ts b/modules/users/scripts/create_user.ts index fd8b68b0..594deb61 100644 --- a/modules/users/scripts/create_user.ts +++ b/modules/users/scripts/create_user.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({ requests: 2, period: 5 * 60 }); + await ctx.modules.rateLimit.throttlePublic({ requests: 2, period: 5 * 60 }); // Create user const user = await ctx.db.user.create({ diff --git a/modules/users/scripts/get_user.ts b/modules/users/scripts/get_user.ts index ada81b7f..9e6dbe20 100644 --- a/modules/users/scripts/get_user.ts +++ b/modules/users/scripts/get_user.ts @@ -13,7 +13,7 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { - await ctx.modules.rateLimit.throttle({}); + await ctx.modules.rateLimit.throttlePublic({}); const users = await ctx.db.user.findMany({ where: { id: { in: req.userIds } }, diff --git a/modules/users/scripts/validate_user_token.ts b/modules/users/scripts/validate_user_token.ts index 7de210da..29e0c6dc 100644 --- a/modules/users/scripts/validate_user_token.ts +++ b/modules/users/scripts/validate_user_token.ts @@ -13,6 +13,8 @@ export async function run( ctx: ScriptContext, req: Request, ): Promise { + await ctx.modules.rateLimit.throttlePublic({}); + const { token } = await ctx.modules.tokens.validate({ token: req.userToken, }); diff --git a/modules/users/tests/create_user_token.ts b/modules/users/tests/create_user_token.ts deleted file mode 100644 index b4fccb6c..00000000 --- a/modules/users/tests/create_user_token.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { test, TestContext } from "../_gen/test.ts"; -import { - assert, - assertEquals, -} from "https://deno.land/std@0.208.0/assert/mod.ts"; - -test("e2e", async (ctx: TestContext) => { - const res = await ctx.modules.users.createUserToken({ - // TODO: - }); -}); diff --git a/modules/users/tests/e2e.ts b/modules/users/tests/e2e.ts index 7e4851ba..a755e2c3 100644 --- a/modules/users/tests/e2e.ts +++ b/modules/users/tests/e2e.ts @@ -4,16 +4,19 @@ import { assertEquals } from "https://deno.land/std@0.217.0/assert/assert_equals import { assertExists } from "https://deno.land/std@0.217.0/assert/assert_exists.ts"; test("e2e", async (ctx: TestContext) => { - const { user, token } = await ctx.modules.users.register({ + const { user } = await ctx.modules.users.createUser({ username: faker.internet.userName(), - identity: { guest: {} }, }); - const { users } = await ctx.modules.users.get({ + const { users } = await ctx.modules.users.getUser({ userIds: [user.id], }); assertExists(users[0]); + const { token } = await ctx.modules.users.createUserToken({ + userId: user.id, + }); + const { userId } = await ctx.modules.users.validateUserToken({ userToken: token.token, }); diff --git a/tests/basic/deno.lock b/tests/basic/deno.lock index 67e629cc..b5b563c7 100644 --- a/tests/basic/deno.lock +++ b/tests/basic/deno.lock @@ -4,9 +4,33 @@ "specifiers": { "npm:@prisma/adapter-pg@^5.9.1": "npm:@prisma/adapter-pg@5.9.1_pg@8.11.3", "npm:@types/node": "npm:@types/node@18.16.19", - "npm:pg@^8.11.3": "npm:pg@8.11.3" + "npm:ajv-formats@^2.1.1": "npm:ajv-formats@2.1.1_ajv@8.12.0", + "npm:ajv@^8.12.0": "npm:ajv@8.12.0", + "npm:pg@^8.11.3": "npm:pg@8.11.3", + "npm:typescript-json-schema@^0.62.0": "npm:typescript-json-schema@0.62.0_@types+node@18.16.19_typescript@5.1.6" }, "npm": { + "@cspotcode/source-map-support@0.8.1": { + "integrity": "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==", + "dependencies": { + "@jridgewell/trace-mapping": "@jridgewell/trace-mapping@0.3.9" + } + }, + "@jridgewell/resolve-uri@3.1.1": { + "integrity": "sha512-dSYZh7HhCDtCKm4QakX0xFpsRDqjjtZf/kjI/v3T3Nwt5r8/qz/M19F9ySyOqU94SXBmeG9ttTul+YnR4LOxFA==", + "dependencies": {} + }, + "@jridgewell/sourcemap-codec@1.4.15": { + "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "dependencies": {} + }, + "@jridgewell/trace-mapping@0.3.9": { + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dependencies": { + "@jridgewell/resolve-uri": "@jridgewell/resolve-uri@3.1.1", + "@jridgewell/sourcemap-codec": "@jridgewell/sourcemap-codec@1.4.15" + } + }, "@prisma/adapter-pg@5.9.1_pg@8.11.3": { "integrity": "sha512-0RhnB1sqLmSzilbIQS75YE5qTNotlGvLWhTG7cp+F59VJdQ5sWWXZ1j2W4Tn+AMb116Yp4AuFe0vggW2f+ujiQ==", "dependencies": { @@ -21,28 +45,204 @@ "debug": "debug@4.3.4" } }, + "@tsconfig/node10@1.0.9": { + "integrity": "sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==", + "dependencies": {} + }, + "@tsconfig/node12@1.0.11": { + "integrity": "sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag==", + "dependencies": {} + }, + "@tsconfig/node14@1.0.3": { + "integrity": "sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow==", + "dependencies": {} + }, + "@tsconfig/node16@1.0.4": { + "integrity": "sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA==", + "dependencies": {} + }, + "@types/json-schema@7.0.15": { + "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==", + "dependencies": {} + }, + "@types/node@16.18.79": { + "integrity": "sha512-Qd7jdLR5zmnIyMhfDrfPqN5tUCvreVpP3Qrf2oSM+F7SNzlb/MwHISGUkdFHtevfkPJ3iAGyeQI/jsbh9EStgQ==", + "dependencies": {} + }, "@types/node@18.16.19": { "integrity": "sha512-IXl7o+R9iti9eBW4Wg2hx1xQDig183jj7YLn8F7udNceyfkbn1ZxmzZXuak20gR40D7pIkIY1kYGx5VIGbaHKA==", "dependencies": {} }, + "acorn-walk@8.3.2": { + "integrity": "sha512-cjkyv4OtNCIeqhHrfS81QWXoCBPExR/J62oyEqepVw8WaQeSqpW2uhuLPh1m9eWhDuOo/jUXVTlifvesOWp/4A==", + "dependencies": {} + }, + "acorn@8.11.3": { + "integrity": "sha512-Y9rRfJG5jcKOE0CLisYbojUjIrIEE7AGMzA/Sm4BslANhbS+cDMpgBdcPT91oJ7OuJ9hYJBx59RjbhxVnrF8Xg==", + "dependencies": {} + }, + "ajv-formats@2.1.1_ajv@8.12.0": { + "integrity": "sha512-Wx0Kx52hxE7C18hkMEggYlEifqWZtYaRgouJor+WMdPnQyEK13vgEWyVNup7SoeeoLMsr4kf5h6dOW11I15MUA==", + "dependencies": { + "ajv": "ajv@8.12.0" + } + }, + "ajv@8.12.0": { + "integrity": "sha512-sRu1kpcO9yLtYxBKvqfTeh9KzZEwO3STyX1HT+4CaDzC6HpTGYhIhPIzj9XuKU7KYDwnaeh5hcOwjy1QuJzBPA==", + "dependencies": { + "fast-deep-equal": "fast-deep-equal@3.1.3", + "json-schema-traverse": "json-schema-traverse@1.0.0", + "require-from-string": "require-from-string@2.0.2", + "uri-js": "uri-js@4.4.1" + } + }, + "ansi-regex@5.0.1": { + "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", + "dependencies": {} + }, + "ansi-styles@4.3.0": { + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": { + "color-convert": "color-convert@2.0.1" + } + }, + "arg@4.1.3": { + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dependencies": {} + }, + "balanced-match@1.0.2": { + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dependencies": {} + }, + "brace-expansion@1.1.11": { + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dependencies": { + "balanced-match": "balanced-match@1.0.2", + "concat-map": "concat-map@0.0.1" + } + }, "buffer-writer@2.0.0": { "integrity": "sha512-a7ZpuTZU1TRtnwyCNW3I5dc0wWNC3VR9S++Ewyk2HHZdrO3CQJqSpd+95Us590V6AL7JqUAH2IwZ/398PmNFgw==", "dependencies": {} }, + "cliui@8.0.1": { + "integrity": "sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ==", + "dependencies": { + "string-width": "string-width@4.2.3", + "strip-ansi": "strip-ansi@6.0.1", + "wrap-ansi": "wrap-ansi@7.0.0" + } + }, + "color-convert@2.0.1": { + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dependencies": { + "color-name": "color-name@1.1.4" + } + }, + "color-name@1.1.4": { + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dependencies": {} + }, + "concat-map@0.0.1": { + "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", + "dependencies": {} + }, + "create-require@1.1.1": { + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dependencies": {} + }, "debug@4.3.4": { "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", "dependencies": { "ms": "ms@2.1.2" } }, + "diff@4.0.2": { + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dependencies": {} + }, + "emoji-regex@8.0.0": { + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dependencies": {} + }, + "escalade@3.1.2": { + "integrity": "sha512-ErCHMCae19vR8vQGe50xIsVomy19rg6gFu3+r3jkEO46suLMWBksvVyoGgQV+jOfl84ZSOSlmv6Gxa89PmTGmA==", + "dependencies": {} + }, + "fast-deep-equal@3.1.3": { + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dependencies": {} + }, + "fs.realpath@1.0.0": { + "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==", + "dependencies": {} + }, + "get-caller-file@2.0.5": { + "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", + "dependencies": {} + }, + "glob@7.2.3": { + "integrity": "sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==", + "dependencies": { + "fs.realpath": "fs.realpath@1.0.0", + "inflight": "inflight@1.0.6", + "inherits": "inherits@2.0.4", + "minimatch": "minimatch@3.1.2", + "once": "once@1.4.0", + "path-is-absolute": "path-is-absolute@1.0.1" + } + }, + "inflight@1.0.6": { + "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==", + "dependencies": { + "once": "once@1.4.0", + "wrappy": "wrappy@1.0.2" + } + }, + "inherits@2.0.4": { + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", + "dependencies": {} + }, + "is-fullwidth-code-point@3.0.0": { + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dependencies": {} + }, + "json-schema-traverse@1.0.0": { + "integrity": "sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug==", + "dependencies": {} + }, + "make-error@1.3.6": { + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dependencies": {} + }, + "minimatch@3.1.2": { + "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", + "dependencies": { + "brace-expansion": "brace-expansion@1.1.11" + } + }, "ms@2.1.2": { "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", "dependencies": {} }, + "once@1.4.0": { + "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==", + "dependencies": { + "wrappy": "wrappy@1.0.2" + } + }, "packet-reader@1.0.0": { "integrity": "sha512-HAKu/fG3HpHFO0AA8WE8q2g+gBJaZ9MG7fcKk+IJPLTGAD6Psw4443l+9DGRbOIh3/aXr7Phy0TjilYivJo5XQ==", "dependencies": {} }, + "path-equal@1.2.5": { + "integrity": "sha512-i73IctDr3F2W+bsOWDyyVm/lqsXO47aY9nsFZUjTT/aljSbkxHxxCoyZ9UUrM8jK0JVod+An+rl48RCsvWM+9g==", + "dependencies": {} + }, + "path-is-absolute@1.0.1": { + "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==", + "dependencies": {} + }, "pg-cloudflare@1.1.1": { "integrity": "sha512-xWPagP/4B6BgFO+EKz3JONXv3YDgvkbVrGw2mTo3D6tVDQRh1e7cqVGvyR3BE+eQgAvx1XhW/iEASj4/jCWl3Q==", "dependencies": {} @@ -116,13 +316,126 @@ "xtend": "xtend@4.0.2" } }, + "punycode@2.3.1": { + "integrity": "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg==", + "dependencies": {} + }, + "require-directory@2.1.1": { + "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", + "dependencies": {} + }, + "require-from-string@2.0.2": { + "integrity": "sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw==", + "dependencies": {} + }, + "safe-stable-stringify@2.4.3": { + "integrity": "sha512-e2bDA2WJT0wxseVd4lsDP4+3ONX6HpMXQa1ZhFQ7SU+GjvORCmShbCMltrtIDfkYhVHrOcPtj+KhmDBdPdZD1g==", + "dependencies": {} + }, "split2@4.2.0": { "integrity": "sha512-UcjcJOWknrNkF6PLX83qcHM6KHgVKNkV62Y8a5uYDVv9ydGQVwAHMKqHdJje1VTWpljG0WYpCDhrCdAOYH4TWg==", "dependencies": {} }, + "string-width@4.2.3": { + "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", + "dependencies": { + "emoji-regex": "emoji-regex@8.0.0", + "is-fullwidth-code-point": "is-fullwidth-code-point@3.0.0", + "strip-ansi": "strip-ansi@6.0.1" + } + }, + "strip-ansi@6.0.1": { + "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", + "dependencies": { + "ansi-regex": "ansi-regex@5.0.1" + } + }, + "ts-node@10.9.2_@types+node@18.16.19_typescript@5.1.6": { + "integrity": "sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==", + "dependencies": { + "@cspotcode/source-map-support": "@cspotcode/source-map-support@0.8.1", + "@tsconfig/node10": "@tsconfig/node10@1.0.9", + "@tsconfig/node12": "@tsconfig/node12@1.0.11", + "@tsconfig/node14": "@tsconfig/node14@1.0.3", + "@tsconfig/node16": "@tsconfig/node16@1.0.4", + "@types/node": "@types/node@18.16.19", + "acorn": "acorn@8.11.3", + "acorn-walk": "acorn-walk@8.3.2", + "arg": "arg@4.1.3", + "create-require": "create-require@1.1.1", + "diff": "diff@4.0.2", + "make-error": "make-error@1.3.6", + "typescript": "typescript@5.1.6", + "v8-compile-cache-lib": "v8-compile-cache-lib@3.0.1", + "yn": "yn@3.1.1" + } + }, + "typescript-json-schema@0.62.0_@types+node@18.16.19_typescript@5.1.6": { + "integrity": "sha512-qRO6pCgyjKJ230QYdOxDRpdQrBeeino4v5p2rYmSD72Jf4rD3O+cJcROv46sQukm46CLWoeusqvBgKpynEv25g==", + "dependencies": { + "@types/json-schema": "@types/json-schema@7.0.15", + "@types/node": "@types/node@16.18.79", + "glob": "glob@7.2.3", + "path-equal": "path-equal@1.2.5", + "safe-stable-stringify": "safe-stable-stringify@2.4.3", + "ts-node": "ts-node@10.9.2_@types+node@18.16.19_typescript@5.1.6", + "typescript": "typescript@5.1.6", + "yargs": "yargs@17.7.2" + } + }, + "typescript@5.1.6": { + "integrity": "sha512-zaWCozRZ6DLEWAWFrVDz1H6FVXzUSfTy5FUMWsQlU8Ym5JP9eO4xkTIROFCQvhQf61z6O/G6ugw3SgAnvvm+HA==", + "dependencies": {} + }, + "uri-js@4.4.1": { + "integrity": "sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg==", + "dependencies": { + "punycode": "punycode@2.3.1" + } + }, + "v8-compile-cache-lib@3.0.1": { + "integrity": "sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==", + "dependencies": {} + }, + "wrap-ansi@7.0.0": { + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "ansi-styles@4.3.0", + "string-width": "string-width@4.2.3", + "strip-ansi": "strip-ansi@6.0.1" + } + }, + "wrappy@1.0.2": { + "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==", + "dependencies": {} + }, "xtend@4.0.2": { "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", "dependencies": {} + }, + "y18n@5.0.8": { + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dependencies": {} + }, + "yargs-parser@21.1.1": { + "integrity": "sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw==", + "dependencies": {} + }, + "yargs@17.7.2": { + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", + "dependencies": { + "cliui": "cliui@8.0.1", + "escalade": "escalade@3.1.2", + "get-caller-file": "get-caller-file@2.0.5", + "require-directory": "require-directory@2.1.1", + "string-width": "string-width@4.2.3", + "y18n": "y18n@5.0.8", + "yargs-parser": "yargs-parser@21.1.1" + } + }, + "yn@3.1.1": { + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dependencies": {} } } }, @@ -131,6 +444,25 @@ "https://esm.sh/ajv@^8.12.0": "https://esm.sh/ajv@8.12.0" }, "remote": { + "https://deno.land/std@0.140.0/_util/assert.ts": "e94f2eb37cebd7f199952e242c77654e43333c1ac4c5c700e929ea3aa5489f74", + "https://deno.land/std@0.140.0/_util/os.ts": "3b4c6e27febd119d36a416d7a97bd3b0251b77c88942c8f16ee5953ea13e2e49", + "https://deno.land/std@0.140.0/bytes/bytes_list.ts": "67eb118e0b7891d2f389dad4add35856f4ad5faab46318ff99653456c23b025d", + "https://deno.land/std@0.140.0/bytes/equals.ts": "fc16dff2090cced02497f16483de123dfa91e591029f985029193dfaa9d894c9", + "https://deno.land/std@0.140.0/bytes/mod.ts": "763f97d33051cc3f28af1a688dfe2830841192a9fea0cbaa55f927b49d49d0bf", + "https://deno.land/std@0.140.0/fmt/colors.ts": "30455035d6d728394781c10755351742dd731e3db6771b1843f9b9e490104d37", + "https://deno.land/std@0.140.0/fs/_util.ts": "0fb24eb4bfebc2c194fb1afdb42b9c3dda12e368f43e8f2321f84fc77d42cb0f", + "https://deno.land/std@0.140.0/fs/ensure_dir.ts": "9dc109c27df4098b9fc12d949612ae5c9c7169507660dcf9ad90631833209d9d", + "https://deno.land/std@0.140.0/io/buffer.ts": "bd0c4bf53db4b4be916ca5963e454bddfd3fcd45039041ea161dbf826817822b", + "https://deno.land/std@0.140.0/path/_constants.ts": "df1db3ffa6dd6d1252cc9617e5d72165cd2483df90e93833e13580687b6083c3", + "https://deno.land/std@0.140.0/path/_interface.ts": "ee3b431a336b80cf445441109d089b70d87d5e248f4f90ff906820889ecf8d09", + "https://deno.land/std@0.140.0/path/_util.ts": "c1e9686d0164e29f7d880b2158971d805b6e0efc3110d0b3e24e4b8af2190d2b", + "https://deno.land/std@0.140.0/path/common.ts": "bee563630abd2d97f99d83c96c2fa0cca7cee103e8cb4e7699ec4d5db7bd2633", + "https://deno.land/std@0.140.0/path/glob.ts": "cb5255638de1048973c3e69e420c77dc04f75755524cb3b2e160fe9277d939ee", + "https://deno.land/std@0.140.0/path/mod.ts": "d3e68d0abb393fb0bf94a6d07c46ec31dc755b544b13144dee931d8d5f06a52d", + "https://deno.land/std@0.140.0/path/posix.ts": "293cdaec3ecccec0a9cc2b534302dfe308adb6f10861fa183275d6695faace44", + "https://deno.land/std@0.140.0/path/separator.ts": "fe1816cb765a8068afb3e8f13ad272351c85cbc739af56dacfc7d93d710fe0f9", + "https://deno.land/std@0.140.0/path/win32.ts": "31811536855e19ba37a999cd8d1b62078235548d67902ece4aa6b814596dd757", + "https://deno.land/std@0.140.0/streams/conversion.ts": "712585bfa0172a97fb68dd46e784ae8ad59d11b88079d6a4ab098ff42e697d21", "https://deno.land/std@0.208.0/assert/_constants.ts": "8a9da298c26750b28b326b297316cdde860bc237533b07e1337c021379e6b2a9", "https://deno.land/std@0.208.0/assert/_diff.ts": "58e1461cc61d8eb1eacbf2a010932bf6a05b79344b02ca38095f9b805795dc48", "https://deno.land/std@0.208.0/assert/_format.ts": "a69126e8a469009adf4cf2a50af889aca364c349797e63174884a52ff75cf4c7", @@ -163,6 +495,235 @@ "https://deno.land/std@0.208.0/assert/unimplemented.ts": "d56fbeecb1f108331a380f72e3e010a1f161baa6956fd0f7cf3e095ae1a4c75a", "https://deno.land/std@0.208.0/assert/unreachable.ts": "4600dc0baf7d9c15a7f7d234f00c23bca8f3eba8b140286aaca7aa998cf9a536", "https://deno.land/std@0.208.0/fmt/colors.ts": "34b3f77432925eb72cf0bfb351616949746768620b8e5ead66da532f93d10ba2", + "https://deno.land/std@0.208.0/fs/_util.ts": "fbf57dcdc9f7bc8128d60301eece608246971a7836a3bb1e78da75314f08b978", + "https://deno.land/std@0.208.0/fs/copy.ts": "ca19e4837965914471df38fbd61e16f9e8adfe89f9cffb0c83615c83ea3fc2bf", + "https://deno.land/std@0.208.0/fs/empty_dir.ts": "7fba29ef2d03f3503cd512616efc0535984cf1bbe7ca9d098e8b4d0d88910120", + "https://deno.land/std@0.208.0/fs/ensure_dir.ts": "dc64c4c75c64721d4e3fb681f1382f803ff3d2868f08563ff923fdd20d071c40", + "https://deno.land/std@0.208.0/fs/ensure_file.ts": "39ac83cc283a20ec2735e956adf5de3e8a3334e0b6820547b5772f71c49ae083", + "https://deno.land/std@0.208.0/fs/ensure_link.ts": "c15e69c48556d78aae31b83e0c0ece04b7b8bc0951412f5b759aceb6fde7f0ac", + "https://deno.land/std@0.208.0/fs/ensure_symlink.ts": "b389c8568f0656d145ac7ece472afe710815cccbb2ebfd19da7978379ae143fe", + "https://deno.land/std@0.208.0/fs/eol.ts": "8565e1e076c5baced170236617150a7833668658e000205d896fc54084309ce1", + "https://deno.land/std@0.208.0/fs/exists.ts": "cb59a853d84871d87acab0e7936a4dac11282957f8e195102c5a7acb42546bb8", + "https://deno.land/std@0.208.0/fs/expand_glob.ts": "4f98c508fc9e40d6311d2f7fd88aaad05235cc506388c22dda315e095305811d", + "https://deno.land/std@0.208.0/fs/mod.ts": "bc3d0acd488cc7b42627044caf47d72019846d459279544e1934418955ba4898", + "https://deno.land/std@0.208.0/fs/move.ts": "b4f8f46730b40c32ea3c0bc8eb0fd0e8139249a698883c7b3756424cf19785c9", + "https://deno.land/std@0.208.0/fs/walk.ts": "c1e6b43f72a46e89b630140308bd51a4795d416a416b4cfb7cd4bd1e25946723", + "https://deno.land/std@0.208.0/path/_common/assert_path.ts": "061e4d093d4ba5aebceb2c4da3318bfe3289e868570e9d3a8e327d91c2958946", + "https://deno.land/std@0.208.0/path/_common/basename.ts": "0d978ff818f339cd3b1d09dc914881f4d15617432ae519c1b8fdc09ff8d3789a", + "https://deno.land/std@0.208.0/path/_common/common.ts": "9e4233b2eeb50f8b2ae10ecc2108f58583aea6fd3e8907827020282dc2b76143", + "https://deno.land/std@0.208.0/path/_common/constants.ts": "e49961f6f4f48039c0dfed3c3f93e963ca3d92791c9d478ac5b43183413136e0", + "https://deno.land/std@0.208.0/path/_common/dirname.ts": "2ba7fb4cc9fafb0f38028f434179579ce61d4d9e51296fad22b701c3d3cd7397", + "https://deno.land/std@0.208.0/path/_common/format.ts": "11aa62e316dfbf22c126917f5e03ea5fe2ee707386555a8f513d27ad5756cf96", + "https://deno.land/std@0.208.0/path/_common/from_file_url.ts": "ef1bf3197d2efbf0297a2bdbf3a61d804b18f2bcce45548ae112313ec5be3c22", + "https://deno.land/std@0.208.0/path/_common/glob_to_reg_exp.ts": "5c3c2b79fc2294ec803d102bd9855c451c150021f452046312819fbb6d4dc156", + "https://deno.land/std@0.208.0/path/_common/normalize.ts": "2ba7fb4cc9fafb0f38028f434179579ce61d4d9e51296fad22b701c3d3cd7397", + "https://deno.land/std@0.208.0/path/_common/normalize_string.ts": "88c472f28ae49525f9fe82de8c8816d93442d46a30d6bb5063b07ff8a89ff589", + "https://deno.land/std@0.208.0/path/_common/relative.ts": "1af19d787a2a84b8c534cc487424fe101f614982ae4851382c978ab2216186b4", + "https://deno.land/std@0.208.0/path/_common/strip_trailing_separators.ts": "7ffc7c287e97bdeeee31b155828686967f222cd73f9e5780bfe7dfb1b58c6c65", + "https://deno.land/std@0.208.0/path/_common/to_file_url.ts": "a8cdd1633bc9175b7eebd3613266d7c0b6ae0fb0cff24120b6092ac31662f9ae", + "https://deno.land/std@0.208.0/path/_interface.ts": "6471159dfbbc357e03882c2266d21ef9afdb1e4aa771b0545e90db58a0ba314b", + "https://deno.land/std@0.208.0/path/_os.ts": "30b0c2875f360c9296dbe6b7f2d528f0f9c741cecad2e97f803f5219e91b40a2", + "https://deno.land/std@0.208.0/path/basename.ts": "04bb5ef3e86bba8a35603b8f3b69537112cdd19ce64b77f2522006da2977a5f3", + "https://deno.land/std@0.208.0/path/common.ts": "f4d061c7d0b95a65c2a1a52439edec393e906b40f1caf4604c389fae7caa80f5", + "https://deno.land/std@0.208.0/path/dirname.ts": "88a0a71c21debafc4da7a4cd44fd32e899462df458fbca152390887d41c40361", + "https://deno.land/std@0.208.0/path/extname.ts": "2da4e2490f3b48b7121d19fb4c91681a5e11bd6bd99df4f6f47d7a71bb6ecdf2", + "https://deno.land/std@0.208.0/path/format.ts": "3457530cc85d1b4bab175f9ae73998b34fd456c830d01883169af0681b8894fb", + "https://deno.land/std@0.208.0/path/from_file_url.ts": "e7fa233ea1dff9641e8d566153a24d95010110185a6f418dd2e32320926043f8", + "https://deno.land/std@0.208.0/path/glob.ts": "a00a81a55c02bbe074ab21a50b6495c6f7795f54cd718c824adaa92c6c9b7419", + "https://deno.land/std@0.208.0/path/glob_to_regexp.ts": "74d7448c471e293d03f05ccb968df4365fed6aaa508506b6325a8efdc01d8271", + "https://deno.land/std@0.208.0/path/is_absolute.ts": "67232b41b860571c5b7537f4954c88d86ae2ba45e883ee37d3dec27b74909d13", + "https://deno.land/std@0.208.0/path/is_glob.ts": "567dce5c6656bdedfc6b3ee6c0833e1e4db2b8dff6e62148e94a917f289c06ad", + "https://deno.land/std@0.208.0/path/join.ts": "98d3d76c819af4a11a81d5ba2dbb319f1ce9d63fc2b615597d4bcfddd4a89a09", + "https://deno.land/std@0.208.0/path/join_globs.ts": "9b84d5103b63d3dbed4b2cf8b12477b2ad415c7d343f1488505162dc0e5f4db8", + "https://deno.land/std@0.208.0/path/mod.ts": "3defabebc98279e62b392fee7a6937adc932a8f4dcd2471441e36c15b97b00e0", + "https://deno.land/std@0.208.0/path/normalize.ts": "aa95be9a92c7bd4f9dc0ba51e942a1973e2b93d266cd74f5ca751c136d520b66", + "https://deno.land/std@0.208.0/path/normalize_glob.ts": "674baa82e1c00b6cb153bbca36e06f8e0337cb8062db6d905ab5de16076ca46b", + "https://deno.land/std@0.208.0/path/parse.ts": "d87ff0deef3fb495bc0d862278ff96da5a06acf0625ca27769fc52ac0d3d6ece", + "https://deno.land/std@0.208.0/path/posix/_util.ts": "ecf49560fedd7dd376c6156cc5565cad97c1abe9824f4417adebc7acc36c93e5", + "https://deno.land/std@0.208.0/path/posix/basename.ts": "a630aeb8fd8e27356b1823b9dedd505e30085015407caa3396332752f6b8406a", + "https://deno.land/std@0.208.0/path/posix/common.ts": "e781d395dc76f6282e3f7dd8de13194abb8b04a82d109593141abc6e95755c8b", + "https://deno.land/std@0.208.0/path/posix/dirname.ts": "f48c9c42cc670803b505478b7ef162c7cfa9d8e751b59d278b2ec59470531472", + "https://deno.land/std@0.208.0/path/posix/extname.ts": "ee7f6571a9c0a37f9218fbf510c440d1685a7c13082c348d701396cc795e0be0", + "https://deno.land/std@0.208.0/path/posix/format.ts": "b94876f77e61bfe1f147d5ccb46a920636cd3cef8be43df330f0052b03875968", + "https://deno.land/std@0.208.0/path/posix/from_file_url.ts": "b97287a83e6407ac27bdf3ab621db3fccbf1c27df0a1b1f20e1e1b5acf38a379", + "https://deno.land/std@0.208.0/path/posix/glob_to_regexp.ts": "6ed00c71fbfe0ccc35977c35444f94e82200b721905a60bd1278b1b768d68b1a", + "https://deno.land/std@0.208.0/path/posix/is_absolute.ts": "159900a3422d11069d48395568217eb7fc105ceda2683d03d9b7c0f0769e01b8", + "https://deno.land/std@0.208.0/path/posix/is_glob.ts": "ec4fbc604b9db8487f7b56ab0e759b24a971ab6a45f7b0b698bc39b8b9f9680f", + "https://deno.land/std@0.208.0/path/posix/join.ts": "0c0d84bdc344876930126640011ec1b888e6facf74153ffad9ef26813aa2a076", + "https://deno.land/std@0.208.0/path/posix/join_globs.ts": "f4838d54b1f60a34a40625a3293f6e583135348be1b2974341ac04743cb26121", + "https://deno.land/std@0.208.0/path/posix/mod.ts": "f1b08a7f64294b7de87fc37190d63b6ce5b02889af9290c9703afe01951360ae", + "https://deno.land/std@0.208.0/path/posix/normalize.ts": "11de90a94ab7148cc46e5a288f7d732aade1d616bc8c862f5560fa18ff987b4b", + "https://deno.land/std@0.208.0/path/posix/normalize_glob.ts": "10a1840c628ebbab679254d5fa1c20e59106102354fb648a1765aed72eb9f3f9", + "https://deno.land/std@0.208.0/path/posix/parse.ts": "199208f373dd93a792e9c585352bfc73a6293411bed6da6d3bc4f4ef90b04c8e", + "https://deno.land/std@0.208.0/path/posix/relative.ts": "e2f230608b0f083e6deaa06e063943e5accb3320c28aef8d87528fbb7fe6504c", + "https://deno.land/std@0.208.0/path/posix/resolve.ts": "51579d83159d5c719518c9ae50812a63959bbcb7561d79acbdb2c3682236e285", + "https://deno.land/std@0.208.0/path/posix/separator.ts": "0b6573b5f3269a3164d8edc9cefc33a02dd51003731c561008c8bb60220ebac1", + "https://deno.land/std@0.208.0/path/posix/to_file_url.ts": "08d43ea839ee75e9b8b1538376cfe95911070a655cd312bc9a00f88ef14967b6", + "https://deno.land/std@0.208.0/path/posix/to_namespaced_path.ts": "c9228a0e74fd37e76622cd7b142b8416663a9b87db643302fa0926b5a5c83bdc", + "https://deno.land/std@0.208.0/path/relative.ts": "23d45ede8b7ac464a8299663a43488aad6b561414e7cbbe4790775590db6349c", + "https://deno.land/std@0.208.0/path/resolve.ts": "5b184efc87155a0af9fa305ff68a109e28de9aee81fc3e77cd01380f19daf867", + "https://deno.land/std@0.208.0/path/separator.ts": "40a3e9a4ad10bef23bc2cd6c610291b6c502a06237c2c4cd034a15ca78dedc1f", + "https://deno.land/std@0.208.0/path/to_file_url.ts": "edaafa089e0bce386e1b2d47afe7c72e379ff93b28a5829a5885e4b6c626d864", + "https://deno.land/std@0.208.0/path/to_namespaced_path.ts": "cf8734848aac3c7527d1689d2adf82132b1618eff3cc523a775068847416b22a", + "https://deno.land/std@0.208.0/path/windows/_util.ts": "f32b9444554c8863b9b4814025c700492a2b57ff2369d015360970a1b1099d54", + "https://deno.land/std@0.208.0/path/windows/basename.ts": "8a9dbf7353d50afbc5b221af36c02a72c2d1b2b5b9f7c65bf6a5a2a0baf88ad3", + "https://deno.land/std@0.208.0/path/windows/common.ts": "e781d395dc76f6282e3f7dd8de13194abb8b04a82d109593141abc6e95755c8b", + "https://deno.land/std@0.208.0/path/windows/dirname.ts": "5c2aa541384bf0bd9aca821275d2a8690e8238fa846198ef5c7515ce31a01a94", + "https://deno.land/std@0.208.0/path/windows/extname.ts": "07f4fa1b40d06a827446b3e3bcc8d619c5546b079b8ed0c77040bbef716c7614", + "https://deno.land/std@0.208.0/path/windows/format.ts": "343019130d78f172a5c49fdc7e64686a7faf41553268961e7b6c92a6d6548edf", + "https://deno.land/std@0.208.0/path/windows/from_file_url.ts": "d53335c12b0725893d768be3ac6bf0112cc5b639d2deb0171b35988493b46199", + "https://deno.land/std@0.208.0/path/windows/glob_to_regexp.ts": "290755e18ec6c1a4f4d711c3390537358e8e3179581e66261a0cf348b1a13395", + "https://deno.land/std@0.208.0/path/windows/is_absolute.ts": "245b56b5f355ede8664bd7f080c910a97e2169972d23075554ae14d73722c53c", + "https://deno.land/std@0.208.0/path/windows/is_glob.ts": "ec4fbc604b9db8487f7b56ab0e759b24a971ab6a45f7b0b698bc39b8b9f9680f", + "https://deno.land/std@0.208.0/path/windows/join.ts": "e6600bf88edeeef4e2276e155b8de1d5dec0435fd526ba2dc4d37986b2882f16", + "https://deno.land/std@0.208.0/path/windows/join_globs.ts": "f4838d54b1f60a34a40625a3293f6e583135348be1b2974341ac04743cb26121", + "https://deno.land/std@0.208.0/path/windows/mod.ts": "d7040f461465c2c21c1c68fc988ef0bdddd499912138cde3abf6ad60c7fb3814", + "https://deno.land/std@0.208.0/path/windows/normalize.ts": "9deebbf40c81ef540b7b945d4ccd7a6a2c5a5992f791e6d3377043031e164e69", + "https://deno.land/std@0.208.0/path/windows/normalize_glob.ts": "344ff5ed45430495b9a3d695567291e50e00b1b3b04ea56712a2acf07ab5c128", + "https://deno.land/std@0.208.0/path/windows/parse.ts": "120faf778fe1f22056f33ded069b68e12447668fcfa19540c0129561428d3ae5", + "https://deno.land/std@0.208.0/path/windows/relative.ts": "026855cd2c36c8f28f1df3c6fbd8f2449a2aa21f48797a74700c5d872b86d649", + "https://deno.land/std@0.208.0/path/windows/resolve.ts": "5ff441ab18a2346abadf778121128ee71bda4d0898513d4639a6ca04edca366b", + "https://deno.land/std@0.208.0/path/windows/separator.ts": "ae21f27015f10510ed1ac4a0ba9c4c9c967cbdd9d9e776a3e4967553c397bd5d", + "https://deno.land/std@0.208.0/path/windows/to_file_url.ts": "8e9ea9e1ff364aa06fa72999204229952d0a279dbb876b7b838b2b2fea55cce3", + "https://deno.land/std@0.208.0/path/windows/to_namespaced_path.ts": "e0f4d4a5e77f28a5708c1a33ff24360f35637ba6d8f103d19661255ef7bfd50d", + "https://deno.land/std@0.208.0/yaml/_dumper/dumper.ts": "717403d0e700de783f2ef5c906b3d7245383e1509fc050e7ff5d4a53a03dbf40", + "https://deno.land/std@0.208.0/yaml/_dumper/dumper_state.ts": "f0d0673ceea288334061ca34b63954c2bb5feb5bf6de5e4cfe9a942cdf6e5efe", + "https://deno.land/std@0.208.0/yaml/_error.ts": "b59e2c76ce5a47b1b9fa0ff9f96c1dd92ea1e1b17ce4347ece5944a95c3c1a84", + "https://deno.land/std@0.208.0/yaml/_loader/loader.ts": "63ec7f0a265dbbabc54b25a4beefff7650e205160a2d75c7d8f8363b5f84851a", + "https://deno.land/std@0.208.0/yaml/_loader/loader_state.ts": "0841870b467169269d7c2dfa75cd288c319bc06f65edd9e42c29e5fced91c7a4", + "https://deno.land/std@0.208.0/yaml/_mark.ts": "dcd8585dee585e024475e9f3fe27d29740670fb64ebb970388094cad0fc11d5d", + "https://deno.land/std@0.208.0/yaml/_state.ts": "ef03d55ec235d48dcfbecc0ab3ade90bfae69a61094846e08003421c2cf5cfc6", + "https://deno.land/std@0.208.0/yaml/_type/binary.ts": "24d49614463a7339a8a16d894919c2ec18a10588ae360ec352093b60e2cc8b0d", + "https://deno.land/std@0.208.0/yaml/_type/bool.ts": "5bfa75da84343d45347b521ba4e5aeace9fe6f53447405290d53315a3fc20e66", + "https://deno.land/std@0.208.0/yaml/_type/float.ts": "056bd3cb9c5586238b20517511014fb24b0e36f98f9f6073e12da308b6b9808a", + "https://deno.land/std@0.208.0/yaml/_type/function.ts": "ff574fe84a750695302864e1c31b93f12d14ada4bde79a5f93197fc33ad17471", + "https://deno.land/std@0.208.0/yaml/_type/int.ts": "563ad074f0fa7aecf6b6c3d84135bcc95a8269dcc15de878de20ce868fd773fa", + "https://deno.land/std@0.208.0/yaml/_type/map.ts": "7b105e4ab03a361c61e7e335a0baf4d40f06460b13920e5af3fb2783a1464000", + "https://deno.land/std@0.208.0/yaml/_type/merge.ts": "8192bf3e4d637f32567917f48bb276043da9cf729cf594e5ec191f7cd229337e", + "https://deno.land/std@0.208.0/yaml/_type/mod.ts": "060e2b3d38725094b77ea3a3f05fc7e671fced8e67ca18e525be98c4aa8f4bbb", + "https://deno.land/std@0.208.0/yaml/_type/nil.ts": "606e8f0c44d73117c81abec822f89ef81e40f712258c74f186baa1af659b8887", + "https://deno.land/std@0.208.0/yaml/_type/omap.ts": "cfe59a294726f5cea705c39a61fd2b08199cf48f4ccd6b040cb550ec0f38d0a1", + "https://deno.land/std@0.208.0/yaml/_type/pairs.ts": "0032fdfe57558d21696a4f8cf5b5cfd1f698743177080affc18629685c905666", + "https://deno.land/std@0.208.0/yaml/_type/regexp.ts": "1ce118de15b2da43b4bd8e4395f42d448b731acf3bdaf7c888f40789f9a95f8b", + "https://deno.land/std@0.208.0/yaml/_type/seq.ts": "95333abeec8a7e4d967b8c8328b269e342a4bbdd2585395549b9c4f58c8533a2", + "https://deno.land/std@0.208.0/yaml/_type/set.ts": "f28ba44e632ef2a6eb580486fd47a460445eeddbdf1dbc739c3e62486f566092", + "https://deno.land/std@0.208.0/yaml/_type/str.ts": "a67a3c6e429d95041399e964015511779b1130ea5889fa257c48457bd3446e31", + "https://deno.land/std@0.208.0/yaml/_type/timestamp.ts": "706ea80a76a73e48efaeb400ace087da1f927647b53ad6f754f4e06d51af087f", + "https://deno.land/std@0.208.0/yaml/_type/undefined.ts": "94a316ca450597ccbc6750cbd79097ad0d5f3a019797eed3c841a040c29540ba", + "https://deno.land/std@0.208.0/yaml/_utils.ts": "26b311f0d42a7ce025060bd6320a68b50e52fd24a839581eb31734cd48e20393", + "https://deno.land/std@0.208.0/yaml/mod.ts": "28ecda6652f3e7a7735ee29c247bfbd32a2e2fc5724068e9fd173ec4e59f66f7", + "https://deno.land/std@0.208.0/yaml/parse.ts": "1fbbda572bf3fff578b6482c0d8b85097a38de3176bf3ab2ca70c25fb0c960ef", + "https://deno.land/std@0.208.0/yaml/schema.ts": "96908b78dc50c340074b93fc1598d5e7e2fe59103f89ff81e5a49b2dedf77a67", + "https://deno.land/std@0.208.0/yaml/schema/core.ts": "fa406f18ceedc87a50e28bb90ec7a4c09eebb337f94ef17468349794fa828639", + "https://deno.land/std@0.208.0/yaml/schema/default.ts": "0047e80ae8a4a93293bc4c557ae8a546aabd46bb7165b9d9b940d57b4d88bde9", + "https://deno.land/std@0.208.0/yaml/schema/extended.ts": "0784416bf062d20a1626b53c03380e265b3e39b9409afb9f4cb7d659fd71e60d", + "https://deno.land/std@0.208.0/yaml/schema/failsafe.ts": "d219ab5febc43f770917d8ec37735a4b1ad671149846cbdcade767832b42b92b", + "https://deno.land/std@0.208.0/yaml/schema/json.ts": "5f41dd7c2f1ad545ef6238633ce9ee3d444dfc5a18101e1768bd5504bf90e5e5", + "https://deno.land/std@0.208.0/yaml/schema/mod.ts": "4472e827bab5025e92bc2eb2eeefa70ecbefc64b2799b765c69af84822efef32", + "https://deno.land/std@0.208.0/yaml/stringify.ts": "fffc09c65c68d3d63f8159e8cbaa3f489bc20a8e55b4fbb61a8c2e9f914d1d02", + "https://deno.land/std@0.208.0/yaml/type.ts": "65553da3da3c029b6589c6e4903f0afbea6768be8fca61580711457151f2b30f", + "https://deno.land/std@0.211.0/assert/assert.ts": "bec068b2fccdd434c138a555b19a2c2393b71dfaada02b7d568a01541e67cdc5", + "https://deno.land/std@0.211.0/assert/assertion_error.ts": "9f689a101ee586c4ce92f52fa7ddd362e86434ffdf1f848e45987dc7689976b8", + "https://deno.land/std@0.211.0/encoding/_util.ts": "beacef316c1255da9bc8e95afb1fa56ed69baef919c88dc06ae6cb7a6103d376", + "https://deno.land/std@0.211.0/encoding/base32.ts": "44d06f3e6f6f8fa4971a4538ff9ea23ba116c14eae8c1fbb348801d10c911214", + "https://deno.land/std@0.211.0/fs/_create_walk_entry.ts": "5d9d2aaec05bcf09a06748b1684224d33eba7a4de24cf4cf5599991ca6b5b412", + "https://deno.land/std@0.211.0/fs/_get_file_info_type.ts": "da7bec18a7661dba360a1db475b826b18977582ce6fc9b25f3d4ee0403fe8cbd", + "https://deno.land/std@0.211.0/fs/_is_same_path.ts": "709c95868345fea051c58b9e96af95cff94e6ae98dfcff2b66dee0c212c4221f", + "https://deno.land/std@0.211.0/fs/_is_subdir.ts": "9a0a8664420f21ad92030c4ee266ec6251797d17b4a90cce5c2f381c2e1d7626", + "https://deno.land/std@0.211.0/fs/_to_path_string.ts": "29bfc9c6c112254961d75cbf6ba814d6de5349767818eb93090cecfa9665591e", + "https://deno.land/std@0.211.0/fs/copy.ts": "dc0f68c4b6c3b090bfdb909387e309f6169b746bd713927c9507c9ef545d71f6", + "https://deno.land/std@0.211.0/fs/empty_dir.ts": "4f01e6d56e2aa8d90ad60f20bc25601f516b00f6c3044cdf6863a058791d91aa", + "https://deno.land/std@0.211.0/fs/ensure_dir.ts": "139a6cf85c145003db9a2b470862ba153422e7deab2fe3c5b04c5b1cfb73d738", + "https://deno.land/std@0.211.0/fs/ensure_file.ts": "ac5cfde94786b0284d2c8e9f7f9425269bea1b2140612b4aea1f20b508870f59", + "https://deno.land/std@0.211.0/fs/ensure_link.ts": "d42af2edefeaa9817873ec6e46dc5d209ac4d744f8c69c5ecc2dffade78465b6", + "https://deno.land/std@0.211.0/fs/ensure_symlink.ts": "aee3f1655700f60090b4a3037f5b6c07ab37c36807cccad746ce89987719e6d2", + "https://deno.land/std@0.211.0/fs/eol.ts": "c9807291f78361d49fd986a9be04654610c615c5e2ec63d748976197d30ff206", + "https://deno.land/std@0.211.0/fs/exists.ts": "d2757ef764eaf5c6c5af7228e8447db2de42ab084a2dae540097f905723d83f5", + "https://deno.land/std@0.211.0/fs/expand_glob.ts": "25ef968ebbf6751f658e2ed595746576c5ddf7723146672538396d27bbd23894", + "https://deno.land/std@0.211.0/fs/mod.ts": "107f5afa4424c2d3ce2f7e9266173198da30302c69af662c720115fe504dc5ee", + "https://deno.land/std@0.211.0/fs/move.ts": "39e0d7ccb88a566d20b949712020e766b15ef1ec19159573d11f949bd677909c", + "https://deno.land/std@0.211.0/fs/walk.ts": "f04cc83ad3b27b5a5d078c831a01c7406069474bf280d5db015d937149a60128", + "https://deno.land/std@0.211.0/json/common.ts": "867f6b42545468bfcf86d85b9682165c6d2ae2dcc3816f6c817e705d2d9b4e39", + "https://deno.land/std@0.211.0/jsonc/mod.ts": "82722888823e1af5a8f7918bf810ea581f68081064d529218533acad6cb7c2bc", + "https://deno.land/std@0.211.0/jsonc/parse.ts": "1ec8ab68fdbc4dcc25b48a8436e07421eefd0327cb7e9a79ee1bf08893a75276", + "https://deno.land/std@0.211.0/path/_common/assert_path.ts": "2ca275f36ac1788b2acb60fb2b79cb06027198bc2ba6fb7e163efaedde98c297", + "https://deno.land/std@0.211.0/path/_common/basename.ts": "569744855bc8445f3a56087fd2aed56bdad39da971a8d92b138c9913aecc5fa2", + "https://deno.land/std@0.211.0/path/_common/common.ts": "6157c7ec1f4db2b4a9a187efd6ce76dcaf1e61cfd49f87e40d4ea102818df031", + "https://deno.land/std@0.211.0/path/_common/constants.ts": "dc5f8057159f4b48cd304eb3027e42f1148cf4df1fb4240774d3492b5d12ac0c", + "https://deno.land/std@0.211.0/path/_common/dirname.ts": "684df4aa71a04bbcc346c692c8485594fc8a90b9408dfbc26ff32cf3e0c98cc8", + "https://deno.land/std@0.211.0/path/_common/format.ts": "92500e91ea5de21c97f5fe91e178bae62af524b72d5fcd246d6d60ae4bcada8b", + "https://deno.land/std@0.211.0/path/_common/from_file_url.ts": "d672bdeebc11bf80e99bf266f886c70963107bdd31134c4e249eef51133ceccf", + "https://deno.land/std@0.211.0/path/_common/glob_to_reg_exp.ts": "2007aa87bed6eb2c8ae8381adcc3125027543d9ec347713c1ad2c68427330770", + "https://deno.land/std@0.211.0/path/_common/normalize.ts": "684df4aa71a04bbcc346c692c8485594fc8a90b9408dfbc26ff32cf3e0c98cc8", + "https://deno.land/std@0.211.0/path/_common/normalize_string.ts": "dfdf657a1b1a7db7999f7c575ee7e6b0551d9c20f19486c6c3f5ff428384c965", + "https://deno.land/std@0.211.0/path/_common/relative.ts": "faa2753d9b32320ed4ada0733261e3357c186e5705678d9dd08b97527deae607", + "https://deno.land/std@0.211.0/path/_common/strip_trailing_separators.ts": "7024a93447efcdcfeaa9339a98fa63ef9d53de363f1fbe9858970f1bba02655a", + "https://deno.land/std@0.211.0/path/_common/to_file_url.ts": "7f76adbc83ece1bba173e6e98a27c647712cab773d3f8cbe0398b74afc817883", + "https://deno.land/std@0.211.0/path/_interface.ts": "a1419fcf45c0ceb8acdccc94394e3e94f99e18cfd32d509aab514c8841799600", + "https://deno.land/std@0.211.0/path/_os.ts": "8fb9b90fb6b753bd8c77cfd8a33c2ff6c5f5bc185f50de8ca4ac6a05710b2c15", + "https://deno.land/std@0.211.0/path/basename.ts": "5d341aadb7ada266e2280561692c165771d071c98746fcb66da928870cd47668", + "https://deno.land/std@0.211.0/path/common.ts": "973e019d3cfa6a134a13f1fda3f7efbaf400a64365d7a7b96f66afe373a09dc5", + "https://deno.land/std@0.211.0/path/dirname.ts": "85bd955bf31d62c9aafdd7ff561c4b5fb587d11a9a5a45e2b01aedffa4238a7c", + "https://deno.land/std@0.211.0/path/extname.ts": "593303db8ae8c865cbd9ceec6e55d4b9ac5410c1e276bfd3131916591b954441", + "https://deno.land/std@0.211.0/path/format.ts": "98fad25f1af7b96a48efb5b67378fcc8ed77be895df8b9c733b86411632162af", + "https://deno.land/std@0.211.0/path/from_file_url.ts": "911833ae4fd10a1c84f6271f36151ab785955849117dc48c6e43b929504ee069", + "https://deno.land/std@0.211.0/path/glob.ts": "04510962905d4b1513b44da9cb195914e0fa46c24359f6feaca20848d797dcb0", + "https://deno.land/std@0.211.0/path/glob_to_regexp.ts": "83c5fd36a8c86f5e72df9d0f45317f9546afa2ce39acaafe079d43a865aced08", + "https://deno.land/std@0.211.0/path/is_absolute.ts": "4791afc8bfd0c87f0526eaa616b0d16e7b3ab6a65b62942e50eac68de4ef67d7", + "https://deno.land/std@0.211.0/path/is_glob.ts": "a65f6195d3058c3050ab905705891b412ff942a292bcbaa1a807a74439a14141", + "https://deno.land/std@0.211.0/path/join.ts": "ae2ec5ca44c7e84a235fd532e4a0116bfb1f2368b394db1c4fb75e3c0f26a33a", + "https://deno.land/std@0.211.0/path/join_globs.ts": "e9589869a33dc3982101898ee50903db918ca00ad2614dbe3934d597d7b1fbea", + "https://deno.land/std@0.211.0/path/mod.ts": "8e1ffe983557e9637184ccb84bd6b0447e319f4a28bfad7f3f41ee050579e5e6", + "https://deno.land/std@0.211.0/path/normalize.ts": "4155743ccceeed319b350c1e62e931600272fad8ad00c417b91df093867a8352", + "https://deno.land/std@0.211.0/path/normalize_glob.ts": "98ee8268fad271193603271c203ae973280b5abfbdd2cbca1053fd2af71869ca", + "https://deno.land/std@0.211.0/path/parse.ts": "65e8e285f1a63b714e19ef24b68f56e76934c3df0b6e65fd440d3991f4f8aefb", + "https://deno.land/std@0.211.0/path/posix/_util.ts": "1e3937da30f080bfc99fe45d7ed23c47dd8585c5e473b2d771380d3a6937cf9d", + "https://deno.land/std@0.211.0/path/posix/basename.ts": "39ee27a29f1f35935d3603ccf01d53f3d6e0c5d4d0f84421e65bd1afeff42843", + "https://deno.land/std@0.211.0/path/posix/common.ts": "809cc86e79db8171b9a97ac397d56b9588c25a8f3062f483c8d651a2b6739daa", + "https://deno.land/std@0.211.0/path/posix/dirname.ts": "6535d2bdd566118963537b9dda8867ba9e2a361015540dc91f5afbb65c0cce8b", + "https://deno.land/std@0.211.0/path/posix/extname.ts": "8d36ae0082063c5e1191639699e6f77d3acf501600a3d87b74943f0ae5327427", + "https://deno.land/std@0.211.0/path/posix/format.ts": "185e9ee2091a42dd39e2a3b8e4925370ee8407572cee1ae52838aed96310c5c1", + "https://deno.land/std@0.211.0/path/posix/from_file_url.ts": "951aee3a2c46fd0ed488899d024c6352b59154c70552e90885ed0c2ab699bc40", + "https://deno.land/std@0.211.0/path/posix/glob_to_regexp.ts": "54d3ff40f309e3732ab6e5b19d7111d2d415248bcd35b67a99defcbc1972e697", + "https://deno.land/std@0.211.0/path/posix/is_absolute.ts": "cebe561ad0ae294f0ce0365a1879dcfca8abd872821519b4fcc8d8967f888ede", + "https://deno.land/std@0.211.0/path/posix/is_glob.ts": "8a8b08c08bf731acf2c1232218f1f45a11131bc01de81e5f803450a5914434b9", + "https://deno.land/std@0.211.0/path/posix/join.ts": "aef88d5fa3650f7516730865dbb951594d1a955b785e2450dbee93b8e32694f3", + "https://deno.land/std@0.211.0/path/posix/join_globs.ts": "35ddd5f321d79e1fc72d2ec9a8d8863f0bb1431125e57bb2661799278d4ee9cd", + "https://deno.land/std@0.211.0/path/posix/mod.ts": "9dfff9f3618ba6990eb8495dadef13871e5756419b25079b6b905a4ebf790926", + "https://deno.land/std@0.211.0/path/posix/normalize.ts": "baeb49816a8299f90a0237d214cef46f00ba3e95c0d2ceb74205a6a584b58a91", + "https://deno.land/std@0.211.0/path/posix/normalize_glob.ts": "0f01bcfb0791144f0e901fd2cc706432baf84828c393f3c25c53583f03d0c0b7", + "https://deno.land/std@0.211.0/path/posix/parse.ts": "d5bac4eb21262ab168eead7e2196cb862940c84cee572eafedd12a0d34adc8fb", + "https://deno.land/std@0.211.0/path/posix/relative.ts": "3907d6eda41f0ff723d336125a1ad4349112cd4d48f693859980314d5b9da31c", + "https://deno.land/std@0.211.0/path/posix/resolve.ts": "bac20d9921beebbbb2b73706683b518b1d0c1b1da514140cee409e90d6b2913a", + "https://deno.land/std@0.211.0/path/posix/separator.ts": "6530f253a33d92d8f8a1d1d7fa7fad2992c739ad9886dde72e4e78793f1cfd49", + "https://deno.land/std@0.211.0/path/posix/to_file_url.ts": "7aa752ba66a35049e0e4a4be5a0a31ac6b645257d2e031142abb1854de250aaf", + "https://deno.land/std@0.211.0/path/posix/to_namespaced_path.ts": "28b216b3c76f892a4dca9734ff1cc0045d135532bfd9c435ae4858bfa5a2ebf0", + "https://deno.land/std@0.211.0/path/relative.ts": "ab739d727180ed8727e34ed71d976912461d98e2b76de3d3de834c1066667add", + "https://deno.land/std@0.211.0/path/resolve.ts": "a6f977bdb4272e79d8d0ed4333e3d71367cc3926acf15ac271f1d059c8494d8d", + "https://deno.land/std@0.211.0/path/separator.ts": "2b5a590d4f1942e70650ee7421d161c24ec7d3b94b49981e4138ae07397fb2d2", + "https://deno.land/std@0.211.0/path/to_file_url.ts": "88f049b769bce411e2d2db5bd9e6fd9a185a5fbd6b9f5ad8f52bef517c4ece1b", + "https://deno.land/std@0.211.0/path/to_namespaced_path.ts": "b706a4103b104cfadc09600a5f838c2ba94dbcdb642344557122dda444526e40", + "https://deno.land/std@0.211.0/path/windows/_util.ts": "d5f47363e5293fced22c984550d5e70e98e266cc3f31769e1710511803d04808", + "https://deno.land/std@0.211.0/path/windows/basename.ts": "e2dbf31d1d6385bfab1ce38c333aa290b6d7ae9e0ecb8234a654e583cf22f8fe", + "https://deno.land/std@0.211.0/path/windows/common.ts": "809cc86e79db8171b9a97ac397d56b9588c25a8f3062f483c8d651a2b6739daa", + "https://deno.land/std@0.211.0/path/windows/dirname.ts": "33e421be5a5558a1346a48e74c330b8e560be7424ed7684ea03c12c21b627bc9", + "https://deno.land/std@0.211.0/path/windows/extname.ts": "165a61b00d781257fda1e9606a48c78b06815385e7d703232548dbfc95346bef", + "https://deno.land/std@0.211.0/path/windows/format.ts": "bbb5ecf379305b472b1082cd2fdc010e44a0020030414974d6029be9ad52aeb6", + "https://deno.land/std@0.211.0/path/windows/from_file_url.ts": "ced2d587b6dff18f963f269d745c4a599cf82b0c4007356bd957cb4cb52efc01", + "https://deno.land/std@0.211.0/path/windows/glob_to_regexp.ts": "6dcd1242bd8907aa9660cbdd7c93446e6927b201112b0cba37ca5d80f81be51b", + "https://deno.land/std@0.211.0/path/windows/is_absolute.ts": "4a8f6853f8598cf91a835f41abed42112cebab09478b072e4beb00ec81f8ca8a", + "https://deno.land/std@0.211.0/path/windows/is_glob.ts": "8a8b08c08bf731acf2c1232218f1f45a11131bc01de81e5f803450a5914434b9", + "https://deno.land/std@0.211.0/path/windows/join.ts": "e0b3356615c1a75c56ebb6a7311157911659e11fd533d80d724800126b761ac3", + "https://deno.land/std@0.211.0/path/windows/join_globs.ts": "35ddd5f321d79e1fc72d2ec9a8d8863f0bb1431125e57bb2661799278d4ee9cd", + "https://deno.land/std@0.211.0/path/windows/mod.ts": "e739f7e783b69fb7956bed055e117201ccb071a7917c09f87c5c8c2b54369d38", + "https://deno.land/std@0.211.0/path/windows/normalize.ts": "78126170ab917f0ca355a9af9e65ad6bfa5be14d574c5fb09bb1920f52577780", + "https://deno.land/std@0.211.0/path/windows/normalize_glob.ts": "49c634af33a7c6bc738885c4b34633278b7ab47bd47bf11281b2190970b823e2", + "https://deno.land/std@0.211.0/path/windows/parse.ts": "b9239edd892a06a06625c1b58425e199f018ce5649ace024d144495c984da734", + "https://deno.land/std@0.211.0/path/windows/relative.ts": "3e1abc7977ee6cc0db2730d1f9cb38be87b0ce4806759d271a70e4997fc638d7", + "https://deno.land/std@0.211.0/path/windows/resolve.ts": "75b2e3e1238d840782cee3d8864d82bfaa593c7af8b22f19c6422cf82f330ab3", + "https://deno.land/std@0.211.0/path/windows/separator.ts": "2bbcc551f64810fb43252185bd1d33d66e0477d74bd52f03b89f5dc21a3dd486", + "https://deno.land/std@0.211.0/path/windows/to_file_url.ts": "1cd63fd35ec8d1370feaa4752eccc4cc05ea5362a878be8dc7db733650995484", + "https://deno.land/std@0.211.0/path/windows/to_namespaced_path.ts": "4ffa4fb6fae321448d5fe810b3ca741d84df4d7897e61ee29be961a6aac89a4c", "https://deno.land/std@0.214.0/assert/assert.ts": "bec068b2fccdd434c138a555b19a2c2393b71dfaada02b7d568a01541e67cdc5", "https://deno.land/std@0.214.0/assert/assertion_error.ts": "9f689a101ee586c4ce92f52fa7ddd362e86434ffdf1f848e45987dc7689976b8", "https://deno.land/std@0.214.0/async/delay.ts": "8e1d18fe8b28ff95885e2bc54eccec1713f57f756053576d8228e6ca110793ad", @@ -295,6 +856,18 @@ "https://deno.land/std@0.217.0/assert/unimplemented.ts": "47ca67d1c6dc53abd0bd729b71a31e0825fc452dbcd4fde4ca06789d5644e7fd", "https://deno.land/std@0.217.0/assert/unreachable.ts": "38cfecb95d8b06906022d2f9474794fca4161a994f83354fd079cac9032b5145", "https://deno.land/std@0.217.0/fmt/colors.ts": "d239d84620b921ea520125d778947881f62c50e78deef2657073840b8af9559a", + "https://deno.land/x/deno_cache@0.6.2/auth_tokens.ts": "5d1d56474c54a9d152e44d43ea17c2e6a398dd1e9682c69811a313567c01ee1e", + "https://deno.land/x/deno_cache@0.6.2/cache.ts": "58b53c128b742757efcad10af9a3871f23b4e200674cb5b0ddf61164fb9b2fe7", + "https://deno.land/x/deno_cache@0.6.2/deno_dir.ts": "1ea355b8ba11c630d076b222b197cfc937dd81e5a4a260938997da99e8ff93a0", + "https://deno.land/x/deno_cache@0.6.2/deps.ts": "12cca94516cf2d3ed42fccd4b721ecd8060679253f077d83057511045b0081aa", + "https://deno.land/x/deno_cache@0.6.2/dirs.ts": "009c6f54e0b610914d6ce9f72f6f6ccfffd2d47a79a19061e0a9eb4253836069", + "https://deno.land/x/deno_cache@0.6.2/disk_cache.ts": "66a1e604a8d564b6dd0500326cac33d08b561d331036bf7272def80f2f7952aa", + "https://deno.land/x/deno_cache@0.6.2/file_fetcher.ts": "4f3e4a2c78a5ca1e4812099e5083f815a8525ab20d389b560b3517f6b1161dd6", + "https://deno.land/x/deno_cache@0.6.2/http_cache.ts": "407135eaf2802809ed373c230d57da7ef8dff923c4abf205410b9b99886491fd", + "https://deno.land/x/deno_cache@0.6.2/lib/deno_cache_dir.generated.js": "59f8defac32e8ebf2a30f7bc77e9d88f0e60098463fb1b75e00b9791a4bbd733", + "https://deno.land/x/deno_cache@0.6.2/lib/snippets/deno_cache_dir-a2aecaa9536c9402/fs.js": "cbe3a976ed63c72c7cb34ef845c27013033a3b11f9d8d3e2c4aa5dda2c0c7af6", + "https://deno.land/x/deno_cache@0.6.2/mod.ts": "b4004287e1c6123d7f07fe9b5b3e94ce6d990c4102949a89c527c68b19627867", + "https://deno.land/x/deno_cache@0.6.2/util.ts": "f3f5a0cfc60051f09162942fb0ee87a0e27b11a12aec4c22076e3006be4cc1e2", "https://deno.land/x/deno_faker@v1.0.3/lib/address.ts": "d461912c0a8c14fb6d277016e4e2e0098fcba4dee0fe77f5de248c7fc2aaa601", "https://deno.land/x/deno_faker@v1.0.3/lib/commerce.ts": "797e10dd360b1f63b2d877b368db5bedabb90c07d5ccb4cc63fded644648c8b5", "https://deno.land/x/deno_faker@v1.0.3/lib/company.ts": "c241dd2ccfcee7a400b94badcdb5ee9657784dd47a86417b54952913023cbd11", @@ -1533,6 +2106,25 @@ "https://deno.land/x/deno_faker@v1.0.3/vendor/mersenne.ts": "8a61935ca2f91b925d9e8cf262eaf8b3277d091f791c8b4f93f995359db1a9a7", "https://deno.land/x/deno_faker@v1.0.3/vendor/unique.ts": "b8bb044d4caf0bb1a868bd26839bb5822e2013e8385f119db7029631e5a53e0b", "https://deno.land/x/deno_faker@v1.0.3/vendor/user-agent.ts": "b95c7bda4ad37ba25b60c4431227361eabba70db14456abb69227d6536ea93fb", + "https://deno.land/x/deno_graph@0.53.0/deno_graph_wasm.generated.js": "2cbaec012743f138172c0aff377c589ca1dd25331b77acada8ea4aafd6ec8bb4", + "https://deno.land/x/deno_graph@0.53.0/loader.ts": "a2e757383908f4a51659fe1b1203386887ebb17756bac930a64856d613d8d57d", + "https://deno.land/x/deno_graph@0.53.0/media_type.ts": "a89a1b38d07c160e896de9ceb99285ba8391940140558304171066b5c3ef7609", + "https://deno.land/x/deno_graph@0.53.0/mod.ts": "e4bdddf09d8332394ac4b2e7084f7f4fbbbf09dff344cac9bd60f5e20b4f12e0", + "https://deno.land/x/denoflate@1.2.1/mod.ts": "f5628e44b80b3d80ed525afa2ba0f12408e3849db817d47a883b801f9ce69dd6", + "https://deno.land/x/denoflate@1.2.1/pkg/denoflate.js": "b9f9ad9457d3f12f28b1fb35c555f57443427f74decb403113d67364e4f2caf4", + "https://deno.land/x/denoflate@1.2.1/pkg/denoflate_bg.wasm.js": "d581956245407a2115a3d7e8d85a9641c032940a8e810acbd59ca86afd34d44d", + "https://deno.land/x/dir@1.5.1/data_local_dir/mod.ts": "91eb1c4bfadfbeda30171007bac6d85aadacd43224a5ed721bbe56bc64e9eb66", + "https://deno.land/x/esbuild@v0.20.1/mod.js": "d50e500b53ce67e31116beba3916b0f9275c0e1cc20bc5cadc0fc1b7a3b06fd9", + "https://deno.land/x/esbuild_deno_loader@0.8.5/deps.ts": "9bc65a79ff37923c11d7470823aaa0ba9cd2e044046619a81feae074d3a2969d", + "https://deno.land/x/esbuild_deno_loader@0.8.5/mod.ts": "28524460bef46d487221b01ade6ed913d2e127de7eeee025ab75b34b491283da", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/deno.ts": "b0af3e430c068f18c6fa48c2083a1b4354b6c303e16fb37855e02fcafb95f36d", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/loader_native.ts": "17fe4702b285597d1ffc191cc85063bbb003956c2a67c18bac5c0c23e153561e", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/loader_portable.ts": "d999f452ef3d8ec2dd3c8443f542adf57efc8a2cd59b29cc41f5b3d7dff512e5", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/plugin_deno_loader.ts": "bbd8a462033508a92d19066034711f9e1fcb46d9e39d70050fc2fc9a07e521bc", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/plugin_deno_resolver.ts": "4a2d495a47ea513ba2692c33ac26b6ac8cbe765dad1c77b76fcc62aa8bfc97f5", + "https://deno.land/x/esbuild_deno_loader@0.8.5/src/shared.ts": "33052684aeb542ebd24da372816bbbf885cd090a7ab0fde7770801f7f5b49572", + "https://deno.land/x/importmap@0.2.1/_util.ts": "ada9a9618b537e6c0316c048a898352396c882b9f2de38aba18fd3f2950ede89", + "https://deno.land/x/importmap@0.2.1/mod.ts": "ae3d1cd7eabd18c01a4960d57db471126b020f23b37ef14e1359bbb949227ade", "https://deno.land/x/postgres@v0.17.2/client.ts": "315a27543a86c5703555d9be06f291ecd51bf403ee25a5bdc7d1c3387e8116ad", "https://deno.land/x/postgres@v0.17.2/client/error.ts": "ceecfa85128738885b9e190bcddf048bf790733905f4af521edfbac14a3f74cb", "https://deno.land/x/postgres@v0.17.2/connection/auth.ts": "db15c1659742ef4d2791b32834950278dc7a40cb931f8e434e6569298e58df51", @@ -1555,6 +2147,8 @@ "https://deno.land/x/postgres@v0.17.2/query/types.ts": "540f6f973d493d63f2c0059a09f3368071f57931bba68bea408a635a3e0565d6", "https://deno.land/x/postgres@v0.17.2/utils/deferred.ts": "5420531adb6c3ea29ca8aac57b9b59bd3e4b9a938a4996bbd0947a858f611080", "https://deno.land/x/postgres@v0.17.2/utils/utils.ts": "ca47193ea03ff5b585e487a06f106d367e509263a960b787197ce0c03113a738", + "https://deno.land/x/wasmbuild@0.14.1/cache.ts": "89eea5f3ce6035a1164b3e655c95f21300498920575ade23161421f5b01967f4", + "https://deno.land/x/wasmbuild@0.14.1/loader.ts": "d98d195a715f823151cbc8baa3f32127337628379a02d9eb2a3c5902dbccfc02", "https://esm.sh/ajv-formats@2.1.1": "575b3830618970ddc3aba96310bf4df7358bb37fcea101f58b36897ff3ac2ea7", "https://esm.sh/ajv@8.12.0": "cc1a73af661466c7f4e6a94d93ece78542d700f2165bdb16a531e9db8856c5aa", "https://esm.sh/v135/ajv-formats@2.1.1/denonext/ajv-formats.mjs": "06092e00b42202633ae6dab4b53287c133af882ddb14c6707277cdb237634967",