From f5edc10d2a42b6198d036b67a295fa9fec2b1d23 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Thu, 8 Aug 2024 10:25:55 +0700 Subject: [PATCH 1/8] add assignment id,add url --- .env.example | 2 +- .../20240115064335_init/migration.sql | 29 ----------- prisma/migrations/migration_lock.toml | 3 -- prisma/schema.prisma | 1 + src/database/database.ts | 1 + src/scraper/extractAssignmentsFromCheerio.ts | 8 ++- src/scraper/updateAll.ts | 49 ++++++++++--------- tests/parseAssignmentId.test.ts | 49 +++++++++++++++++++ tests/updateAll.test.ts | 0 9 files changed, 85 insertions(+), 57 deletions(-) delete mode 100644 prisma/migrations/20240115064335_init/migration.sql delete mode 100644 prisma/migrations/migration_lock.toml create mode 100644 tests/parseAssignmentId.test.ts create mode 100644 tests/updateAll.test.ts diff --git a/.env.example b/.env.example index 217ae0a..15bf4c4 100644 --- a/.env.example +++ b/.env.example @@ -1,7 +1,7 @@ DATABASE_URL= # for docker (compose prod) -POSTGRES_PASSWORD= +POSTGRES_PASSWORD=postgresql://[user[:password]]@[host][:port][/dbname] # Discord DISCORD_TOKEN= diff --git a/prisma/migrations/20240115064335_init/migration.sql b/prisma/migrations/20240115064335_init/migration.sql deleted file mode 100644 index 8b0787f..0000000 --- a/prisma/migrations/20240115064335_init/migration.sql +++ /dev/null @@ -1,29 +0,0 @@ --- CreateTable -CREATE TABLE "Assignment" ( - "mcvCourseID" INTEGER NOT NULL, - "assignmentName" VARCHAR(255) NOT NULL, - - CONSTRAINT "Assignment_pkey" PRIMARY KEY ("assignmentName") -); - --- CreateTable -CREATE TABLE "Course" ( - "mcvID" INTEGER NOT NULL, - "courseID" VARCHAR(255) NOT NULL, - "title" VARCHAR(500) NOT NULL, - "year" INTEGER NOT NULL, - "semester" INTEGER NOT NULL, - - CONSTRAINT "Course_pkey" PRIMARY KEY ("mcvID") -); - --- CreateTable -CREATE TABLE "NotificationChannel" ( - "guildID" VARCHAR(255) NOT NULL, - "channelID" VARCHAR(255) NOT NULL, - - CONSTRAINT "NotificationChannel_pkey" PRIMARY KEY ("guildID") -); - --- AddForeignKey -ALTER TABLE "Assignment" ADD CONSTRAINT "Assignment_mcvCourseID_fkey" FOREIGN KEY ("mcvCourseID") REFERENCES "Course"("mcvID") ON DELETE RESTRICT ON UPDATE CASCADE; diff --git a/prisma/migrations/migration_lock.toml b/prisma/migrations/migration_lock.toml deleted file mode 100644 index fbffa92..0000000 --- a/prisma/migrations/migration_lock.toml +++ /dev/null @@ -1,3 +0,0 @@ -# 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/prisma/schema.prisma b/prisma/schema.prisma index d2b1da5..7a2b6f3 100644 --- a/prisma/schema.prisma +++ b/prisma/schema.prisma @@ -6,6 +6,7 @@ model Assignment{ mcvCourseID Int course Course @relation(fields:[mcvCourseID], references: [mcvID]) assignmentName String @id @db.VarChar(255) + assignmentID Int? } model Course{ mcvID Int @id diff --git a/src/database/database.ts b/src/database/database.ts index 4bc54c9..6c76986 100644 --- a/src/database/database.ts +++ b/src/database/database.ts @@ -137,6 +137,7 @@ namespace db { export type NotificationChannel = PrismaNotificationChannel export type Course = PrismaCourse export type Assignment = PrismaAssignment +export type CourseWithAssignments = Course & { assignments: Array} export default db diff --git a/src/scraper/extractAssignmentsFromCheerio.ts b/src/scraper/extractAssignmentsFromCheerio.ts index fd1a350..2933fa6 100644 --- a/src/scraper/extractAssignmentsFromCheerio.ts +++ b/src/scraper/extractAssignmentsFromCheerio.ts @@ -9,15 +9,19 @@ export default async function extractAssignmentsFromCheerio( const assignments: Array = [] for (let i = 0; i < assignmentNameNodes.length; i++) { const ele = assignmentNameNodes[i] + let assignmentLink = $(ele).attr("href"); + const assignmentIdStr: string = assignmentLink!.match(/^.*\/(\d+)$/)![1]; + let assignmentId: number = parseInt(assignmentIdStr,10); const assignment: Assignment = { mcvCourseID: mcvID, assignmentName: $(ele).text(), + assignmentID: assignmentId } const found = await db.assignmentExists(assignment) if (!found) { - console.log('found new assignment',assignment) + // console.log('found new assignment',assignment) assignments.push(assignment) - db.saveAssignment(assignment) + await db.saveAssignment(assignment) } } return assignments diff --git a/src/scraper/updateAll.ts b/src/scraper/updateAll.ts index 1e5f79a..4abd003 100644 --- a/src/scraper/updateAll.ts +++ b/src/scraper/updateAll.ts @@ -1,4 +1,8 @@ -import db, { Assignment, Course } from '../database/database' +import db, { + Assignment, + Course, + CourseWithAssignments, +} from '../database/database' import updateAssignments from './updateAssignments' import updateCourses from './updateCourses' import * as option from 'fp-ts/Option' @@ -10,33 +14,34 @@ import * as option from 'fp-ts/Option' */ export async function updateAll(): Promise { await updateCourses() - // console.log(assignmentsCache.keys(), coursesCache.keys()) const coursesList = await db.getAllCoursesOfTargetSemester() - const assignments: Array = [] - for await (const courses of coursesList) { - const newAssignments = await updateAssignments(courses.mcvID) - if (option.isSome(newAssignments)) { - newAssignments.value.forEach((element) => { - assignments.push(element) + const unfilteredCoursesWithAssignments: Array = + await Promise.all( + coursesList.map(async (course) => { + const newAssignments = await updateAssignments(course.mcvID) + let newAssignmentsUnwrapped: Assignment[] = option.getOrElse( + () => [] as Assignment[] + )(newAssignments) + const result: CourseWithAssignments = { + ...course, + assignments: newAssignmentsUnwrapped, + } + return result }) + ) + const coursesWithAssignments = unfilteredCoursesWithAssignments.filter( + (course) => { + return course.assignments.length != 0 } - } - const messageObject: Record> = {} - if (assignments.length == 0) { + ) + if (coursesWithAssignments.length == 0) { return '' } - for (const assignment of assignments) { - const course = (await db.getCourse(assignment!.mcvCourseID)) as Course - if (messageObject[course.title] == null) { - messageObject[course.title] = [] - } - messageObject[course.title].push(assignment!.assignmentName) - } let message: string = '## New Assignments!!' - for (const courseTitle in messageObject) { - message += `\n- ${courseTitle}` - for (const assignmentName of messageObject[courseTitle]) { - message += `\n - ${assignmentName}` + for (const course of coursesWithAssignments) { + message += `\n- ${course.title}` + for (const assignment of course.assignments) { + message += `\n - [${assignment.assignmentName}](https://www.mycourseville.com/?q=courseville/worksheet/${course.mcvID}/${assignment.assignmentID})` } } return message diff --git a/tests/parseAssignmentId.test.ts b/tests/parseAssignmentId.test.ts new file mode 100644 index 0000000..60b360e --- /dev/null +++ b/tests/parseAssignmentId.test.ts @@ -0,0 +1,49 @@ +const mockSaveAssignment = jest.fn().mockImplementation(()=>{}) +const mockAssignmentExists = jest + .fn() + .mockImplementation(async () => false) +jest.mock('../src/database/database', () => { + return { + assignmentExists: mockAssignmentExists, + saveAssignment: mockSaveAssignment, + } +}) + +import * as db from '../src/database/database' +import extractAssignmentsFromCheerio from '../src/scraper/extractAssignmentsFromCheerio' +import * as cheerio from 'cheerio' + +describe('', () => { + it('', async () => { + let html = ` + + + + + + + + + + + +
+ HW + + Excercise 1 +
+ + + ` + let $ = cheerio.load(html); + // console.log($('tbody tr td:nth-child(2) a').toArray()) + await extractAssignmentsFromCheerio(123, $) + expect(mockAssignmentExists).toHaveBeenCalledTimes(1); + expect(mockSaveAssignment).toHaveBeenCalledTimes(1) + expect(mockSaveAssignment).toHaveBeenCalledWith({ + mcvCourseID: 123, + assignmentName: 'Excercise 1', + assignmentID: 4567, + }) + }) +}) diff --git a/tests/updateAll.test.ts b/tests/updateAll.test.ts new file mode 100644 index 0000000..e69de29 From 9f83dcd39c9fde9254e174bfd84f05ba93b85add Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Thu, 8 Aug 2024 11:06:54 +0700 Subject: [PATCH 2/8] add migrations --- .../20240115064335_init/migration.sql | 29 +++++++++++++++++++ .../migration.sql | 2 ++ prisma/migrations/migration_lock.toml | 3 ++ 3 files changed, 34 insertions(+) create mode 100644 prisma/migrations/20240115064335_init/migration.sql create mode 100644 prisma/migrations/20240808035830_add_assignment_url/migration.sql create mode 100644 prisma/migrations/migration_lock.toml diff --git a/prisma/migrations/20240115064335_init/migration.sql b/prisma/migrations/20240115064335_init/migration.sql new file mode 100644 index 0000000..8b0787f --- /dev/null +++ b/prisma/migrations/20240115064335_init/migration.sql @@ -0,0 +1,29 @@ +-- CreateTable +CREATE TABLE "Assignment" ( + "mcvCourseID" INTEGER NOT NULL, + "assignmentName" VARCHAR(255) NOT NULL, + + CONSTRAINT "Assignment_pkey" PRIMARY KEY ("assignmentName") +); + +-- CreateTable +CREATE TABLE "Course" ( + "mcvID" INTEGER NOT NULL, + "courseID" VARCHAR(255) NOT NULL, + "title" VARCHAR(500) NOT NULL, + "year" INTEGER NOT NULL, + "semester" INTEGER NOT NULL, + + CONSTRAINT "Course_pkey" PRIMARY KEY ("mcvID") +); + +-- CreateTable +CREATE TABLE "NotificationChannel" ( + "guildID" VARCHAR(255) NOT NULL, + "channelID" VARCHAR(255) NOT NULL, + + CONSTRAINT "NotificationChannel_pkey" PRIMARY KEY ("guildID") +); + +-- AddForeignKey +ALTER TABLE "Assignment" ADD CONSTRAINT "Assignment_mcvCourseID_fkey" FOREIGN KEY ("mcvCourseID") REFERENCES "Course"("mcvID") ON DELETE RESTRICT ON UPDATE CASCADE; diff --git a/prisma/migrations/20240808035830_add_assignment_url/migration.sql b/prisma/migrations/20240808035830_add_assignment_url/migration.sql new file mode 100644 index 0000000..b3fbdc8 --- /dev/null +++ b/prisma/migrations/20240808035830_add_assignment_url/migration.sql @@ -0,0 +1,2 @@ +-- AlterTable +ALTER TABLE "Assignment" ADD COLUMN "assignmentID" INTEGER; diff --git a/prisma/migrations/migration_lock.toml b/prisma/migrations/migration_lock.toml new file mode 100644 index 0000000..fbffa92 --- /dev/null +++ b/prisma/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 From 8bcaa6615ec34944534d24045bac73f1ab229917 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Sat, 10 Aug 2024 20:44:18 +0700 Subject: [PATCH 3/8] migrate db --- jest.config.js | 3 + src/scraper/extractAssignmentsFromCheerio.ts | 4 +- src/scraper/scrapeAssignmentsOfPage.ts | 5 +- src/scraper/updateAll.ts | 3 +- tests/tsconfig.json | 111 ++++++++++++++++++ .../determineYearAndSemester.unit.test.ts} | 18 +-- ...xtractAssignmentsFromCheerio.unit.test.ts} | 13 +- tests/unit/updateAll.unit.test.ts | 35 ++++++ .../updateCourses.unit.test.ts} | 21 ++-- tests/updateAll.test.ts | 0 tsconfig.json | 6 +- 11 files changed, 184 insertions(+), 35 deletions(-) create mode 100644 tests/tsconfig.json rename tests/{determineYearAndSemester.test.ts => unit/determineYearAndSemester.unit.test.ts} (71%) rename tests/{parseAssignmentId.test.ts => unit/extractAssignmentsFromCheerio.unit.test.ts} (79%) create mode 100644 tests/unit/updateAll.unit.test.ts rename tests/{stopNotifyAfterEncounteredError.test.ts => unit/updateCourses.unit.test.ts} (69%) delete mode 100644 tests/updateAll.test.ts diff --git a/jest.config.js b/jest.config.js index b413e10..a26e829 100644 --- a/jest.config.js +++ b/jest.config.js @@ -2,4 +2,7 @@ module.exports = { preset: 'ts-jest', testEnvironment: 'node', + moduleNameMapper: { + "^@/(.*)$": "/src/$1" + } }; \ No newline at end of file diff --git a/src/scraper/extractAssignmentsFromCheerio.ts b/src/scraper/extractAssignmentsFromCheerio.ts index 2933fa6..856637c 100644 --- a/src/scraper/extractAssignmentsFromCheerio.ts +++ b/src/scraper/extractAssignmentsFromCheerio.ts @@ -9,9 +9,9 @@ export default async function extractAssignmentsFromCheerio( const assignments: Array = [] for (let i = 0; i < assignmentNameNodes.length; i++) { const ele = assignmentNameNodes[i] - let assignmentLink = $(ele).attr("href"); + const assignmentLink = $(ele).attr("href"); const assignmentIdStr: string = assignmentLink!.match(/^.*\/(\d+)$/)![1]; - let assignmentId: number = parseInt(assignmentIdStr,10); + const assignmentId: number = parseInt(assignmentIdStr,10); const assignment: Assignment = { mcvCourseID: mcvID, assignmentName: $(ele).text(), diff --git a/src/scraper/scrapeAssignmentsOfPage.ts b/src/scraper/scrapeAssignmentsOfPage.ts index 149efc5..912038e 100644 --- a/src/scraper/scrapeAssignmentsOfPage.ts +++ b/src/scraper/scrapeAssignmentsOfPage.ts @@ -2,10 +2,7 @@ import { option } from 'fp-ts' import fetchAndCatch from '../utils/fetchAndCatch' import { Assignment } from '../database/database' import * as cheerio from 'cheerio' -import { hasEncounteredError } from '../server' -import errorFetchingNotify from '../utils/errorFetchingNotify' import LoadMoreAssignmentsResponse from '../interfaces/LoadMoreAssignmentsResponse' -import { NotifyMessage } from '../config/config' import extractAssignmentsFromCheerio from './extractAssignmentsFromCheerio' /** @@ -38,7 +35,7 @@ export default async function scrapeAssignmentsOfPage( let assignments = await extractAssignmentsFromCheerio(mcvID, $) if (resultJson.all == undefined || resultJson.all !== true) { - let optionalResult = await scrapeAssignmentsOfPage(mcvID, next + 5); + const optionalResult = await scrapeAssignmentsOfPage(mcvID, next + 5); if(option.isSome(optionalResult)){ assignments = assignments.concat(optionalResult.value); } diff --git a/src/scraper/updateAll.ts b/src/scraper/updateAll.ts index 4abd003..aaf0c9b 100644 --- a/src/scraper/updateAll.ts +++ b/src/scraper/updateAll.ts @@ -1,6 +1,5 @@ import db, { Assignment, - Course, CourseWithAssignments, } from '../database/database' import updateAssignments from './updateAssignments' @@ -19,7 +18,7 @@ export async function updateAll(): Promise { await Promise.all( coursesList.map(async (course) => { const newAssignments = await updateAssignments(course.mcvID) - let newAssignmentsUnwrapped: Assignment[] = option.getOrElse( + const newAssignmentsUnwrapped: Assignment[] = option.getOrElse( () => [] as Assignment[] )(newAssignments) const result: CourseWithAssignments = { diff --git a/tests/tsconfig.json b/tests/tsconfig.json new file mode 100644 index 0000000..c67f6d8 --- /dev/null +++ b/tests/tsconfig.json @@ -0,0 +1,111 @@ +{ + "compilerOptions": { + /* Visit https://aka.ms/tsconfig to read more about this file */ + + /* Projects */ + // "incremental": true, /* Save .tsbuildinfo files to allow for incremental compilation of projects. */ + // "composite": true, /* Enable constraints that allow a TypeScript project to be used with project references. */ + // "tsBuildInfoFile": "./.tsbuildinfo", /* Specify the path to .tsbuildinfo incremental compilation file. */ + // "disableSourceOfProjectReferenceRedirect": true, /* Disable preferring source files instead of declaration files when referencing composite projects. */ + // "disableSolutionSearching": true, /* Opt a project out of multi-project reference checking when editing. */ + // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ + + /* Language and Environment */ + "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ + // "jsx": "preserve", /* Specify what JSX code is generated. */ + // "experimentalDecorators": true, /* Enable experimental support for legacy experimental decorators. */ + // "emitDecoratorMetadata": true, /* Emit design-type metadata for decorated declarations in source files. */ + // "jsxFactory": "", /* Specify the JSX factory function used when targeting React JSX emit, e.g. 'React.createElement' or 'h'. */ + // "jsxFragmentFactory": "", /* Specify the JSX Fragment reference used for fragments when targeting React JSX emit e.g. 'React.Fragment' or 'Fragment'. */ + // "jsxImportSource": "", /* Specify module specifier used to import the JSX factory functions when using 'jsx: react-jsx*'. */ + // "reactNamespace": "", /* Specify the object invoked for 'createElement'. This only applies when targeting 'react' JSX emit. */ + // "noLib": true, /* Disable including any library files, including the default lib.d.ts. */ + // "useDefineForClassFields": true, /* Emit ECMAScript-standard-compliant class fields. */ + // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ + + /* Modules */ + "module": "commonjs", /* Specify what module code is generated. */ + "rootDir": "../", /* Specify the root folder within your source files. */ + // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */ + // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ + "paths": { + "@/*": ["../src/*"] + }, /* Specify a set of entries that re-map imports to additional lookup locations. */ + // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ + // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ + // "types": [], /* Specify type package names to be included without being referenced in a source file. */ + // "allowUmdGlobalAccess": true, /* Allow accessing UMD globals from modules. */ + // "moduleSuffixes": [], /* List of file name suffixes to search when resolving a module. */ + // "allowImportingTsExtensions": true, /* Allow imports to include TypeScript file extensions. Requires '--moduleResolution bundler' and either '--noEmit' or '--emitDeclarationOnly' to be set. */ + // "resolvePackageJsonExports": true, /* Use the package.json 'exports' field when resolving package imports. */ + // "resolvePackageJsonImports": true, /* Use the package.json 'imports' field when resolving imports. */ + // "customConditions": [], /* Conditions to set in addition to the resolver-specific defaults when resolving imports. */ + // "resolveJsonModule": true, /* Enable importing .json files. */ + // "allowArbitraryExtensions": true, /* Enable importing files with any extension, provided a declaration file is present. */ + // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */ + + /* JavaScript Support */ + // "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */ + // "checkJs": true, /* Enable error reporting in type-checked JavaScript files. */ + // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */ + + /* Emit */ + // "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */ + // "declarationMap": true, /* Create sourcemaps for d.ts files. */ + // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */ + // "sourceMap": true, /* Create source map files for emitted JavaScript files. */ + // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ + // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ + "outDir": "./build", /* Specify an output folder for all emitted files. */ + // "removeComments": true, /* Disable emitting comments. */ + // "noEmit": true, /* Disable emitting files from a compilation. */ + // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ + // "importsNotUsedAsValues": "remove", /* Specify emit/checking behavior for imports that are only used for types. */ + // "downlevelIteration": true, /* Emit more compliant, but verbose and less performant JavaScript for iteration. */ + // "sourceRoot": "", /* Specify the root path for debuggers to find the reference source code. */ + // "mapRoot": "", /* Specify the location where debugger should locate map files instead of generated locations. */ + // "inlineSources": true, /* Include source code in the sourcemaps inside the emitted JavaScript. */ + // "emitBOM": true, /* Emit a UTF-8 Byte Order Mark (BOM) in the beginning of output files. */ + // "newLine": "crlf", /* Set the newline character for emitting files. */ + // "stripInternal": true, /* Disable emitting declarations that have '@internal' in their JSDoc comments. */ + // "noEmitHelpers": true, /* Disable generating custom helper functions like '__extends' in compiled output. */ + // "noEmitOnError": true, /* Disable emitting files if any type checking errors are reported. */ + // "preserveConstEnums": true, /* Disable erasing 'const enum' declarations in generated code. */ + // "declarationDir": "./", /* Specify the output directory for generated declaration files. */ + // "preserveValueImports": true, /* Preserve unused imported values in the JavaScript output that would otherwise be removed. */ + + /* Interop Constraints */ + // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ + // "verbatimModuleSyntax": true, /* Do not transform or elide any imports or exports not marked as type-only, ensuring they are written in the output file's format based on the 'module' setting. */ + "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ + "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ + "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + + /* Type Checking */ + "strict": true, /* Enable all strict type-checking options. */ + "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ + "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ + // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ + // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ + // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ + // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ + "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ + // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ + // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ + // "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */ + // "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */ + // "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */ + // "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */ + // "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */ + // "allowUnusedLabels": true, /* Disable error reporting for unused labels. */ + // "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */ + + /* Completeness */ + // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ + }, +} diff --git a/tests/determineYearAndSemester.test.ts b/tests/unit/determineYearAndSemester.unit.test.ts similarity index 71% rename from tests/determineYearAndSemester.test.ts rename to tests/unit/determineYearAndSemester.unit.test.ts index db31499..73529d1 100644 --- a/tests/determineYearAndSemester.test.ts +++ b/tests/unit/determineYearAndSemester.unit.test.ts @@ -13,7 +13,7 @@ global.fetch = jest.fn(async () => { } }) as jest.Mock -jest.mock('../src/env/env', () => { +jest.mock('@/env/env', () => { return { COOKIE: 'cookie', ERROR_FETCHING_NOTIFICATION: false, @@ -21,8 +21,8 @@ jest.mock('../src/env/env', () => { } }) -jest.mock('../src/server', () => { - const actualModule = jest.requireActual('../src/server') +jest.mock('@/server', () => { + const actualModule = jest.requireActual('@/server') return { __esModule: true, ...actualModule, @@ -33,23 +33,23 @@ jest.mock('../src/server', () => { } }) -jest.mock('../src/database/database', () => { +jest.mock('@/database/database', () => { return { __esModule: true, } }) const mockErrorFetchingNotify = jest.fn() -jest.mock('../src/utils/errorFetchingNotify', () => ({ +jest.mock('@/utils/errorFetchingNotify', () => ({ __esModule: true, default: mockErrorFetchingNotify, })) -import fetchAndCatch from '../src/utils/fetchAndCatch' -import { determineYearAndSemester } from '../src/scraper/determineYearAndSemester' +import fetchAndCatch from '@/utils/fetchAndCatch' +import { determineYearAndSemester } from '@/scraper/determineYearAndSemester' import { option } from 'fp-ts' -import { targetSemester, targetYear } from '../src/config/config' -import responseToCheerio from '../src/utils/responseToCheerio' +import { targetSemester, targetYear } from '@/config/config' +import responseToCheerio from '@/utils/responseToCheerio' let $: cheerio.Root describe('stop notify after encountered error', () => { diff --git a/tests/parseAssignmentId.test.ts b/tests/unit/extractAssignmentsFromCheerio.unit.test.ts similarity index 79% rename from tests/parseAssignmentId.test.ts rename to tests/unit/extractAssignmentsFromCheerio.unit.test.ts index 60b360e..85311c7 100644 --- a/tests/parseAssignmentId.test.ts +++ b/tests/unit/extractAssignmentsFromCheerio.unit.test.ts @@ -2,20 +2,19 @@ const mockSaveAssignment = jest.fn().mockImplementation(()=>{}) const mockAssignmentExists = jest .fn() .mockImplementation(async () => false) -jest.mock('../src/database/database', () => { +jest.mock('@/database/database', () => { return { assignmentExists: mockAssignmentExists, saveAssignment: mockSaveAssignment, } }) -import * as db from '../src/database/database' -import extractAssignmentsFromCheerio from '../src/scraper/extractAssignmentsFromCheerio' +import extractAssignmentsFromCheerio from '@/scraper/extractAssignmentsFromCheerio' import * as cheerio from 'cheerio' -describe('', () => { - it('', async () => { - let html = ` +describe('parse assignments', () => { + it('assignmentId', async () => { + const html = ` @@ -35,7 +34,7 @@ describe('', () => { ` - let $ = cheerio.load(html); + const $ = cheerio.load(html); // console.log($('tbody tr td:nth-child(2) a').toArray()) await extractAssignmentsFromCheerio(123, $) expect(mockAssignmentExists).toHaveBeenCalledTimes(1); diff --git a/tests/unit/updateAll.unit.test.ts b/tests/unit/updateAll.unit.test.ts new file mode 100644 index 0000000..5faee6d --- /dev/null +++ b/tests/unit/updateAll.unit.test.ts @@ -0,0 +1,35 @@ +import {Course} from "@prisma/client"; + +let mockUpdateCourses = jest.fn(); +let mockUpdateAssignments = jest.fn(); +let mockDatabase = {}; + +jest.mock("@/scraper/updateAll",()=>{ + return { + __default: mockUpdateCourses + } +}) + +jest.mock("@/database/database",()=>{ + return { + __default: mockDatabase + } +}) + +jest.mock("@/scraper/updateAssignments",()=>{ + return { + __default: mockUpdateAssignments + } +}) + +describe("",()=>{ + let coursesFromUpdate: Course[] = []; + let assignmentsOfCourseFromUpdate: Record = {}; + beforeAll(()=>{ + mockUpdateCourses.mockImplementation(()=>coursesFromUpdate); + mockUpdateAssignments.mockImplementation(()=>coursesFromUpdate); + }) + it("normal",()=>{ + + }) +}) \ No newline at end of file diff --git a/tests/stopNotifyAfterEncounteredError.test.ts b/tests/unit/updateCourses.unit.test.ts similarity index 69% rename from tests/stopNotifyAfterEncounteredError.test.ts rename to tests/unit/updateCourses.unit.test.ts index 84c0b18..8c04872 100644 --- a/tests/stopNotifyAfterEncounteredError.test.ts +++ b/tests/unit/updateCourses.unit.test.ts @@ -1,8 +1,6 @@ -global.fetch = jest.fn(async () => { - throw new Error('') -}) as jest.Mock +global.fetch = jest.fn(); -jest.mock('../src/env/env', () => { +jest.mock('@/env/env', () => { return { COOKIE: 'cookie', ERROR_FETCHING_NOTIFICATION: false, @@ -10,8 +8,8 @@ jest.mock('../src/env/env', () => { } }) -jest.mock('../src/server', () => { - const actualModule = jest.requireActual('../src/server') +jest.mock('@/server', () => { + const actualModule = jest.requireActual('@/server') return { __esModule: true, ...actualModule, @@ -23,15 +21,20 @@ jest.mock('../src/server', () => { }) const mockErrorFetchingNotify = jest.fn() -jest.mock('../src/utils/errorFetchingNotify', () => ({ +jest.mock('@/utils/errorFetchingNotify', () => ({ __esModule: true, default: mockErrorFetchingNotify, })) -import updateCourses from '../src/scraper/updateCourses' -import { hasEncounteredError } from '../src/server' +import updateCourses from '@/scraper/updateCourses' +import { hasEncounteredError } from '@/server' describe('stop notify after encountered error', () => { + beforeAll(()=>{ + (global.fetch as jest.Mock).mockImplementation((async () => { + throw new Error('') + })) + }) beforeEach(() => { hasEncounteredError.set(false) mockErrorFetchingNotify.mockClear() diff --git a/tests/updateAll.test.ts b/tests/updateAll.test.ts deleted file mode 100644 index e69de29..0000000 diff --git a/tsconfig.json b/tsconfig.json index 8c0d6f7..3f92a05 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -26,10 +26,12 @@ /* Modules */ "module": "commonjs", /* Specify what module code is generated. */ - "rootDir": "./src", /* Specify the root folder within your source files. */ + "rootDir": "./", /* Specify the root folder within your source files. */ // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */ // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ - // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */ + "paths": { + "@/*": ["./src/*"] + }, /* Specify a set of entries that re-map imports to additional lookup locations. */ // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ // "types": [], /* Specify type package names to be included without being referenced in a source file. */ From 481d0d37be0a53c38df638b871710f42aa32a7e3 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Mon, 12 Aug 2024 16:42:08 +0700 Subject: [PATCH 4/8] fix eslint config, edit updateAll to support multiple messages once --- .github/workflows/on_push.yml | 3 +- eslint.config.mjs | 31 +- package-lock.json | 2088 ++++++++--------- package.json | 5 +- .../migration.sql | 11 + prisma/schema.prisma | 5 +- src/commands/update.ts | 33 +- src/config/config.ts | 5 + src/database/database.ts | 4 +- src/env/env.ts | 4 +- src/scraper/determineYearAndSemester.ts | 4 +- src/scraper/updateAll.ts | 89 +- src/scraper/updateCourses.ts | 2 +- src/server.ts | 3 +- src/start.ts | 2 +- src/utils/MutableWrapper.ts | 12 +- src/utils/fetchAndCatch.ts | 10 +- src/utils/updateHandler.ts | 30 +- .../determineYearAndSemester.unit.test.ts | 4 +- tests/unit/updateAll.unit.test.ts | 289 ++- tests/unit/updateCourses.unit.test.ts | 6 +- tsconfig.json | 30 +- 22 files changed, 1482 insertions(+), 1188 deletions(-) create mode 100644 prisma/migrations/20240811035309_change_unique_constraint/migration.sql diff --git a/.github/workflows/on_push.yml b/.github/workflows/on_push.yml index 719ee54..f9860c7 100644 --- a/.github/workflows/on_push.yml +++ b/.github/workflows/on_push.yml @@ -12,4 +12,5 @@ jobs: node-version: 'latest' - run: npm ci - run: npm test - - run: npm run lint \ No newline at end of file + - run: npm run lint + - run: npm run checkStyle \ No newline at end of file diff --git a/eslint.config.mjs b/eslint.config.mjs index dec2dc1..7cdda02 100644 --- a/eslint.config.mjs +++ b/eslint.config.mjs @@ -1,27 +1,26 @@ import globals from "globals"; -import pluginJs from "@eslint/js"; import tseslint from "typescript-eslint"; +import tsPlugin from "@typescript-eslint/eslint-plugin"; +import tsParser from "@typescript-eslint/parser"; import unusedImports from "eslint-plugin-unused-imports"; - -export default [ - { - files: ["src/**/*.ts"] - }, - { - ignores: ["build/**/*"] - }, - { languageOptions: { globals: globals.browser } }, - pluginJs.configs.recommended, - ...tseslint.configs.recommended, +export default tseslint.config( { + files: ["src/**/*.ts","tests/**/*.ts"], + languageOptions: { + parser: tsParser, + globals: globals.node + }, plugins: { - "unused-imports": unusedImports + "unused-imports": unusedImports, + "@typescript-eslint": tsPlugin }, rules: { + ...tsPlugin.configs.recommended.rules, "no-unused-vars": "off", "no-undef": "off", "unused-imports/no-unused-imports": "error", - "@typescript-eslint/no-unused-vars": [ + "@typescript-eslint/no-unused-vars": + [ "error", { "varsIgnorePattern": "^_", @@ -29,7 +28,7 @@ export default [ "caughtErrorsIgnorePattern": "^_" } ], - // "@typescript-eslint/no-explicit-any": "off" + "@typescript-eslint/no-explicit-any": "off" } }, -]; \ No newline at end of file +) \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index ba4ea60..810e7e1 100644 --- a/package-lock.json +++ b/package-lock.json @@ -16,6 +16,7 @@ "envalid": "^8.0.0", "express": "^4.18.2", "fp-ts": "^2.16.7", + "mcv-discord-bot": "file:", "node-cache": "^5.1.2", "zod": "^3.23.8" }, @@ -38,6 +39,7 @@ "prisma": "^5.16.1", "ts-jest": "^29.1.5", "ts-node": "^10.9.2", + "tsconfig-paths": "^4.2.0", "typescript": "^5.5.3", "typescript-eslint": "^7.15.0" } @@ -55,16 +57,6 @@ "node": ">=6.0.0" } }, - "node_modules/@ampproject/remapping/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/@babel/code-frame": { "version": "7.24.7", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.24.7.tgz", @@ -79,30 +71,30 @@ } }, "node_modules/@babel/compat-data": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.24.7.tgz", - "integrity": "sha512-qJzAIcv03PyaWqxRgO4mSU3lihncDT296vnyuE2O8uA4w3UHWI4S3hgeZd1L8W1Bft40w9JxJ2b412iDUFFRhw==", + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.25.2.tgz", + "integrity": "sha512-bYcppcpKBvX4znYaPEeFau03bp89ShqNMLs+rmdptMw+heSZh9+z84d2YG+K7cYLbWwzdjtDoW/uqZmPjulClQ==", "dev": true, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/core": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.24.7.tgz", - "integrity": "sha512-nykK+LEK86ahTkX/3TgauT0ikKoNCfKHEaZYTUVupJdTLzGNvrblu4u6fa7DhZONAltdf8e662t/abY8idrd/g==", + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.25.2.tgz", + "integrity": "sha512-BBt3opiCOxUr9euZ5/ro/Xv8/V7yJ5bjYMqG/C1YAo8MIKAnumZalCN+msbci3Pigy4lIQfPUpfMM27HMGaYEA==", "dev": true, "dependencies": { "@ampproject/remapping": "^2.2.0", "@babel/code-frame": "^7.24.7", - "@babel/generator": "^7.24.7", - "@babel/helper-compilation-targets": "^7.24.7", - "@babel/helper-module-transforms": "^7.24.7", - "@babel/helpers": "^7.24.7", - "@babel/parser": "^7.24.7", - "@babel/template": "^7.24.7", - "@babel/traverse": "^7.24.7", - "@babel/types": "^7.24.7", + "@babel/generator": "^7.25.0", + "@babel/helper-compilation-targets": "^7.25.2", + "@babel/helper-module-transforms": "^7.25.2", + "@babel/helpers": "^7.25.0", + "@babel/parser": "^7.25.0", + "@babel/template": "^7.25.0", + "@babel/traverse": "^7.25.2", + "@babel/types": "^7.25.2", "convert-source-map": "^2.0.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.2", @@ -117,45 +109,13 @@ "url": "https://opencollective.com/babel" } }, - "node_modules/@babel/core/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/@babel/core/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, - "node_modules/@babel/core/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "bin": { - "semver": "bin/semver.js" - } - }, "node_modules/@babel/generator": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.24.7.tgz", - "integrity": "sha512-oipXieGC3i45Y1A41t4tAqpnEZWgB/lC6Ehh6+rOviR5XWpTtMmLN+fGjz9vOiNRt0p6RtO6DtD0pdU3vpqdSA==", + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.25.0.tgz", + "integrity": "sha512-3LEEcj3PVW8pW2R1SR1M89g/qrYk/m/mB/tLqn7dn4sbBUQyTqnlod+II2U4dqiGtUmkcnAmkMDralTFZttRiw==", "dev": true, "dependencies": { - "@babel/types": "^7.24.7", + "@babel/types": "^7.25.0", "@jridgewell/gen-mapping": "^0.3.5", "@jridgewell/trace-mapping": "^0.3.25", "jsesc": "^2.5.1" @@ -164,25 +124,15 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/generator/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/@babel/helper-compilation-targets": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.24.7.tgz", - "integrity": "sha512-ctSdRHBi20qWOfy27RUb4Fhp07KSJ3sXcuSvTrXrc4aG8NSYDo1ici3Vhg9bg69y5bj0Mr1lh0aeEgTvc12rMg==", + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.25.2.tgz", + "integrity": "sha512-U2U5LsSaZ7TAt3cfaymQ8WHh0pxvdHoEk6HVpaexxixjyEquMh0L0YNJNM6CTGKMXV1iksi0iZkGw4AcFkPaaw==", "dev": true, "dependencies": { - "@babel/compat-data": "^7.24.7", - "@babel/helper-validator-option": "^7.24.7", - "browserslist": "^4.22.2", + "@babel/compat-data": "^7.25.2", + "@babel/helper-validator-option": "^7.24.8", + "browserslist": "^4.23.1", "lru-cache": "^5.1.1", "semver": "^6.3.1" }, @@ -190,67 +140,6 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/helper-compilation-targets/node_modules/lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "dependencies": { - "yallist": "^3.0.2" - } - }, - "node_modules/@babel/helper-compilation-targets/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "bin": { - "semver": "bin/semver.js" - } - }, - "node_modules/@babel/helper-compilation-targets/node_modules/yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - }, - "node_modules/@babel/helper-environment-visitor": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-environment-visitor/-/helper-environment-visitor-7.24.7.tgz", - "integrity": "sha512-DoiN84+4Gnd0ncbBOM9AZENV4a5ZiL39HYMyZJGZ/AZEykHYdJw0wW3kdcsh9/Kn+BRXHLkkklZ51ecPKmI1CQ==", - "dev": true, - "dependencies": { - "@babel/types": "^7.24.7" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-function-name": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.24.7.tgz", - "integrity": "sha512-FyoJTsj/PEUWu1/TYRiXTIHc8lbw+TDYkZuoE43opPS5TrI7MyONBE1oNvfguEXAD9yhQRrVBnXdXzSLQl9XnA==", - "dev": true, - "dependencies": { - "@babel/template": "^7.24.7", - "@babel/types": "^7.24.7" - }, - "engines": { - "node": ">=6.9.0" - } - }, - "node_modules/@babel/helper-hoist-variables": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.24.7.tgz", - "integrity": "sha512-MJJwhkoGy5c4ehfoRyrJ/owKeMl19U54h27YYftT0o2teQ3FJ3nQUf/I3LlJsX4l3qlw7WRXUmiyajvHXoTubQ==", - "dev": true, - "dependencies": { - "@babel/types": "^7.24.7" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/helper-module-imports": { "version": "7.24.7", "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.24.7.tgz", @@ -265,16 +154,15 @@ } }, "node_modules/@babel/helper-module-transforms": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.24.7.tgz", - "integrity": "sha512-1fuJEwIrp+97rM4RWdO+qrRsZlAeL1lQJoPqtCYWv0NL115XM93hIH4CSRln2w52SqvmY5hqdtauB6QFCDiZNQ==", + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.25.2.tgz", + "integrity": "sha512-BjyRAbix6j/wv83ftcVJmBt72QtHI56C7JXZoG2xATiLpmoC7dpd8WnkikExHDVPpi/3qCmO6WY1EaXOluiecQ==", "dev": true, "dependencies": { - "@babel/helper-environment-visitor": "^7.24.7", "@babel/helper-module-imports": "^7.24.7", "@babel/helper-simple-access": "^7.24.7", - "@babel/helper-split-export-declaration": "^7.24.7", - "@babel/helper-validator-identifier": "^7.24.7" + "@babel/helper-validator-identifier": "^7.24.7", + "@babel/traverse": "^7.25.2" }, "engines": { "node": ">=6.9.0" @@ -284,9 +172,9 @@ } }, "node_modules/@babel/helper-plugin-utils": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.7.tgz", - "integrity": "sha512-Rq76wjt7yz9AAc1KnlRKNAi/dMSVWgDRx43FHoJEbcYU6xOWaE2dVPwcdTukJrjxS65GITyfbvEYHvkirZ6uEg==", + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.24.8.tgz", + "integrity": "sha512-FFWx5142D8h2Mgr/iPVGH5G7w6jDn4jUSpZTyDnQO0Yn7Ks2Kuz6Pci8H6MPCoUJegd/UZQ3tAvfLCxQSnWWwg==", "dev": true, "engines": { "node": ">=6.9.0" @@ -305,22 +193,10 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/helper-split-export-declaration": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.24.7.tgz", - "integrity": "sha512-oy5V7pD+UvfkEATUKvIjvIAH/xCzfsFVw7ygW2SI6NClZzquT+mwdTfgfdbUiceh6iQO0CHtCPsyze/MZ2YbAA==", - "dev": true, - "dependencies": { - "@babel/types": "^7.24.7" - }, - "engines": { - "node": ">=6.9.0" - } - }, "node_modules/@babel/helper-string-parser": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.7.tgz", - "integrity": "sha512-7MbVt6xrwFQbunH2DNQsAP5sTGxfqQtErvBIvIMi6EQnbgUOuVYanvREcmFrOPhoXBrTtjhhP+lW+o5UfK+tDg==", + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-string-parser/-/helper-string-parser-7.24.8.tgz", + "integrity": "sha512-pO9KhhRcuUyGnJWwyEgnRJTSIZHiT+vMD0kPeD+so0l7mxkMT19g3pjY9GTnHySck/hDzq+dtW/4VgnMkippsQ==", "dev": true, "engines": { "node": ">=6.9.0" @@ -336,22 +212,22 @@ } }, "node_modules/@babel/helper-validator-option": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.24.7.tgz", - "integrity": "sha512-yy1/KvjhV/ZCL+SM7hBrvnZJ3ZuT9OuZgIJAGpPEToANvc3iM6iDvBnRjtElWibHU6n8/LPR/EjX9EtIEYO3pw==", + "version": "7.24.8", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.24.8.tgz", + "integrity": "sha512-xb8t9tD1MHLungh/AIoWYN+gVHaB9kwlu8gffXGSt3FFEIT7RjS+xWbc2vUD1UTZdIpKj/ab3rdqJ7ufngyi2Q==", "dev": true, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/helpers": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.24.7.tgz", - "integrity": "sha512-NlmJJtvcw72yRJRcnCmGvSi+3jDEg8qFu3z0AFoymmzLx5ERVWyzd9kVXr7Th9/8yIJi2Zc6av4Tqz3wFs8QWg==", + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.25.0.tgz", + "integrity": "sha512-MjgLZ42aCm0oGjJj8CtSM3DB8NOOf8h2l7DCTePJs29u+v7yO/RBX9nShlKMgFnRks/Q4tBAe7Hxnov9VkGwLw==", "dev": true, "dependencies": { - "@babel/template": "^7.24.7", - "@babel/types": "^7.24.7" + "@babel/template": "^7.25.0", + "@babel/types": "^7.25.0" }, "engines": { "node": ">=6.9.0" @@ -422,11 +298,35 @@ "node": ">=0.8.0" } }, + "node_modules/@babel/highlight/node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/@babel/highlight/node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, "node_modules/@babel/parser": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.24.7.tgz", - "integrity": "sha512-9uUYRm6OqQrCqQdG1iCBwBPZgN8ciDBro2nIOFaiRz1/BCxaI7CNvQbDHvsArAC7Tw9Hda/B3U+6ui9u4HWXPw==", + "version": "7.25.3", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.25.3.tgz", + "integrity": "sha512-iLTJKDbJ4hMvFPgQwwsVoxtHyWpKKPBrxkANrSYewDPaPpT5py5yeVkgPIJ7XYXhndxJpaA3PyALSXQ7u8e/Dw==", "dev": true, + "dependencies": { + "@babel/types": "^7.25.2" + }, "bin": { "parser": "bin/babel-parser.js" }, @@ -612,33 +512,30 @@ } }, "node_modules/@babel/template": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.24.7.tgz", - "integrity": "sha512-jYqfPrU9JTF0PmPy1tLYHW4Mp4KlgxJD9l2nP9fD6yT/ICi554DmrWBAEYpIelzjHf1msDP3PxJIRt/nFNfBig==", + "version": "7.25.0", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.25.0.tgz", + "integrity": "sha512-aOOgh1/5XzKvg1jvVz7AVrx2piJ2XBi227DHmbY6y+bM9H2FlN+IfecYu4Xl0cNiiVejlsCri89LUsbj8vJD9Q==", "dev": true, "dependencies": { "@babel/code-frame": "^7.24.7", - "@babel/parser": "^7.24.7", - "@babel/types": "^7.24.7" + "@babel/parser": "^7.25.0", + "@babel/types": "^7.25.0" }, "engines": { "node": ">=6.9.0" } }, "node_modules/@babel/traverse": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.24.7.tgz", - "integrity": "sha512-yb65Ed5S/QAcewNPh0nZczy9JdYXkkAbIsEo+P7BE7yO3txAY30Y/oPa3QkQ5It3xVG2kpKMg9MsdxZaO31uKA==", + "version": "7.25.3", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.25.3.tgz", + "integrity": "sha512-HefgyP1x754oGCsKmV5reSmtV7IXj/kpaE1XYY+D9G5PvKKoFfSbiS4M77MdjuwlZKDIKFCffq9rPU+H/s3ZdQ==", "dev": true, "dependencies": { "@babel/code-frame": "^7.24.7", - "@babel/generator": "^7.24.7", - "@babel/helper-environment-visitor": "^7.24.7", - "@babel/helper-function-name": "^7.24.7", - "@babel/helper-hoist-variables": "^7.24.7", - "@babel/helper-split-export-declaration": "^7.24.7", - "@babel/parser": "^7.24.7", - "@babel/types": "^7.24.7", + "@babel/generator": "^7.25.0", + "@babel/parser": "^7.25.3", + "@babel/template": "^7.25.0", + "@babel/types": "^7.25.2", "debug": "^4.3.1", "globals": "^11.1.0" }, @@ -646,23 +543,6 @@ "node": ">=6.9.0" } }, - "node_modules/@babel/traverse/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, "node_modules/@babel/traverse/node_modules/globals": { "version": "11.12.0", "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", @@ -672,19 +552,13 @@ "node": ">=4" } }, - "node_modules/@babel/traverse/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, "node_modules/@babel/types": { - "version": "7.24.7", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.24.7.tgz", - "integrity": "sha512-XEFXSlxiG5td2EJRe8vOmRbaXVgfcBlszKujvVmWIK/UpywWljQCfzAv3RQCGujWQ1RD4YYWEAqDXfuJiy8f5Q==", + "version": "7.25.2", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.25.2.tgz", + "integrity": "sha512-YTnYtra7W9e6/oAZEHj0bJehPRUlLH9/fbpT5LfB0NhQXyALCRkRs3zH9v07IYhkgpqX6Z78FnuccZr/l4Fs4Q==", "dev": true, "dependencies": { - "@babel/helper-string-parser": "^7.24.7", + "@babel/helper-string-parser": "^7.24.8", "@babel/helper-validator-identifier": "^7.24.7", "to-fast-properties": "^2.0.0" }, @@ -710,6 +584,16 @@ "node": ">=12" } }, + "node_modules/@cspotcode/source-map-support/node_modules/@jridgewell/trace-mapping": { + "version": "0.3.9", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", + "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "dev": true, + "dependencies": { + "@jridgewell/resolve-uri": "^3.0.3", + "@jridgewell/sourcemap-codec": "^1.4.10" + } + }, "node_modules/@discordjs/builders": { "version": "1.8.2", "resolved": "https://registry.npmjs.org/@discordjs/builders/-/builders-1.8.2.tgz", @@ -785,6 +669,14 @@ "url": "https://github.com/discordjs/discord.js?sponsor" } }, + "node_modules/@discordjs/rest/node_modules/undici": { + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/undici/-/undici-6.13.0.tgz", + "integrity": "sha512-Q2rtqmZWrbP8nePMq7mOJIN98M0fYvSgV89vwl/BQRT4mDOeY2GXZngfGpcBBhtky3woM7G24wZV3Q304Bv6cw==", + "engines": { + "node": ">=18.0" + } + }, "node_modules/@discordjs/util": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@discordjs/util/-/util-1.1.0.tgz", @@ -876,29 +768,6 @@ "url": "https://opencollective.com/eslint" } }, - "node_modules/@eslint/eslintrc/node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true - }, - "node_modules/@eslint/eslintrc/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, "node_modules/@eslint/eslintrc/node_modules/globals": { "version": "13.24.0", "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", @@ -914,24 +783,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/@eslint/eslintrc/node_modules/js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, - "node_modules/@eslint/eslintrc/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, "node_modules/@eslint/eslintrc/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -945,9 +796,9 @@ } }, "node_modules/@eslint/js": { - "version": "9.6.0", - "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.6.0.tgz", - "integrity": "sha512-D9B0/3vNg44ZeWbYMpBoXqNP4j6eQD5vNwIlGAuFRRzK/WtT/jvDQW3Bi9kkf3PMDMlM7Yi+73VLUsn5bJcl8A==", + "version": "9.9.0", + "resolved": "https://registry.npmjs.org/@eslint/js/-/js-9.9.0.tgz", + "integrity": "sha512-hhetes6ZHP3BlXLxmd8K2SNgkhNSi+UcecbnwWKwpP7kyi/uC75DJ1lOOBO3xrC4jyojtGE3YxKZPHfk4yrgug==", "dev": true, "engines": { "node": "^18.18.0 || ^20.9.0 || >=21.1.0" @@ -968,29 +819,6 @@ "node": ">=10.10.0" } }, - "node_modules/@humanwhocodes/config-array/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/@humanwhocodes/config-array/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, "node_modules/@humanwhocodes/module-importer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz", @@ -1027,6 +855,89 @@ "node": ">=8" } }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/js-yaml": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", + "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@istanbuljs/load-nyc-config/node_modules/resolve-from": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", + "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, "node_modules/@istanbuljs/schema": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", @@ -1215,16 +1126,6 @@ } } }, - "node_modules/@jest/reporters/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/@jest/schemas": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/@jest/schemas/-/schemas-29.6.3.tgz", @@ -1251,16 +1152,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/@jest/source-map/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/@jest/test-result": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/@jest/test-result/-/test-result-29.7.0.tgz", @@ -1317,16 +1208,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/@jest/transform/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/@jest/types": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/@jest/types/-/types-29.6.3.tgz", @@ -1351,21 +1232,11 @@ "dev": true, "dependencies": { "@jridgewell/set-array": "^1.2.1", - "@jridgewell/sourcemap-codec": "^1.4.10", - "@jridgewell/trace-mapping": "^0.3.24" - }, - "engines": { - "node": ">=6.0.0" - } - }, - "node_modules/@jridgewell/gen-mapping/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" + "@jridgewell/sourcemap-codec": "^1.4.10", + "@jridgewell/trace-mapping": "^0.3.24" + }, + "engines": { + "node": ">=6.0.0" } }, "node_modules/@jridgewell/resolve-uri": { @@ -1387,19 +1258,19 @@ } }, "node_modules/@jridgewell/sourcemap-codec": { - "version": "1.4.15", - "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.4.15.tgz", - "integrity": "sha512-eF2rxCRulEKXHTRiDrDy6erMYWqNw4LPdQ8UQA4huuxaQsVeRPFl2oM8oDGxMFhJUWZf9McpLtJasDDZb/Bpeg==", + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz", + "integrity": "sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ==", "dev": true }, "node_modules/@jridgewell/trace-mapping": { - "version": "0.3.9", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz", - "integrity": "sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ==", + "version": "0.3.25", + "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", + "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", "dev": true, "dependencies": { - "@jridgewell/resolve-uri": "^3.0.3", - "@jridgewell/sourcemap-codec": "^1.4.10" + "@jridgewell/resolve-uri": "^3.1.0", + "@jridgewell/sourcemap-codec": "^1.4.14" } }, "node_modules/@nodelib/fs.scandir": { @@ -1438,9 +1309,9 @@ } }, "node_modules/@prisma/client": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@prisma/client/-/client-5.16.1.tgz", - "integrity": "sha512-wM9SKQjF0qLxdnOZIVAIMKiz6Hu7vDt4FFAih85K1dk/Rr2mdahy6d3QP41K62N9O0DJJA//gUDA3Mp49xsKIg==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/@prisma/client/-/client-5.18.0.tgz", + "integrity": "sha512-BWivkLh+af1kqC89zCJYkHsRcyWsM8/JHpsDMM76DjP3ZdEquJhXa4IeX+HkWPnwJ5FanxEJFZZDTWiDs/Kvyw==", "hasInstallScript": true, "engines": { "node": ">=16.13" @@ -1455,54 +1326,54 @@ } }, "node_modules/@prisma/debug": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@prisma/debug/-/debug-5.16.1.tgz", - "integrity": "sha512-JsNgZAg6BD9RInLSrg7ZYzo11N7cVvYArq3fHGSD89HSgtN0VDdjV6bib7YddbcO6snzjchTiLfjeTqBjtArVQ==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/@prisma/debug/-/debug-5.18.0.tgz", + "integrity": "sha512-f+ZvpTLidSo3LMJxQPVgAxdAjzv5OpzAo/eF8qZqbwvgi2F5cTOI9XCpdRzJYA0iGfajjwjOKKrVq64vkxEfUw==", "devOptional": true }, "node_modules/@prisma/engines": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@prisma/engines/-/engines-5.16.1.tgz", - "integrity": "sha512-KkyF3eIUtBIyp5A/rJHCtwQO18OjpGgx18PzjyGcJDY/+vNgaVyuVd+TgwBgeq6NLdd1XMwRCI+58vinHsAdfA==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/@prisma/engines/-/engines-5.18.0.tgz", + "integrity": "sha512-ofmpGLeJ2q2P0wa/XaEgTnX/IsLnvSp/gZts0zjgLNdBhfuj2lowOOPmDcfKljLQUXMvAek3lw5T01kHmCG8rg==", "devOptional": true, "hasInstallScript": true, "dependencies": { - "@prisma/debug": "5.16.1", - "@prisma/engines-version": "5.16.0-24.34ace0eb2704183d2c05b60b52fba5c43c13f303", - "@prisma/fetch-engine": "5.16.1", - "@prisma/get-platform": "5.16.1" + "@prisma/debug": "5.18.0", + "@prisma/engines-version": "5.18.0-25.4c784e32044a8a016d99474bd02a3b6123742169", + "@prisma/fetch-engine": "5.18.0", + "@prisma/get-platform": "5.18.0" } }, "node_modules/@prisma/engines-version": { - "version": "5.16.0-24.34ace0eb2704183d2c05b60b52fba5c43c13f303", - "resolved": "https://registry.npmjs.org/@prisma/engines-version/-/engines-version-5.16.0-24.34ace0eb2704183d2c05b60b52fba5c43c13f303.tgz", - "integrity": "sha512-HkT2WbfmFZ9WUPyuJHhkiADxazHg8Y4gByrTSVeb3OikP6tjQ7txtSUGu9OBOBH0C13dPKN2qqH12xKtHu/Hiw==", + "version": "5.18.0-25.4c784e32044a8a016d99474bd02a3b6123742169", + "resolved": "https://registry.npmjs.org/@prisma/engines-version/-/engines-version-5.18.0-25.4c784e32044a8a016d99474bd02a3b6123742169.tgz", + "integrity": "sha512-a/+LpJj8vYU3nmtkg+N3X51ddbt35yYrRe8wqHTJtYQt7l1f8kjIBcCs6sHJvodW/EK5XGvboOiwm47fmNrbgg==", "devOptional": true }, "node_modules/@prisma/fetch-engine": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@prisma/fetch-engine/-/fetch-engine-5.16.1.tgz", - "integrity": "sha512-oOkjaPU1lhcA/Rvr4GVfd1NLJBwExgNBE36Ueq7dr71kTMwy++a3U3oLd2ZwrV9dj9xoP6LjCcky799D9nEt4w==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/@prisma/fetch-engine/-/fetch-engine-5.18.0.tgz", + "integrity": "sha512-I/3u0x2n31rGaAuBRx2YK4eB7R/1zCuayo2DGwSpGyrJWsZesrV7QVw7ND0/Suxeo/vLkJ5OwuBqHoCxvTHpOg==", "devOptional": true, "dependencies": { - "@prisma/debug": "5.16.1", - "@prisma/engines-version": "5.16.0-24.34ace0eb2704183d2c05b60b52fba5c43c13f303", - "@prisma/get-platform": "5.16.1" + "@prisma/debug": "5.18.0", + "@prisma/engines-version": "5.18.0-25.4c784e32044a8a016d99474bd02a3b6123742169", + "@prisma/get-platform": "5.18.0" } }, "node_modules/@prisma/get-platform": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/@prisma/get-platform/-/get-platform-5.16.1.tgz", - "integrity": "sha512-R4IKnWnMkR2nUAbU5gjrPehdQYUUd7RENFD2/D+xXTNhcqczp0N+WEGQ3ViyI3+6mtVcjjNIMdnUTNyu3GxIgA==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/@prisma/get-platform/-/get-platform-5.18.0.tgz", + "integrity": "sha512-Tk+m7+uhqcKDgnMnFN0lRiH7Ewea0OEsZZs9pqXa7i3+7svS3FSCqDBCaM9x5fmhhkufiG0BtunJVDka+46DlA==", "devOptional": true, "dependencies": { - "@prisma/debug": "5.16.1" + "@prisma/debug": "5.18.0" } }, "node_modules/@sapphire/async-queue": { - "version": "1.5.2", - "resolved": "https://registry.npmjs.org/@sapphire/async-queue/-/async-queue-1.5.2.tgz", - "integrity": "sha512-7X7FFAA4DngXUl95+hYbUF19bp1LGiffjJtu7ygrZrbdCSsdDDBaSjB7Akw0ZbOu6k0xpXyljnJ6/RZUvLfRdg==", + "version": "1.5.3", + "resolved": "https://registry.npmjs.org/@sapphire/async-queue/-/async-queue-1.5.3.tgz", + "integrity": "sha512-x7zadcfJGxFka1Q3f8gCts1F0xMwCKbZweM85xECGI0hBTeIZJGGCrHgLggihBoprlQ/hBmDR5LKfIPqnmHM3w==", "engines": { "node": ">=v14.0.0", "npm": ">=7.0.0" @@ -1554,9 +1425,9 @@ } }, "node_modules/@tsconfig/node10": { - "version": "1.0.9", - "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.9.tgz", - "integrity": "sha512-jNsYVVxU8v5g43Erja32laIDHXeoNvFEpX33OK4d6hljo3jDhCBDhx5dhCCTMWUojscpAagGiRkBKxpdl9fxqA==", + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.11.tgz", + "integrity": "sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw==", "dev": true }, "node_modules/@tsconfig/node12": { @@ -1659,9 +1530,9 @@ } }, "node_modules/@types/express-serve-static-core": { - "version": "4.17.43", - "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.43.tgz", - "integrity": "sha512-oaYtiBirUOPQGSWNGPWnzyAFJ0BP3cwvN4oWZQY+zUBwpVIGsKUkpBpSztp74drYcjavs7SKFZ4DX1V2QeN8rg==", + "version": "4.19.5", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.19.5.tgz", + "integrity": "sha512-y6W03tvrACO72aijJ5uF02FRq5cgDR9lUxddQ8vyF+GvmjJQqbzDcJngEjURc+ZsG31VI3hODNZJ2URj86pzmg==", "dev": true, "dependencies": { "@types/node": "*", @@ -1732,17 +1603,17 @@ "dev": true }, "node_modules/@types/node": { - "version": "20.14.10", - "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.10.tgz", - "integrity": "sha512-MdiXf+nDuMvY0gJKxyfZ7/6UFsETO7mGKF54MVD/ekJS6HdFtpZFBgrh6Pseu64XTb2MLyFPlbW6hj8HYRQNOQ==", + "version": "20.14.15", + "resolved": "https://registry.npmjs.org/@types/node/-/node-20.14.15.tgz", + "integrity": "sha512-Fz1xDMCF/B00/tYSVMlmK7hVeLh7jE5f3B7X1/hmV0MJBwE27KlS7EvD/Yp+z1lm8mVhwV5w+n8jOZG8AfTlKw==", "dependencies": { "undici-types": "~5.26.4" } }, "node_modules/@types/qs": { - "version": "6.9.12", - "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.12.tgz", - "integrity": "sha512-bZcOkJ6uWrL0Qb2NAWKa7TBU+mJHPzhx9jjLL1KHF+XpzEcR7EXHvjbHlGtR/IsP1vyPrehuS6XqkmaePy//mg==", + "version": "6.9.15", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.15.tgz", + "integrity": "sha512-uXHQKES6DQKKCLh441Xv/dwxOq1TVS3JPUMlEqoEglvlhR6Mxnlew/Xq/LRVHpLyk7iK3zODe1qYHIMltO7XGg==", "dev": true }, "node_modules/@types/range-parser": { @@ -1762,14 +1633,14 @@ } }, "node_modules/@types/serve-static": { - "version": "1.15.5", - "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.5.tgz", - "integrity": "sha512-PDRk21MnK70hja/YF8AHfC7yIsiQHn1rcXx7ijCFBX/k+XQJhQT/gw3xekXKJvx+5SXaMMS8oqQy09Mzvz2TuQ==", + "version": "1.15.7", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.15.7.tgz", + "integrity": "sha512-W8Ym+h8nhuRwaKPaDw34QUkwsGi6Rc4yYqvKFo5rm2FUEhCFbzVWrxXUxuKK8TASjWsysJY0nsmNCGhCOIsrOw==", "dev": true, "dependencies": { "@types/http-errors": "*", - "@types/mime": "*", - "@types/node": "*" + "@types/node": "*", + "@types/send": "*" } }, "node_modules/@types/stack-utils": { @@ -1779,17 +1650,17 @@ "dev": true }, "node_modules/@types/ws": { - "version": "8.5.10", - "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.10.tgz", - "integrity": "sha512-vmQSUcfalpIq0R9q7uTo2lXs6eGIpt9wtnLdMv9LVpIjCA/+ufZRozlVoVelIYixx1ugCBKDhn89vnsEGOCx9A==", + "version": "8.5.12", + "resolved": "https://registry.npmjs.org/@types/ws/-/ws-8.5.12.tgz", + "integrity": "sha512-3tPRkv1EtkDpzlgyKyI8pGsGZAGPEaXeu0DOj5DI25Ja91bdAYddYHbADRYVrZMRbfW+1l5YwXVDKohDJNQxkQ==", "dependencies": { "@types/node": "*" } }, "node_modules/@types/yargs": { - "version": "17.0.32", - "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.32.tgz", - "integrity": "sha512-xQ67Yc/laOG5uMfX/093MRlGGCIBzZMarVa+gfNKJxWAIgykYpVGkBdbqEzGDDfCrVUj6Hiff4mTZ5BA6TmAog==", + "version": "17.0.33", + "resolved": "https://registry.npmjs.org/@types/yargs/-/yargs-17.0.33.tgz", + "integrity": "sha512-WpxBCKWPLr4xSsHgz511rFJAM+wS28w2zEO1QDNY5zM/S8ok70NNfztH0xwhqKyaK0OHCbN98LDAZuy1ctxDkA==", "dev": true, "dependencies": { "@types/yargs-parser": "*" @@ -1801,14 +1672,94 @@ "integrity": "sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ==", "dev": true }, + "node_modules/@typescript-eslint/eslint-plugin": { + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.18.0.tgz", + "integrity": "sha512-94EQTWZ40mzBc42ATNIBimBEDltSJ9RQHCC8vc/PDbxi4k8dVwUAv4o98dk50M1zB+JGFxp43FP7f8+FP8R6Sw==", + "dev": true, + "dependencies": { + "@eslint-community/regexpp": "^4.10.0", + "@typescript-eslint/scope-manager": "7.18.0", + "@typescript-eslint/type-utils": "7.18.0", + "@typescript-eslint/utils": "7.18.0", + "@typescript-eslint/visitor-keys": "7.18.0", + "graphemer": "^1.4.0", + "ignore": "^5.3.1", + "natural-compare": "^1.4.0", + "ts-api-utils": "^1.3.0" + }, + "engines": { + "node": "^18.18.0 || >=20.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "@typescript-eslint/parser": "^7.0.0", + "eslint": "^8.56.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/@typescript-eslint/parser": { + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-7.18.0.tgz", + "integrity": "sha512-4Z+L8I2OqhZV8qA132M4wNL30ypZGYOQVBfMgxDH/K5UX0PNqTu1c6za9ST5r9+tavvHiTWmBnKzpCJ/GlVFtg==", + "dev": true, + "dependencies": { + "@typescript-eslint/scope-manager": "7.18.0", + "@typescript-eslint/types": "7.18.0", + "@typescript-eslint/typescript-estree": "7.18.0", + "@typescript-eslint/visitor-keys": "7.18.0", + "debug": "^4.3.4" + }, + "engines": { + "node": "^18.18.0 || >=20.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.56.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, "node_modules/@typescript-eslint/scope-manager": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-7.15.0.tgz", - "integrity": "sha512-Q/1yrF/XbxOTvttNVPihxh1b9fxamjEoz2Os/Pe38OHwxC24CyCqXxGTOdpb4lt6HYtqw9HetA/Rf6gDGaMPlw==", + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-7.18.0.tgz", + "integrity": "sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA==", + "dev": true, + "dependencies": { + "@typescript-eslint/types": "7.18.0", + "@typescript-eslint/visitor-keys": "7.18.0" + }, + "engines": { + "node": "^18.18.0 || >=20.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + } + }, + "node_modules/@typescript-eslint/type-utils": { + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-7.18.0.tgz", + "integrity": "sha512-XL0FJXuCLaDuX2sYqZUUSOJ2sG5/i1AAze+axqmLnSkNEVMVYLF+cbwlB2w8D1tinFuSikHmFta+P+HOofrLeA==", "dev": true, "dependencies": { - "@typescript-eslint/types": "7.15.0", - "@typescript-eslint/visitor-keys": "7.15.0" + "@typescript-eslint/typescript-estree": "7.18.0", + "@typescript-eslint/utils": "7.18.0", + "debug": "^4.3.4", + "ts-api-utils": "^1.3.0" }, "engines": { "node": "^18.18.0 || >=20.0.0" @@ -1816,12 +1767,20 @@ "funding": { "type": "opencollective", "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.56.0" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } } }, "node_modules/@typescript-eslint/types": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-7.15.0.tgz", - "integrity": "sha512-aV1+B1+ySXbQH0pLK0rx66I3IkiZNidYobyfn0WFsdGhSXw+P3YOqeTq5GED458SfB24tg+ux3S+9g118hjlTw==", + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-7.18.0.tgz", + "integrity": "sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ==", "dev": true, "engines": { "node": "^18.18.0 || >=20.0.0" @@ -1832,13 +1791,13 @@ } }, "node_modules/@typescript-eslint/typescript-estree": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-7.15.0.tgz", - "integrity": "sha512-gjyB/rHAopL/XxfmYThQbXbzRMGhZzGw6KpcMbfe8Q3nNQKStpxnUKeXb0KiN/fFDR42Z43szs6rY7eHk0zdGQ==", + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-7.18.0.tgz", + "integrity": "sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA==", "dev": true, "dependencies": { - "@typescript-eslint/types": "7.15.0", - "@typescript-eslint/visitor-keys": "7.15.0", + "@typescript-eslint/types": "7.18.0", + "@typescript-eslint/visitor-keys": "7.18.0", "debug": "^4.3.4", "globby": "^11.1.0", "is-glob": "^4.0.3", @@ -1868,23 +1827,6 @@ "balanced-match": "^1.0.0" } }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, "node_modules/@typescript-eslint/typescript-estree/node_modules/minimatch": { "version": "9.0.5", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-9.0.5.tgz", @@ -1900,19 +1842,47 @@ "url": "https://github.com/sponsors/isaacs" } }, - "node_modules/@typescript-eslint/typescript-estree/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true + "node_modules/@typescript-eslint/typescript-estree/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@typescript-eslint/utils": { + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-7.18.0.tgz", + "integrity": "sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw==", + "dev": true, + "dependencies": { + "@eslint-community/eslint-utils": "^4.4.0", + "@typescript-eslint/scope-manager": "7.18.0", + "@typescript-eslint/types": "7.18.0", + "@typescript-eslint/typescript-estree": "7.18.0" + }, + "engines": { + "node": "^18.18.0 || >=20.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/typescript-eslint" + }, + "peerDependencies": { + "eslint": "^8.56.0" + } }, "node_modules/@typescript-eslint/visitor-keys": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-7.15.0.tgz", - "integrity": "sha512-Hqgy/ETgpt2L5xueA/zHHIl4fJI2O4XUE9l4+OIfbJIRSnTJb/QscncdqqZzofQegIJugRIF57OJea1khw2SDw==", + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-7.18.0.tgz", + "integrity": "sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg==", "dev": true, "dependencies": { - "@typescript-eslint/types": "7.15.0", + "@typescript-eslint/types": "7.18.0", "eslint-visitor-keys": "^3.4.3" }, "engines": { @@ -1930,20 +1900,14 @@ "dev": true }, "node_modules/@vladfrangu/async_event_emitter": { - "version": "2.4.1", - "resolved": "https://registry.npmjs.org/@vladfrangu/async_event_emitter/-/async_event_emitter-2.4.1.tgz", - "integrity": "sha512-cedU1DrzO4oliUigSAOqSgts6wEfGGSbpO1hYxvKbz8sr7a0meyP3GxnL6hIUtBK0nMG6zHfIYWcqOIb+MRI7w==", + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/@vladfrangu/async_event_emitter/-/async_event_emitter-2.4.5.tgz", + "integrity": "sha512-J7T3gUr3Wz0l7Ni1f9upgBZ7+J22/Q1B7dl0X6fG+fTsD+H+31DIosMHj4Um1dWQwqbcQ3oQf+YS2foYkDc9cQ==", "engines": { "node": ">=v14.0.0", "npm": ">=7.0.0" } }, - "node_modules/abbrev": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", - "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", - "dev": true - }, "node_modules/accepts": { "version": "1.3.8", "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz", @@ -1978,10 +1942,13 @@ } }, "node_modules/acorn-walk": { - "version": "8.3.2", - "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.2.tgz", - "integrity": "sha512-cjkyv4OtNCIeqhHrfS81QWXoCBPExR/J62oyEqepVw8WaQeSqpW2uhuLPh1m9eWhDuOo/jUXVTlifvesOWp/4A==", + "version": "8.3.3", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.3.tgz", + "integrity": "sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==", "dev": true, + "dependencies": { + "acorn": "^8.11.0" + }, "engines": { "node": ">=0.4.0" } @@ -2061,13 +2028,10 @@ "dev": true }, "node_modules/argparse": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", - "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", - "dev": true, - "dependencies": { - "sprintf-js": "~1.0.2" - } + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", + "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", + "dev": true }, "node_modules/array-buffer-byte-length": { "version": "1.0.1", @@ -2197,6 +2161,12 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/async": { + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/async/-/async-3.2.5.tgz", + "integrity": "sha512-baNZyqaaLhyLVKm/DlvdW051MSgO6b8eVfIezl9E5PqWxFgzLm/wQntEW4zOytVburDEr0JlALEpdOFwvErLsg==", + "dev": true + }, "node_modules/available-typed-arrays": { "version": "1.0.7", "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", @@ -2265,15 +2235,6 @@ "node": ">=8" } }, - "node_modules/babel-plugin-istanbul/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", - "dev": true, - "bin": { - "semver": "bin/semver.js" - } - }, "node_modules/babel-plugin-jest-hoist": { "version": "29.6.3", "resolved": "https://registry.npmjs.org/babel-plugin-jest-hoist/-/babel-plugin-jest-hoist-29.6.3.tgz", @@ -2369,6 +2330,30 @@ "npm": "1.2.8000 || >= 1.4.16" } }, + "node_modules/body-parser/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/body-parser/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/body-parser/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, "node_modules/boolbase": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", @@ -2397,9 +2382,9 @@ } }, "node_modules/browserslist": { - "version": "4.23.1", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.1.tgz", - "integrity": "sha512-TUfofFo/KsK/bWZ9TWQ5O26tsWW4Uhmt8IYklbnUa70udB6P2wA7w7o4PY4muaEPBQaAX+CEnmmIA41NVHtPVw==", + "version": "4.23.3", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.23.3.tgz", + "integrity": "sha512-btwCFJVjI4YWDNfau8RhZ+B1Q/VLoUITrm3RlP6y1tYGWIOa+InuYiRGXUBXo8nA1qKmHMyLB/iVQg5TT4eFoA==", "dev": true, "funding": [ { @@ -2416,10 +2401,10 @@ } ], "dependencies": { - "caniuse-lite": "^1.0.30001629", - "electron-to-chromium": "^1.4.796", - "node-releases": "^2.0.14", - "update-browserslist-db": "^1.0.16" + "caniuse-lite": "^1.0.30001646", + "electron-to-chromium": "^1.5.4", + "node-releases": "^2.0.18", + "update-browserslist-db": "^1.1.0" }, "bin": { "browserslist": "cli.js" @@ -2476,6 +2461,18 @@ "semver": "^7.0.0" } }, + "node_modules/builtins/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/bytes": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz", @@ -2521,9 +2518,9 @@ } }, "node_modules/caniuse-lite": { - "version": "1.0.30001640", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001640.tgz", - "integrity": "sha512-lA4VMpW0PSUrFnkmVuEKBUovSWKhj7puyCg8StBChgu298N1AtuF1sKWEvfDuimSEDbhlb/KqPKC3fs1HbuQUA==", + "version": "1.0.30001651", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001651.tgz", + "integrity": "sha512-9Cf+Xv1jJNe1xPZLGuUXLNkE1BoDkqRqYyFJ9TDYSqhduqA4hu4oR9HluGoWYQC/aj8WHjsGVV+bwkh0+tegRg==", "dev": true, "funding": [ { @@ -2556,27 +2553,6 @@ "url": "https://github.com/chalk/chalk?sponsor=1" } }, - "node_modules/chalk/node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/chalk/node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/char-regex": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/char-regex/-/char-regex-1.0.2.tgz", @@ -2587,20 +2563,24 @@ } }, "node_modules/cheerio": { - "version": "1.0.0-rc.12", - "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0-rc.12.tgz", - "integrity": "sha512-VqR8m68vM46BNnuZ5NtnGBKIE/DfN0cRIzg9n40EIq9NOv90ayxLBXA8fXC5gquFRGJSTRqBq25Jt2ECLR431Q==", + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/cheerio/-/cheerio-1.0.0.tgz", + "integrity": "sha512-quS9HgjQpdaXOvsZz82Oz7uxtXiy6UIsIQcpBj7HRw2M63Skasm9qlDocAM7jNuaxdhpPU7c4kJN+gA5MCu4ww==", "dependencies": { "cheerio-select": "^2.1.0", "dom-serializer": "^2.0.0", "domhandler": "^5.0.3", - "domutils": "^3.0.1", - "htmlparser2": "^8.0.1", - "parse5": "^7.0.0", - "parse5-htmlparser2-tree-adapter": "^7.0.0" + "domutils": "^3.1.0", + "encoding-sniffer": "^0.2.0", + "htmlparser2": "^9.1.0", + "parse5": "^7.1.2", + "parse5-htmlparser2-tree-adapter": "^7.0.0", + "parse5-parser-stream": "^7.1.2", + "undici": "^6.19.5", + "whatwg-mimetype": "^4.0.0" }, "engines": { - "node": ">= 6" + "node": ">=18.17" }, "funding": { "url": "https://github.com/cheeriojs/cheerio?sponsor=1" @@ -2646,6 +2626,18 @@ "fsevents": "~2.3.2" } }, + "node_modules/chokidar/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/ci-info": { "version": "3.9.0", "resolved": "https://registry.npmjs.org/ci-info/-/ci-info-3.9.0.tgz", @@ -2886,11 +2878,20 @@ } }, "node_modules/debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "version": "4.3.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.6.tgz", + "integrity": "sha512-O/09Bd4Z1fBrU4VzkhFqVgpPzaGbw6Sm9FEkBT1A/YBXQFGuuSxa1dN2nxgxS34JmKXqYx8CZAwEVoJFImUXIg==", + "dev": true, "dependencies": { - "ms": "2.0.0" + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } } }, "node_modules/dedent": { @@ -3041,6 +3042,14 @@ "url": "https://github.com/discordjs/discord.js?sponsor" } }, + "node_modules/discord.js/node_modules/undici": { + "version": "6.13.0", + "resolved": "https://registry.npmjs.org/undici/-/undici-6.13.0.tgz", + "integrity": "sha512-Q2rtqmZWrbP8nePMq7mOJIN98M0fYvSgV89vwl/BQRT4mDOeY2GXZngfGpcBBhtky3woM7G24wZV3Q304Bv6cw==", + "engines": { + "node": ">=18.0" + } + }, "node_modules/doctrine": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz", @@ -3120,10 +3129,25 @@ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow==" }, + "node_modules/ejs": { + "version": "3.1.10", + "resolved": "https://registry.npmjs.org/ejs/-/ejs-3.1.10.tgz", + "integrity": "sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA==", + "dev": true, + "dependencies": { + "jake": "^10.8.5" + }, + "bin": { + "ejs": "bin/cli.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/electron-to-chromium": { - "version": "1.4.818", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.4.818.tgz", - "integrity": "sha512-eGvIk2V0dGImV9gWLq8fDfTTsCAeMDwZqEPMr+jMInxZdnp9Us8UpovYpRCf9NQ7VOFgrN2doNSgvISbsbNpxA==", + "version": "1.5.6", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.6.tgz", + "integrity": "sha512-jwXWsM5RPf6j9dPYzaorcBSUg6AiqocPEyMpkchkvntaH9HGfOOMZwxMJjDY/XEs3T5dM7uyH1VhRMkqUU9qVw==", "dev": true }, "node_modules/emittery": { @@ -3152,6 +3176,18 @@ "node": ">= 0.8" } }, + "node_modules/encoding-sniffer": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/encoding-sniffer/-/encoding-sniffer-0.2.0.tgz", + "integrity": "sha512-ju7Wq1kg04I3HtiYIOrUrdfdDvkyO9s5XM8QAj/bN61Yo/Vb4vgJxy5vi4Yxk01gWHbrofpPtpxM8bKger9jhg==", + "dependencies": { + "iconv-lite": "^0.6.3", + "whatwg-encoding": "^3.1.1" + }, + "funding": { + "url": "https://github.com/fb55/encoding-sniffer?sponsor=1" + } + }, "node_modules/entities": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/entities/-/entities-4.5.0.tgz", @@ -3328,13 +3364,16 @@ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow==" }, - "node_modules/escape-string-regexp": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", - "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "node_modules/escape-string-regexp": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", "dev": true, "engines": { - "node": ">=8" + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, "node_modules/eslint": { @@ -3407,6 +3446,18 @@ "eslint": ">=6.0.0" } }, + "node_modules/eslint-compat-utils/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/eslint-config-standard": { "version": "17.1.0", "resolved": "https://registry.npmjs.org/eslint-config-standard/-/eslint-config-standard-17.1.0.tgz", @@ -3456,12 +3507,6 @@ "ms": "^2.1.1" } }, - "node_modules/eslint-import-resolver-node/node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true - }, "node_modules/eslint-module-utils": { "version": "2.8.1", "resolved": "https://registry.npmjs.org/eslint-module-utils/-/eslint-module-utils-2.8.1.tgz", @@ -3488,12 +3533,6 @@ "ms": "^2.1.1" } }, - "node_modules/eslint-module-utils/node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true - }, "node_modules/eslint-plugin-es-x": { "version": "7.8.0", "resolved": "https://registry.npmjs.org/eslint-plugin-es-x/-/eslint-plugin-es-x-7.8.0.tgz", @@ -3567,19 +3606,37 @@ "node": ">=0.10.0" } }, - "node_modules/eslint-plugin-import/node_modules/ms": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", - "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==", - "dev": true - }, - "node_modules/eslint-plugin-import/node_modules/semver": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", - "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", + "node_modules/eslint-plugin-import/node_modules/json5": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", + "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", "dev": true, + "dependencies": { + "minimist": "^1.2.0" + }, "bin": { - "semver": "bin/semver.js" + "json5": "lib/cli.js" + } + }, + "node_modules/eslint-plugin-import/node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/eslint-plugin-import/node_modules/tsconfig-paths": { + "version": "3.15.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", + "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", + "dev": true, + "dependencies": { + "@types/json5": "^0.0.29", + "json5": "^1.0.2", + "minimist": "^1.2.6", + "strip-bom": "^3.0.0" } }, "node_modules/eslint-plugin-n": { @@ -3625,6 +3682,18 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/eslint-plugin-n/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/eslint-plugin-n/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -3638,9 +3707,9 @@ } }, "node_modules/eslint-plugin-promise": { - "version": "6.4.0", - "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-6.4.0.tgz", - "integrity": "sha512-/KWWRaD3fGkVCZsdR0RU53PSthFmoHVhZl+y9+6DqeDLSikLdlUVpVEAmI6iCRR5QyOjBYBqHZV/bdv4DJ4Gtw==", + "version": "6.6.0", + "resolved": "https://registry.npmjs.org/eslint-plugin-promise/-/eslint-plugin-promise-6.6.0.tgz", + "integrity": "sha512-57Zzfw8G6+Gq7axm2Pdo3gW/Rx3h9Yywgn61uE/3elTCOePEHVrn2i5CdfBwA1BLK0Q0WqctICIUSqXZW/VprQ==", "dev": true, "engines": { "node": "^12.22.0 || ^14.17.0 || >=16.0.0" @@ -3719,69 +3788,6 @@ "node": "^12.22.0 || ^14.17.0 || >=16.0.0" } }, - "node_modules/eslint/node_modules/argparse": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz", - "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==", - "dev": true - }, - "node_modules/eslint/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/eslint/node_modules/escape-string-regexp": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint/node_modules/find-up": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", - "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", - "dev": true, - "dependencies": { - "locate-path": "^6.0.0", - "path-exists": "^4.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint/node_modules/glob-parent": { - "version": "6.0.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", - "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", - "dev": true, - "dependencies": { - "is-glob": "^4.0.3" - }, - "engines": { - "node": ">=10.13.0" - } - }, "node_modules/eslint/node_modules/globals": { "version": "13.24.0", "resolved": "https://registry.npmjs.org/globals/-/globals-13.24.0.tgz", @@ -3797,54 +3803,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/eslint/node_modules/js-yaml": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", - "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", - "dev": true, - "dependencies": { - "argparse": "^2.0.1" - }, - "bin": { - "js-yaml": "bin/js-yaml.js" - } - }, - "node_modules/eslint/node_modules/locate-path": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", - "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", - "dev": true, - "dependencies": { - "p-locate": "^5.0.0" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, - "node_modules/eslint/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, - "node_modules/eslint/node_modules/p-locate": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", - "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", - "dev": true, - "dependencies": { - "p-limit": "^3.0.2" - }, - "engines": { - "node": ">=10" - }, - "funding": { - "url": "https://github.com/sponsors/sindresorhus" - } - }, "node_modules/eslint/node_modules/type-fest": { "version": "0.20.2", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.20.2.tgz", @@ -3888,9 +3846,9 @@ } }, "node_modules/esquery": { - "version": "1.5.0", - "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.5.0.tgz", - "integrity": "sha512-YQLXUplAwJgCydQ78IMJywZCceoqk1oH01OERdSAJc/7U2AylwjhSCLDEtqwg811idIS/9fIU5GjG73IgjKMVg==", + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.6.0.tgz", + "integrity": "sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg==", "dev": true, "dependencies": { "estraverse": "^5.1.0" @@ -4026,6 +3984,19 @@ "node": ">= 0.10.0" } }, + "node_modules/express/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/express/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, "node_modules/fast-deep-equal": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", @@ -4047,6 +4018,18 @@ "node": ">=8.6.0" } }, + "node_modules/fast-glob/node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dev": true, + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", @@ -4089,18 +4072,34 @@ "node": "^10.12.0 || >=12.0.0" } }, - "node_modules/file-entry-cache/node_modules/flat-cache": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", - "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "node_modules/filelist": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/filelist/-/filelist-1.0.4.tgz", + "integrity": "sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q==", "dev": true, "dependencies": { - "flatted": "^3.2.9", - "keyv": "^4.5.3", - "rimraf": "^3.0.2" + "minimatch": "^5.0.1" + } + }, + "node_modules/filelist/node_modules/brace-expansion": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-2.0.1.tgz", + "integrity": "sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0" + } + }, + "node_modules/filelist/node_modules/minimatch": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-5.1.6.tgz", + "integrity": "sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g==", + "dev": true, + "dependencies": { + "brace-expansion": "^2.0.1" }, "engines": { - "node": "^10.12.0 || >=12.0.0" + "node": ">=10" } }, "node_modules/fill-range": { @@ -4132,17 +4131,47 @@ "node": ">= 0.8" } }, + "node_modules/finalhandler/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/finalhandler/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, "node_modules/find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", + "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", "dev": true, "dependencies": { - "locate-path": "^5.0.0", + "locate-path": "^6.0.0", "path-exists": "^4.0.0" }, "engines": { - "node": ">=8" + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/flat-cache": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/flat-cache/-/flat-cache-3.2.0.tgz", + "integrity": "sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw==", + "dev": true, + "dependencies": { + "flatted": "^3.2.9", + "keyv": "^4.5.3", + "rimraf": "^3.0.2" + }, + "engines": { + "node": "^10.12.0 || >=12.0.0" } }, "node_modules/flatted": { @@ -4169,9 +4198,9 @@ } }, "node_modules/fp-ts": { - "version": "2.16.7", - "resolved": "https://registry.npmjs.org/fp-ts/-/fp-ts-2.16.7.tgz", - "integrity": "sha512-Xiux+4mHHPj32/mrqN3XIIqEKk/MousgoC2FIaCwehpPjBI4oDrLvQEyQ/2T1sbTe0s/YIQqF98z+uHJLVoS9Q==" + "version": "2.16.9", + "resolved": "https://registry.npmjs.org/fp-ts/-/fp-ts-2.16.9.tgz", + "integrity": "sha512-+I2+FnVB+tVaxcYyQkHUq7ZdKScaBlX53A41mxQtpIccsfyv8PzdzP7fzp2AY832T4aoK6UZ5WRX/ebGd8uZuQ==" }, "node_modules/fresh": { "version": "0.5.2", @@ -4311,9 +4340,9 @@ } }, "node_modules/get-tsconfig": { - "version": "4.7.5", - "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.7.5.tgz", - "integrity": "sha512-ZCuZCnlqNzjb4QprAzXKdpp/gh6KTxSJuw3IBsPnV/7fV4NxC9ckB+vPTt8w7fJA0TaSD7c55BR47JD6MEDyDw==", + "version": "4.7.6", + "resolved": "https://registry.npmjs.org/get-tsconfig/-/get-tsconfig-4.7.6.tgz", + "integrity": "sha512-ZAqrLlu18NbDdRaHq+AKXzAmqIUPswPWKUchfytdAjiRFnCe5ojG2bstg6mRiZabkKfCoL/e98pbBELIV/YCeA==", "dev": true, "dependencies": { "resolve-pkg-maps": "^1.0.0" @@ -4344,21 +4373,21 @@ } }, "node_modules/glob-parent": { - "version": "5.1.2", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", - "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz", + "integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==", "dev": true, "dependencies": { - "is-glob": "^4.0.1" + "is-glob": "^4.0.3" }, "engines": { - "node": ">= 6" + "node": ">=10.13.0" } }, "node_modules/globals": { - "version": "15.8.0", - "resolved": "https://registry.npmjs.org/globals/-/globals-15.8.0.tgz", - "integrity": "sha512-VZAJ4cewHTExBWDHR6yptdIBlx9YSSZuwojj9Nt5mBRXQzrKakDsVKQ1J63sklLvzAJm0X5+RpO4i3Y2hcOnFw==", + "version": "15.9.0", + "resolved": "https://registry.npmjs.org/globals/-/globals-15.9.0.tgz", + "integrity": "sha512-SmSKyLLKFbSr6rptvP8izbyxJL4ILwqO9Jg23UA0sDlGlu58V59D1//I3vlc0KJphVdUR7vMjHIplYnzBxorQA==", "dev": true, "engines": { "node": ">=18" @@ -4436,12 +4465,12 @@ } }, "node_modules/has-flag": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", - "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", "dev": true, "engines": { - "node": ">=4" + "node": ">=8" } }, "node_modules/has-property-descriptors": { @@ -4510,9 +4539,9 @@ "dev": true }, "node_modules/htmlparser2": { - "version": "8.0.2", - "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-8.0.2.tgz", - "integrity": "sha512-GYdjWKDkbRLkZ5geuHs5NY1puJ+PXwP7+fHPRz06Eirsb9ugf6d8kkXav6ADhcODhFFPMIXyxkxSuMf3D6NCFA==", + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-9.1.0.tgz", + "integrity": "sha512-5zfg6mHUoaer/97TxnGpxmbR7zJtPwIYFMZ/H5ucTlPZhKvtum05yiPK3Mgai3a0DyVxv7qYqoweaEd2nrYQzQ==", "funding": [ "https://github.com/fb55/htmlparser2?sponsor=1", { @@ -4523,8 +4552,8 @@ "dependencies": { "domelementtype": "^2.3.0", "domhandler": "^5.0.3", - "domutils": "^3.0.1", - "entities": "^4.4.0" + "domutils": "^3.1.0", + "entities": "^4.5.0" } }, "node_modules/http-errors": { @@ -4552,20 +4581,20 @@ } }, "node_modules/iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", "dependencies": { - "safer-buffer": ">= 2.1.2 < 3" + "safer-buffer": ">= 2.1.2 < 3.0.0" }, "engines": { "node": ">=0.10.0" } }, "node_modules/ignore": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz", - "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", + "version": "5.3.2", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.2.tgz", + "integrity": "sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g==", "dev": true, "engines": { "node": ">= 4" @@ -4593,19 +4622,10 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/import-fresh/node_modules/resolve-from": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", - "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", - "dev": true, - "engines": { - "node": ">=4" - } - }, "node_modules/import-local": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.1.0.tgz", - "integrity": "sha512-ASB07uLtnDs1o6EHjKpX34BKYDSqnFerfTOJL2HvMqF70LnxpjkzDB8J44oT9pu4AMPkQwf8jl6szgvNd2tRIg==", + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-3.2.0.tgz", + "integrity": "sha512-2SPlun1JUPWoM6t3F0dw0FkCF/jWY8kttcY4f599GLTSjh2OCuuhdTkJQsEcZzBqbXZGKMK2OqW1oZsjtf/gQA==", "dev": true, "dependencies": { "pkg-dir": "^4.2.0", @@ -4758,9 +4778,9 @@ } }, "node_modules/is-core-module": { - "version": "2.14.0", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.14.0.tgz", - "integrity": "sha512-a5dFJih5ZLYlRtDc0dZWP7RiKr6xIKzmn/oAYCDvdLThadVgyJwlaoQPmRtMSpz+rk0OGAgIu+TcM9HUF0fk1A==", + "version": "2.15.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.15.0.tgz", + "integrity": "sha512-Dd+Lb2/zvk9SKy1TGCt1wFJFo/MWBPMX5x7KcvLajWTGuomczdQX61PvY5yK6SVACwpoexWo81IfFyoKY2QnTA==", "dev": true, "dependencies": { "hasown": "^2.0.2" @@ -5023,6 +5043,18 @@ "node": ">=10" } }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/istanbul-lib-report": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.1.tgz", @@ -5037,27 +5069,6 @@ "node": ">=10" } }, - "node_modules/istanbul-lib-report/node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, - "node_modules/istanbul-lib-report/node_modules/supports-color": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", - "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", - "dev": true, - "dependencies": { - "has-flag": "^4.0.0" - }, - "engines": { - "node": ">=8" - } - }, "node_modules/istanbul-lib-source-maps": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-4.0.1.tgz", @@ -5072,29 +5083,6 @@ "node": ">=10" } }, - "node_modules/istanbul-lib-source-maps/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/istanbul-lib-source-maps/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, "node_modules/istanbul-reports": { "version": "3.1.7", "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.1.7.tgz", @@ -5108,6 +5096,24 @@ "node": ">=8" } }, + "node_modules/jake": { + "version": "10.9.2", + "resolved": "https://registry.npmjs.org/jake/-/jake-10.9.2.tgz", + "integrity": "sha512-2P4SQ0HrLQ+fw6llpLnOaGAvN2Zu6778SJMrCUwns4fOoG9ayrTiZk3VV8sCPkVZF8ab0zksVpS8FDY5pRCNBA==", + "dev": true, + "dependencies": { + "async": "^3.2.3", + "chalk": "^4.0.2", + "filelist": "^1.0.4", + "minimatch": "^3.1.2" + }, + "bin": { + "jake": "bin/cli.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/jest": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest/-/jest-29.7.0.tgz", @@ -5568,6 +5574,18 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, + "node_modules/jest-snapshot/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/jest-util": { "version": "29.7.0", "resolved": "https://registry.npmjs.org/jest-util/-/jest-util-29.7.0.tgz", @@ -5648,15 +5666,6 @@ "node": "^14.15.0 || ^16.10.0 || >=18.0.0" } }, - "node_modules/jest-worker/node_modules/has-flag": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", - "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", - "dev": true, - "engines": { - "node": ">=8" - } - }, "node_modules/jest-worker/node_modules/supports-color": { "version": "8.1.1", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-8.1.1.tgz", @@ -5679,13 +5688,12 @@ "dev": true }, "node_modules/js-yaml": { - "version": "3.14.1", - "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.1.tgz", - "integrity": "sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz", + "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==", "dev": true, "dependencies": { - "argparse": "^1.0.7", - "esprima": "^4.0.0" + "argparse": "^2.0.1" }, "bin": { "js-yaml": "bin/js-yaml.js" @@ -5786,15 +5794,18 @@ "dev": true }, "node_modules/locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", + "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", "dev": true, "dependencies": { - "p-locate": "^4.1.0" + "p-locate": "^5.0.0" }, "engines": { - "node": ">=8" + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, "node_modules/lodash": { @@ -5820,15 +5831,12 @@ "integrity": "sha512-QZ1d4xoBHYUeuouhEq3lk3Uq7ldgyFXGBhg04+oRLnIz8o9T65Eh+8YdroUwn846zchkA9yDsDl5CVVaV2nqYw==" }, "node_modules/lru-cache": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", - "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", "dev": true, "dependencies": { - "yallist": "^4.0.0" - }, - "engines": { - "node": ">=10" + "yallist": "^3.0.2" } }, "node_modules/magic-bytes.js": { @@ -5851,6 +5859,18 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/make-dir/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/make-error": { "version": "1.3.6", "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", @@ -5866,6 +5886,10 @@ "tmpl": "1.0.5" } }, + "node_modules/mcv-discord-bot": { + "resolved": "", + "link": true + }, "node_modules/media-typer": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", @@ -5976,9 +6000,10 @@ } }, "node_modules/ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", + "dev": true }, "node_modules/natural-compare": { "version": "1.4.0", @@ -6012,9 +6037,9 @@ "dev": true }, "node_modules/node-releases": { - "version": "2.0.14", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.14.tgz", - "integrity": "sha512-y10wOWt8yZpqXmOgRo77WaHEmhYQYGNA6y421PKsKYWEK8aW+cqAphborZDhqfyKrbZEN92CN1X2KbafY2s7Yw==", + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.18.tgz", + "integrity": "sha512-d9VeXT4SJ7ZeOqGX6R5EM022wpL+eWPooLI+5UpWn2jCT1aosUQEhQP214x33Wkwx3JQMvIm+tIoVOdodFS40g==", "dev": true }, "node_modules/nodemon": { @@ -6045,42 +6070,37 @@ "url": "https://opencollective.com/nodemon" } }, - "node_modules/nodemon/node_modules/debug": { - "version": "4.3.4", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", - "integrity": "sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ==", + "node_modules/nodemon/node_modules/has-flag": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", + "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", "dev": true, - "dependencies": { - "ms": "2.1.2" - }, "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } + "node": ">=4" } }, - "node_modules/nodemon/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true + "node_modules/nodemon/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } }, - "node_modules/nopt": { - "version": "1.0.10", - "resolved": "https://registry.npmjs.org/nopt/-/nopt-1.0.10.tgz", - "integrity": "sha512-NWmpvLSqUrgrAC9HCuxEvb+PSloHpqVu+FqcO4eeF2h5qYRhA7ev6KvelyQAKtegUbC6RypJnlEOhd8vloNKYg==", + "node_modules/nodemon/node_modules/supports-color": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", + "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", "dev": true, "dependencies": { - "abbrev": "1" - }, - "bin": { - "nopt": "bin/nopt.js" + "has-flag": "^3.0.0" }, "engines": { - "node": "*" + "node": ">=4" } }, "node_modules/normalize-path": { @@ -6116,9 +6136,12 @@ } }, "node_modules/object-inspect": { - "version": "1.13.1", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", - "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==", + "version": "1.13.2", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.2.tgz", + "integrity": "sha512-IRZSRuzJiynemAXPYtPe5BoI/RESNYR7TYm50MC5Mqbd3Jmw5y790sErYw3V6SryFJD64b74qQQs9wn5Bg/k3g==", + "engines": { + "node": ">= 0.4" + }, "funding": { "url": "https://github.com/sponsors/ljharb" } @@ -6267,27 +6290,15 @@ } }, "node_modules/p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "dependencies": { - "p-limit": "^2.2.0" - }, - "engines": { - "node": ">=8" - } - }, - "node_modules/p-locate/node_modules/p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", + "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", "dev": true, "dependencies": { - "p-try": "^2.0.0" + "p-limit": "^3.0.2" }, "engines": { - "node": ">=6" + "node": ">=10" }, "funding": { "url": "https://github.com/sponsors/sindresorhus" @@ -6355,6 +6366,17 @@ "url": "https://github.com/inikulin/parse5?sponsor=1" } }, + "node_modules/parse5-parser-stream": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/parse5-parser-stream/-/parse5-parser-stream-7.1.2.tgz", + "integrity": "sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow==", + "dependencies": { + "parse5": "^7.0.0" + }, + "funding": { + "url": "https://github.com/inikulin/parse5?sponsor=1" + } + }, "node_modules/parseurl": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", @@ -6421,29 +6443,81 @@ "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.1.tgz", "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", "dev": true, - "engines": { - "node": ">=8.6" + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pirates": { + "version": "4.0.6", + "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", + "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pkg-dir/node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" }, - "funding": { - "url": "https://github.com/sponsors/jonschlinkert" + "engines": { + "node": ">=8" } }, - "node_modules/pirates": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.6.tgz", - "integrity": "sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==", + "node_modules/pkg-dir/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, "engines": { - "node": ">= 6" + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "node_modules/pkg-dir/node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", "dev": true, "dependencies": { - "find-up": "^4.0.0" + "p-limit": "^2.2.0" }, "engines": { "node": ">=8" @@ -6468,9 +6542,9 @@ } }, "node_modules/prettier": { - "version": "3.3.2", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.2.tgz", - "integrity": "sha512-rAVeHYMcv8ATV5d508CFdn+8/pHPpXeIid1DdrPwXnaAdH7cqjVbpJaT5eq4yRAFU/lsbwYwSF/n5iNrdJHPQA==", + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.3.3.tgz", + "integrity": "sha512-i2tDNA0O5IrMO757lfrdQZCc2jPNDVntV0m/+4whiDfWaTKfMNgR7Qz0NAeGz/nRqF4m5/6CLzbP4/liHt12Ew==", "dev": true, "bin": { "prettier": "bin/prettier.cjs" @@ -6509,13 +6583,13 @@ } }, "node_modules/prisma": { - "version": "5.16.1", - "resolved": "https://registry.npmjs.org/prisma/-/prisma-5.16.1.tgz", - "integrity": "sha512-Z1Uqodk44diztImxALgJJfNl2Uisl9xDRvqybMKEBYJLNKNhDfAHf+ZIJbZyYiBhLMbKU9cYGdDVG5IIXEnL2Q==", + "version": "5.18.0", + "resolved": "https://registry.npmjs.org/prisma/-/prisma-5.18.0.tgz", + "integrity": "sha512-+TrSIxZsh64OPOmaSgVPH7ALL9dfU0jceYaMJXsNrTkFHO7/3RANi5K2ZiPB1De9+KDxCWn7jvRq8y8pvk+o9g==", "devOptional": true, "hasInstallScript": true, "dependencies": { - "@prisma/engines": "5.16.1" + "@prisma/engines": "5.18.0" }, "bin": { "prisma": "build/index.js" @@ -6636,6 +6710,17 @@ "node": ">= 0.8" } }, + "node_modules/raw-body/node_modules/iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/react-is": { "version": "18.3.1", "resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz", @@ -6710,7 +6795,7 @@ "node": ">=8" } }, - "node_modules/resolve-from": { + "node_modules/resolve-cwd/node_modules/resolve-from": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-5.0.0.tgz", "integrity": "sha512-qYg9KP24dD5qka9J47d0aVky0N+b4fTU89LN9iDnjB5waksiC49rvMB0PrUJQGoTmH50XPiqOvAjDfaijGxYZw==", @@ -6719,6 +6804,15 @@ "node": ">=8" } }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, "node_modules/resolve-pkg-maps": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz", @@ -6846,18 +6940,12 @@ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" }, "node_modules/semver": { - "version": "7.6.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", - "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", + "version": "6.3.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", + "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", "dev": true, - "dependencies": { - "lru-cache": "^6.0.0" - }, "bin": { "semver": "bin/semver.js" - }, - "engines": { - "node": ">=10" } }, "node_modules/send": { @@ -6883,6 +6971,19 @@ "node": ">= 0.8.0" } }, + "node_modules/send/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/send/node_modules/debug/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A==" + }, "node_modules/send/node_modules/ms": { "version": "2.1.3", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", @@ -6994,6 +7095,18 @@ "node": ">=10" } }, + "node_modules/simple-update-notifier/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/sisteransi": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/sisteransi/-/sisteransi-1.0.5.tgz", @@ -7046,6 +7159,15 @@ "node": ">=10" } }, + "node_modules/stack-utils/node_modules/escape-string-regexp": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz", + "integrity": "sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, "node_modules/statuses": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz", @@ -7173,15 +7295,15 @@ } }, "node_modules/supports-color": { - "version": "5.5.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", - "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", "dev": true, "dependencies": { - "has-flag": "^3.0.0" + "has-flag": "^4.0.0" }, "engines": { - "node": ">=4" + "node": ">=8" } }, "node_modules/supports-preserve-symlinks-flag": { @@ -7252,13 +7374,10 @@ } }, "node_modules/touch": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.0.tgz", - "integrity": "sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA==", + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/touch/-/touch-3.1.1.tgz", + "integrity": "sha512-r0eojU4bI8MnHr8c5bNo7lJDdI2qXlWWJk6a9EAFG7vbhTjElYhBVS3/miuE0uOuoLdb8Mc/rVfsmm6eo5o9GA==", "dev": true, - "dependencies": { - "nopt": "~1.0.10" - }, "bin": { "nodetouch": "bin/nodetouch.js" } @@ -7276,12 +7395,13 @@ } }, "node_modules/ts-jest": { - "version": "29.1.5", - "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.1.5.tgz", - "integrity": "sha512-UuClSYxM7byvvYfyWdFI+/2UxMmwNyJb0NPkZPQE2hew3RurV7l7zURgOHAd/1I1ZdPpe3GUsXNXAcN8TFKSIg==", + "version": "29.2.4", + "resolved": "https://registry.npmjs.org/ts-jest/-/ts-jest-29.2.4.tgz", + "integrity": "sha512-3d6tgDyhCI29HlpwIq87sNuI+3Q6GLTTCeYRHCs7vDz+/3GCMwEtV9jezLyl4ZtnBgx00I7hm8PCP8cTksMGrw==", "dev": true, "dependencies": { "bs-logger": "0.x", + "ejs": "^3.1.10", "fast-json-stable-stringify": "2.x", "jest-util": "^29.0.0", "json5": "^2.2.3", @@ -7322,6 +7442,18 @@ } } }, + "node_modules/ts-jest/node_modules/semver": { + "version": "7.6.3", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", + "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, "node_modules/ts-mixer": { "version": "6.0.4", "resolved": "https://registry.npmjs.org/ts-mixer/-/ts-mixer-6.0.4.tgz", @@ -7371,27 +7503,17 @@ } }, "node_modules/tsconfig-paths": { - "version": "3.15.0", - "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz", - "integrity": "sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/tsconfig-paths/-/tsconfig-paths-4.2.0.tgz", + "integrity": "sha512-NoZ4roiN7LnbKn9QqE1amc9DJfzvZXxF4xDavcOWt1BPkdx+m+0gJuPM+S0vCe7zTJMYUP0R8pO2XMr+Y8oLIg==", "dev": true, "dependencies": { - "@types/json5": "^0.0.29", - "json5": "^1.0.2", + "json5": "^2.2.2", "minimist": "^1.2.6", "strip-bom": "^3.0.0" - } - }, - "node_modules/tsconfig-paths/node_modules/json5": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.2.tgz", - "integrity": "sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==", - "dev": true, - "dependencies": { - "minimist": "^1.2.0" }, - "bin": { - "json5": "lib/cli.js" + "engines": { + "node": ">=6" } }, "node_modules/tsconfig-paths/node_modules/strip-bom": { @@ -7527,9 +7649,9 @@ } }, "node_modules/typescript": { - "version": "5.5.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.5.3.tgz", - "integrity": "sha512-/hreyEujaB0w76zKo6717l3L0o/qEUtRgdvUBvlkhoWeOVMjMuHNHk0BRBzikzuGDqNmPQbg5ifMEqsHLiIUcQ==", + "version": "5.5.4", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.5.4.tgz", + "integrity": "sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q==", "dev": true, "bin": { "tsc": "bin/tsc", @@ -7540,14 +7662,14 @@ } }, "node_modules/typescript-eslint": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-7.15.0.tgz", - "integrity": "sha512-Ta40FhMXBCwHura4X4fncaCVkVcnJ9jnOq5+Lp4lN8F4DzHZtOwZdRvVBiNUGznUDHPwdGnrnwxmUOU2fFQqFA==", + "version": "7.18.0", + "resolved": "https://registry.npmjs.org/typescript-eslint/-/typescript-eslint-7.18.0.tgz", + "integrity": "sha512-PonBkP603E3tt05lDkbOMyaxJjvKqQrXsnow72sVeOFINDE/qNmnnd+f9b4N+U7W6MXnnYyrhtmF2t08QWwUbA==", "dev": true, "dependencies": { - "@typescript-eslint/eslint-plugin": "7.15.0", - "@typescript-eslint/parser": "7.15.0", - "@typescript-eslint/utils": "7.15.0" + "@typescript-eslint/eslint-plugin": "7.18.0", + "@typescript-eslint/parser": "7.18.0", + "@typescript-eslint/utils": "7.18.0" }, "engines": { "node": "^18.18.0 || >=20.0.0" @@ -7565,139 +7687,6 @@ } } }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/eslint-plugin": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.15.0.tgz", - "integrity": "sha512-uiNHpyjZtFrLwLDpHnzaDlP3Tt6sGMqTCiqmxaN4n4RP0EfYZDODJyddiFDF44Hjwxr5xAcaYxVKm9QKQFJFLA==", - "dev": true, - "dependencies": { - "@eslint-community/regexpp": "^4.10.0", - "@typescript-eslint/scope-manager": "7.15.0", - "@typescript-eslint/type-utils": "7.15.0", - "@typescript-eslint/utils": "7.15.0", - "@typescript-eslint/visitor-keys": "7.15.0", - "graphemer": "^1.4.0", - "ignore": "^5.3.1", - "natural-compare": "^1.4.0", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || >=20.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "@typescript-eslint/parser": "^7.0.0", - "eslint": "^8.56.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/eslint-plugin/node_modules/@typescript-eslint/type-utils": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-7.15.0.tgz", - "integrity": "sha512-SkgriaeV6PDvpA6253PDVep0qCqgbO1IOBiycjnXsszNTVQe5flN5wR5jiczoEoDEnAqYFSFFc9al9BSGVltkg==", - "dev": true, - "dependencies": { - "@typescript-eslint/typescript-estree": "7.15.0", - "@typescript-eslint/utils": "7.15.0", - "debug": "^4.3.4", - "ts-api-utils": "^1.3.0" - }, - "engines": { - "node": "^18.18.0 || >=20.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.56.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/parser": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-7.15.0.tgz", - "integrity": "sha512-k9fYuQNnypLFcqORNClRykkGOMOj+pV6V91R4GO/l1FDGwpqmSwoOQrOHo3cGaH63e+D3ZiCAOsuS/D2c99j/A==", - "dev": true, - "dependencies": { - "@typescript-eslint/scope-manager": "7.15.0", - "@typescript-eslint/types": "7.15.0", - "@typescript-eslint/typescript-estree": "7.15.0", - "@typescript-eslint/visitor-keys": "7.15.0", - "debug": "^4.3.4" - }, - "engines": { - "node": "^18.18.0 || >=20.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.56.0" - }, - "peerDependenciesMeta": { - "typescript": { - "optional": true - } - } - }, - "node_modules/typescript-eslint/node_modules/@typescript-eslint/utils": { - "version": "7.15.0", - "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-7.15.0.tgz", - "integrity": "sha512-hfDMDqaqOqsUVGiEPSMLR/AjTSCsmJwjpKkYQRo1FNbmW4tBwBspYDwO9eh7sKSTwMQgBw9/T4DHudPaqshRWA==", - "dev": true, - "dependencies": { - "@eslint-community/eslint-utils": "^4.4.0", - "@typescript-eslint/scope-manager": "7.15.0", - "@typescript-eslint/types": "7.15.0", - "@typescript-eslint/typescript-estree": "7.15.0" - }, - "engines": { - "node": "^18.18.0 || >=20.0.0" - }, - "funding": { - "type": "opencollective", - "url": "https://opencollective.com/typescript-eslint" - }, - "peerDependencies": { - "eslint": "^8.56.0" - } - }, - "node_modules/typescript-eslint/node_modules/debug": { - "version": "4.3.5", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.5.tgz", - "integrity": "sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg==", - "dev": true, - "dependencies": { - "ms": "2.1.2" - }, - "engines": { - "node": ">=6.0" - }, - "peerDependenciesMeta": { - "supports-color": { - "optional": true - } - } - }, - "node_modules/typescript-eslint/node_modules/ms": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", - "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", - "dev": true - }, "node_modules/unbox-primitive": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/unbox-primitive/-/unbox-primitive-1.0.2.tgz", @@ -7720,11 +7709,11 @@ "dev": true }, "node_modules/undici": { - "version": "6.13.0", - "resolved": "https://registry.npmjs.org/undici/-/undici-6.13.0.tgz", - "integrity": "sha512-Q2rtqmZWrbP8nePMq7mOJIN98M0fYvSgV89vwl/BQRT4mDOeY2GXZngfGpcBBhtky3woM7G24wZV3Q304Bv6cw==", + "version": "6.19.7", + "resolved": "https://registry.npmjs.org/undici/-/undici-6.19.7.tgz", + "integrity": "sha512-HR3W/bMGPSr90i8AAp2C4DM3wChFdJPLrWYpIS++LxS8K+W535qftjt+4MyjNYHeWabMj1nvtmLIi7l++iq91A==", "engines": { - "node": ">=18.0" + "node": ">=18.17" } }, "node_modules/undici-types": { @@ -7807,16 +7796,6 @@ "node": ">=10.12.0" } }, - "node_modules/v8-to-istanbul/node_modules/@jridgewell/trace-mapping": { - "version": "0.3.25", - "resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz", - "integrity": "sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ==", - "dev": true, - "dependencies": { - "@jridgewell/resolve-uri": "^3.1.0", - "@jridgewell/sourcemap-codec": "^1.4.14" - } - }, "node_modules/vary": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", @@ -7834,6 +7813,25 @@ "makeerror": "1.0.12" } }, + "node_modules/whatwg-encoding": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz", + "integrity": "sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ==", + "dependencies": { + "iconv-lite": "0.6.3" + }, + "engines": { + "node": ">=18" + } + }, + "node_modules/whatwg-mimetype": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz", + "integrity": "sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg==", + "engines": { + "node": ">=18" + } + }, "node_modules/which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -7959,9 +7957,9 @@ } }, "node_modules/yallist": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", - "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", "dev": true }, "node_modules/yargs": { diff --git a/package.json b/package.json index eda5b5b..a671147 100644 --- a/package.json +++ b/package.json @@ -7,6 +7,7 @@ "envalid": "^8.0.0", "express": "^4.18.2", "fp-ts": "^2.16.7", + "mcv-discord-bot": "file:", "node-cache": "^5.1.2", "zod": "^3.23.8" }, @@ -29,6 +30,7 @@ "prisma": "^5.16.1", "ts-jest": "^29.1.5", "ts-node": "^10.9.2", + "tsconfig-paths": "^4.2.0", "typescript": "^5.5.3", "typescript-eslint": "^7.15.0" }, @@ -39,7 +41,8 @@ "dev": "prisma generate && prisma db push && nodemon ./src/start.ts", "test": "jest", "lint": "eslint . --fix", - "pretty": "prettier --write ./**/*.ts" + "prettify": "prettier --write './**/*.{ts,yaml,json,js}'", + "checkStyle": "prettier --check './**/*.{ts,yaml,json,js}'" }, "prettier": { "endOfLine": "lf", diff --git a/prisma/migrations/20240811035309_change_unique_constraint/migration.sql b/prisma/migrations/20240811035309_change_unique_constraint/migration.sql new file mode 100644 index 0000000..42dc6ac --- /dev/null +++ b/prisma/migrations/20240811035309_change_unique_constraint/migration.sql @@ -0,0 +1,11 @@ +/* + Warnings: + + - The primary key for the `Assignment` table will be changed. If it partially fails, the table could be left without primary key constraint. + - Made the column `assignmentID` on table `Assignment` required. This step will fail if there are existing NULL values in that column. + +*/ +-- AlterTable +ALTER TABLE "Assignment" DROP CONSTRAINT "Assignment_pkey", +ALTER COLUMN "assignmentID" SET NOT NULL, +ADD CONSTRAINT "Assignment_pkey" PRIMARY KEY ("assignmentID", "mcvCourseID"); diff --git a/prisma/schema.prisma b/prisma/schema.prisma index 7a2b6f3..c27c8c9 100644 --- a/prisma/schema.prisma +++ b/prisma/schema.prisma @@ -5,8 +5,9 @@ datasource db { model Assignment{ mcvCourseID Int course Course @relation(fields:[mcvCourseID], references: [mcvID]) - assignmentName String @id @db.VarChar(255) - assignmentID Int? + assignmentName String @db.VarChar(255) + assignmentID Int + @@id([assignmentID, mcvCourseID]) } model Course{ mcvID Int @id diff --git a/src/commands/update.ts b/src/commands/update.ts index 5a1c7a6..0914a45 100644 --- a/src/commands/update.ts +++ b/src/commands/update.ts @@ -1,7 +1,6 @@ -import { ChatInputCommandInteraction, CacheType, TextChannel } from 'discord.js' -import { updateAll } from '../scraper/updateAll' -import { client } from '../server' -import db, { NotificationChannel } from '../database/database' +import { ChatInputCommandInteraction, CacheType } from 'discord.js' +import { NotificationChannel } from '../database/database' +import updateHandler from '@/utils/updateHandler' export default { name: 'update', @@ -10,26 +9,12 @@ export default { interaction: ChatInputCommandInteraction, _calledChannel: NotificationChannel ) => { - const result = await updateAll() - if (result != '' && result != undefined) { - const notificationChannel = await db.getChannelOfGuild( - interaction.guildId! - ) - if (notificationChannel == null) { - await interaction.editReply( - 'There is no notification NotificationChannel in this server.' - ) - return + const isUpToDate = await updateHandler(); + if(isUpToDate){ + await interaction.editReply('Assignments are up to date!') + } + else{ + await interaction.editReply('Done!') } - const discordChannel = (await client.channels.fetch( - notificationChannel.channelID - )) as TextChannel - discordChannel.send(result) - await interaction.editReply('Done!') - return - } else { - await interaction.editReply('Assignments are up to date!') - return - } }, } diff --git a/src/config/config.ts b/src/config/config.ts index e0a17c6..9efbc5f 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -4,6 +4,11 @@ export const targetYear: MutableWrapper = new MutableWrapper(undefined) export const targetSemester: MutableWrapper = new MutableWrapper(undefined) +export const MAX_DISCORD_MESSAGE_SIZE = 2000 +export const NEW_ASSIGNMENTS_MESSAGE = '## New Assignments!!' +export const NEW_ASSIGNMENTS_MESSAGE_SIZE = [...NEW_ASSIGNMENTS_MESSAGE].length +export const COURSE_MESSAGE_PATTERN = '\n- %s' +export const ASSIGNMENT_MESSAGE_PATTERN = '\n - [%s](https://www.mycourseville.com/?q=courseville/worksheet/%d/%d)' export enum NotifyMessage { FetchingError = 'Error fetching, Might be rate limited or server is down', diff --git a/src/database/database.ts b/src/database/database.ts index 6c76986..9a5a8a0 100644 --- a/src/database/database.ts +++ b/src/database/database.ts @@ -70,8 +70,8 @@ namespace db { export async function getAllCoursesOfTargetSemester(): Promise { return await prisma.course.findMany({ where:{ - year: targetYear.get(), - semester: targetSemester.get() + year: targetYear.value, + semester: targetSemester.value } }) } diff --git a/src/env/env.ts b/src/env/env.ts index d29929e..7da9690 100644 --- a/src/env/env.ts +++ b/src/env/env.ts @@ -43,8 +43,8 @@ if ( ) } if (!env.AUTO_DETERMINE_YEAR_AND_SEMESTER) { - targetYear.set(yearAndSemesterEnv.YEAR!) - targetSemester.set(yearAndSemesterEnv.SEMESTER!) + targetYear.value = yearAndSemesterEnv.YEAR! + targetSemester.value=yearAndSemesterEnv.SEMESTER! } export default env diff --git a/src/scraper/determineYearAndSemester.ts b/src/scraper/determineYearAndSemester.ts index 6537276..abbe3f9 100644 --- a/src/scraper/determineYearAndSemester.ts +++ b/src/scraper/determineYearAndSemester.ts @@ -8,6 +8,6 @@ export async function determineYearAndSemester($: cheerio.Root): Promise { return } const [_beforeSplit, currentYear, currentSemester] = split - targetYear.set(parseInt(currentYear)) - targetSemester.set(parseInt(currentSemester)) + targetYear.value=parseInt(currentYear) + targetSemester.value=parseInt(currentSemester) } diff --git a/src/scraper/updateAll.ts b/src/scraper/updateAll.ts index aaf0c9b..917af68 100644 --- a/src/scraper/updateAll.ts +++ b/src/scraper/updateAll.ts @@ -1,47 +1,82 @@ +import { + ASSIGNMENT_MESSAGE_PATTERN, + COURSE_MESSAGE_PATTERN, + MAX_DISCORD_MESSAGE_SIZE, + NEW_ASSIGNMENTS_MESSAGE, + NEW_ASSIGNMENTS_MESSAGE_SIZE, +} from '@/config/config' import db, { Assignment, - CourseWithAssignments, + Course, } from '../database/database' import updateAssignments from './updateAssignments' import updateCourses from './updateCourses' import * as option from 'fp-ts/Option' +import MutableWrapper from '@/utils/MutableWrapper' +import {format} from 'util' /** * @description update assignments of each course * @returns message containing new added assignments * @throws {Error} */ -export async function updateAll(): Promise { +export async function updateAll(): Promise> { await updateCourses() const coursesList = await db.getAllCoursesOfTargetSemester() - const unfilteredCoursesWithAssignments: Array = - await Promise.all( - coursesList.map(async (course) => { - const newAssignments = await updateAssignments(course.mcvID) - const newAssignmentsUnwrapped: Assignment[] = option.getOrElse( - () => [] as Assignment[] - )(newAssignments) - const result: CourseWithAssignments = { - ...course, - assignments: newAssignmentsUnwrapped, - } - return result - }) + const coursesWithAssignments: Map> = new Map() + for await (const course of coursesList) { + const newAssignments: option.Option = await updateAssignments( + course.mcvID ) - const coursesWithAssignments = unfilteredCoursesWithAssignments.filter( - (course) => { - return course.assignments.length != 0 + if (option.isNone(newAssignments) || newAssignments.value.length == 0) { + continue } - ) - if (coursesWithAssignments.length == 0) { - return '' + coursesWithAssignments.set(course, newAssignments.value) + } + if (coursesWithAssignments.size == 0) { + return [] } - let message: string = '## New Assignments!!' - for (const course of coursesWithAssignments) { - message += `\n- ${course.title}` - for (const assignment of course.assignments) { - message += `\n - [${assignment.assignmentName}](https://www.mycourseville.com/?q=courseville/worksheet/${course.mcvID}/${assignment.assignmentID})` + const messages: string[] = [] + const currentMessage: MutableWrapper = new MutableWrapper( + NEW_ASSIGNMENTS_MESSAGE + ) + const currentMessageSize: MutableWrapper = new MutableWrapper( + NEW_ASSIGNMENTS_MESSAGE_SIZE + ) + for (const [course, assignments] of coursesWithAssignments) { + // const newCourseLine = `\n- ${course.title}` + const newCourseLine = format(COURSE_MESSAGE_PATTERN,course.title) + currentMessageSize.value += [...newCourseLine].length + const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE + if (hasExceeded) { + pushAndReinitialize(messages,currentMessage,currentMessageSize) + } + for (const assignment of assignments) { + // const newAssignmentLine = `\n - [${assignment.assignmentName}](https://www.mycourseville.com/?q=courseville/worksheet/${course.mcvID}/${assignment.assignmentID})` + const newAssignmentLine = format(ASSIGNMENT_MESSAGE_PATTERN,assignment.assignmentName,course.mcvID,assignment.assignmentID) + currentMessageSize.value += [...newAssignmentLine].length + const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE + const isFirstAssignment = assignments[0] == assignment + if (hasExceeded) { + pushAndReinitialize(messages,currentMessage,currentMessageSize) + currentMessage.value += newCourseLine + currentMessageSize.value += [...newCourseLine].length + } else if (isFirstAssignment) { + currentMessage.value += newCourseLine + } + currentMessage.value += newAssignmentLine } } - return message + messages.push(currentMessage.value) + return messages } + +function pushAndReinitialize( + messages: string[], + currentMessage: MutableWrapper, + currentMessageSize: MutableWrapper, +){ + messages.push(currentMessage.value) + currentMessage.value = NEW_ASSIGNMENTS_MESSAGE + currentMessageSize.value = NEW_ASSIGNMENTS_MESSAGE_SIZE +} \ No newline at end of file diff --git a/src/scraper/updateCourses.ts b/src/scraper/updateCourses.ts index 6d7290b..23eeeca 100644 --- a/src/scraper/updateCourses.ts +++ b/src/scraper/updateCourses.ts @@ -23,7 +23,7 @@ export default async function updateCourses(): Promise { determineYearAndSemester($) } const courseElements: cheerio.Element[] = $( - `#courseville-courseicongroup-icon-lineup-${targetYear.get()}-${targetSemester.get()}-join a` + `#courseville-courseicongroup-icon-lineup-${targetYear.value}-${targetSemester.value}-join a` ).toArray() for (let courseElement of courseElements) { courseElement = courseElement as cheerio.TagElement diff --git a/src/server.ts b/src/server.ts index 80d82a1..21ac169 100644 --- a/src/server.ts +++ b/src/server.ts @@ -8,6 +8,7 @@ import formatDateToBangkok from './utils/formatDateToBangkok' import updateHandler from './utils/updateHandler' import MutableWrapper from './utils/MutableWrapper' import DiscordCommandHandler from './interfaces/DiscordCommandHandler' +import {isSome} from 'fp-ts/lib/Option' export const hasEncounteredError: MutableWrapper = new MutableWrapper( false @@ -58,7 +59,7 @@ client.on('ready', async () => { adminDM = await client.users.createDM(env.ADMIN_USER_ID) adminDM.send('server is up!') - if (await updateHandler()) { + if (isSome(await updateHandler())) { intervalId = setInterval(updateHandler, env.DELAY * 1000) } }) diff --git a/src/start.ts b/src/start.ts index 396afda..bc9a2e8 100644 --- a/src/start.ts +++ b/src/start.ts @@ -1,3 +1,3 @@ import { start } from './server' -start() +start() \ No newline at end of file diff --git a/src/utils/MutableWrapper.ts b/src/utils/MutableWrapper.ts index 2431613..4f38542 100644 --- a/src/utils/MutableWrapper.ts +++ b/src/utils/MutableWrapper.ts @@ -1,12 +1,12 @@ export default class MutableWrapper { - value: T + _value: T constructor(value: T) { - this.value = value + this._value = value } - get() { - return this.value + get value() { + return this._value } - set(newValue: T) { - this.value = newValue + set value(newValue: T) { + this._value = newValue } } diff --git a/src/utils/fetchAndCatch.ts b/src/utils/fetchAndCatch.ts index 1322482..d585c1f 100644 --- a/src/utils/fetchAndCatch.ts +++ b/src/utils/fetchAndCatch.ts @@ -19,24 +19,24 @@ export default async function fetchAndCatch( }, body: body, }).catch(async (_e) => { - if (hasEncounteredError.get()) { + if (hasEncounteredError.value) { return } - hasEncounteredError.set(true) + hasEncounteredError.value=true await errorFetchingNotify(NotifyMessage.FetchingError) }) if (response == undefined) { return option.none } if (response.status != 200) { - if (hasEncounteredError.get()) { + if (hasEncounteredError.value) { return option.none } console.log(await response.text()) - hasEncounteredError.set(true) + hasEncounteredError.value=true await errorFetchingNotify(NotifyMessage.FetchingError) return option.none } - hasEncounteredError.set(false) + hasEncounteredError.value=false return option.some(response) } diff --git a/src/utils/updateHandler.ts b/src/utils/updateHandler.ts index 2726b01..3a6cbda 100644 --- a/src/utils/updateHandler.ts +++ b/src/utils/updateHandler.ts @@ -4,31 +4,35 @@ import env from '../env/env' import { updateAll } from '../scraper/updateAll' import { client } from '../server' import formatDateToBangkok from './formatDateToBangkok' +import {none, Option, some} from 'fp-ts/lib/Option' /** * @description update assignments and send message to all notification channels * */ -export default async function updateHandler() { +export default async function updateHandler(): Promise>{ if (env.INTERVAL_LOGGING) { console.log('new interval started at ' + formatDateToBangkok(new Date())) } - let message + let messages: Array = [] try { - message = await updateAll() + messages = await updateAll() + console.log("messages",messages) } catch (e) { console.log(e) - return false + return none } - if (message === '') { - return true + if (messages.length == 0) { + return some(true) } const channels = await db.getAllChannels() - for await (const NotificationChannel of channels) { - const discordChannel = (await client.channels.fetch( - NotificationChannel.channelID - )) as TextChannel - // adminDM.send(message); - await discordChannel.send(message) + for (const message of messages) { + for await (const NotificationChannel of channels) { + const discordChannel = (await client.channels.fetch( + NotificationChannel.channelID + )) as TextChannel + // adminDM.send(message); + await discordChannel.send(message) + } } - return true + return some(false) } diff --git a/tests/unit/determineYearAndSemester.unit.test.ts b/tests/unit/determineYearAndSemester.unit.test.ts index 73529d1..d2d5e5d 100644 --- a/tests/unit/determineYearAndSemester.unit.test.ts +++ b/tests/unit/determineYearAndSemester.unit.test.ts @@ -65,7 +65,7 @@ describe('stop notify after encountered error', () => { test('', async () => { determineYearAndSemester($) - expect(targetYear.get()).toBe(2023) - expect(targetSemester.get()).toBe(3) + expect(targetYear.value).toBe(2023) + expect(targetSemester.value).toBe(3) }) }) diff --git a/tests/unit/updateAll.unit.test.ts b/tests/unit/updateAll.unit.test.ts index 5faee6d..95c3f98 100644 --- a/tests/unit/updateAll.unit.test.ts +++ b/tests/unit/updateAll.unit.test.ts @@ -1,35 +1,282 @@ -import {Course} from "@prisma/client"; +// jest.mock("@/scraper/updateAll") +const mockGetAllCoursesOfTargetSemester = jest.fn() +const mockUpdateAssignments = jest.fn() +jest.mock('@/scraper/updateCourses') -let mockUpdateCourses = jest.fn(); -let mockUpdateAssignments = jest.fn(); -let mockDatabase = {}; - -jest.mock("@/scraper/updateAll",()=>{ +jest.mock('@/database/database', () => { return { - __default: mockUpdateCourses + getAllCoursesOfTargetSemester: mockGetAllCoursesOfTargetSemester, } }) -jest.mock("@/database/database",()=>{ +jest.mock('@/scraper/updateAssignments', () => { return { - __default: mockDatabase + __esModule: true, + default: mockUpdateAssignments, } }) -jest.mock("@/scraper/updateAssignments",()=>{ - return { - __default: mockUpdateAssignments +import { Assignment } from '@/database/database' +import { updateAll } from '@/scraper/updateAll' +import { Course } from '@prisma/client' +import { none, some } from 'fp-ts/lib/Option' +import assert from 'assert' +describe('updateAll', () => { + let coursesFromUpdate: Course[] = [] + let assignmentsOfCourseFromUpdate: Record = {} + const updateAllSpy = jest.fn(updateAll) + const course123 = { + mcvID: 123, + courseID: '2111031', + title: 'How to Rickroll 101', + year: 2023, + semester: 2, } -}) + const course540 = { + mcvID: 540, + courseID: '2111032', + title: 'How to Make Mcv bot 101', + year: 2023, + semester: 2, + } + + beforeAll(() => { + mockGetAllCoursesOfTargetSemester.mockImplementation( + () => coursesFromUpdate + ) + mockUpdateAssignments.mockImplementation((mcvId: number) => { + const assignments = assignmentsOfCourseFromUpdate[mcvId] + if (assignments != undefined) { + return some(assignments) + } else { + return none + } + }) + }) + + beforeEach(() => { + jest.clearAllMocks() + }) + + it('normal', async () => { + coursesFromUpdate = [course123] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'งานที่ 1', + }, + ], + } + const result = await updateAllSpy() + const expected: string[] = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [งานที่ 1](https://www.mycourseville.com/?q=courseville/worksheet/123/456)`, + ] + assertAndExpect(result, expected, updateAllSpy) + }) + + it('2 courses', async () => { + coursesFromUpdate = [course123, course540] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'งานที่ 1', + }, + ], + 540: [ + { + mcvCourseID: 540, + assignmentID: 456, + assignmentName: 'งานที่ 1 นะจ๊ะ', + }, + ], + } + const result = await updateAllSpy() + const expected: string[] = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [งานที่ 1](https://www.mycourseville.com/?q=courseville/worksheet/123/456)`+ + '\n- How to Make Mcv bot 101' + + `\n - [งานที่ 1 นะจ๊ะ](https://www.mycourseville.com/?q=courseville/worksheet/540/456)`, + ] + assertAndExpect(result, expected, updateAllSpy) + }) + + it('no new courses', async () => { + coursesFromUpdate = [] + assignmentsOfCourseFromUpdate = {} + const result = await updateAllSpy() + const expected: string[] = [] + assertAndExpect(result, expected, updateAllSpy) + }) -describe("",()=>{ - let coursesFromUpdate: Course[] = []; - let assignmentsOfCourseFromUpdate: Record = {}; - beforeAll(()=>{ - mockUpdateCourses.mockImplementation(()=>coursesFromUpdate); - mockUpdateAssignments.mockImplementation(()=>coursesFromUpdate); + it('no new assignments', async () => { + coursesFromUpdate = [course123] + assignmentsOfCourseFromUpdate = {} + const result = await updateAllSpy() + const expected: string[] = [] + assertAndExpect(result, expected, updateAllSpy) }) - it("normal",()=>{ + it('exceed limit in the same course', async () => { + coursesFromUpdate = [course123] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'ก'.repeat(850), + }, + { + mcvCourseID: 123, + assignmentID: 789, + assignmentName: 'ข'.repeat(850), + }, + { + mcvCourseID: 123, + assignmentID: 150, + assignmentName: 'ค'.repeat(850), + }, + ], + } + const result = await updateAllSpy() + const expected = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ก'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/456)` + + `\n - [${'ข'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/789)`, + + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ค'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/150)`, + ] + assertAndExpect(result,expected,updateAllSpy); }) -}) \ No newline at end of file + + it('exceed limit in the same course: edge case: 2000 chars after ข', async () => { + coursesFromUpdate = [course123] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'ก'.repeat(850), + }, + { + mcvCourseID: 123, + assignmentID: 789, + assignmentName: 'ข'.repeat(968), + }, + { + mcvCourseID: 123, + assignmentID: 150, + assignmentName: 'ค'.repeat(850), + }, + ], + } + const result = await updateAllSpy() + const expected = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ก'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/456)` + + `\n - [${'ข'.repeat(968)}](https://www.mycourseville.com/?q=courseville/worksheet/123/789)`, + + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ค'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/150)`, + ] + assertAndExpect(result, expected, updateAllSpy) + }) + + it('exceed limit in the same course: edge case: 2001 chars after ข', async () => { + coursesFromUpdate = [course123] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'ก'.repeat(850), + }, + { + mcvCourseID: 123, + assignmentID: 789, + assignmentName: 'ข'.repeat(969), + }, + { + mcvCourseID: 123, + assignmentID: 150, + assignmentName: 'ค'.repeat(849), + }, + ], + } + const result = await updateAllSpy() + const expected = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ก'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/456)`, + + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ข'.repeat(969)}](https://www.mycourseville.com/?q=courseville/worksheet/123/789)` + + `\n - [${'ค'.repeat(849)}](https://www.mycourseville.com/?q=courseville/worksheet/123/150)`, + ] + for (const expectedStr of expected) { + assert.equal([...expectedStr].length <= 2000, true) + } + assertAndExpect(result, expected, updateAllSpy) + }) + + it('exceed limit in the different course', async () => { + coursesFromUpdate = [course123, course540] + assignmentsOfCourseFromUpdate = { + 123: [ + { + mcvCourseID: 123, + assignmentID: 456, + assignmentName: 'ก'.repeat(850), + }, + { + mcvCourseID: 123, + assignmentID: 789, + assignmentName: 'ข'.repeat(850), + }, + ], + 540: [ + { + mcvCourseID: 540, + assignmentID: 150, + assignmentName: 'ค'.repeat(850), + }, + ], + } + const result = await updateAllSpy() + const expected = [ + '## New Assignments!!' + + '\n- How to Rickroll 101' + + `\n - [${'ก'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/456)` + + `\n - [${'ข'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/789)`, + + '## New Assignments!!' + + '\n- How to Make Mcv bot 101' + + `\n - [${'ค'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/540/150)`, + ] + for (const expectedStr of expected) { + assert.equal([...expectedStr].length <= 2000, true) + } + assertAndExpect(result, expected, updateAllSpy) + }) +}) + +function assertAndExpect( + result: unknown, + expected: unknown, + updateAllSpy: jest.Mock +) { + assert.deepEqual(result, expected) + expect(updateAllSpy).toHaveBeenCalledTimes(1) +} diff --git a/tests/unit/updateCourses.unit.test.ts b/tests/unit/updateCourses.unit.test.ts index 8c04872..5f90236 100644 --- a/tests/unit/updateCourses.unit.test.ts +++ b/tests/unit/updateCourses.unit.test.ts @@ -36,7 +36,7 @@ describe('stop notify after encountered error', () => { })) }) beforeEach(() => { - hasEncounteredError.set(false) + hasEncounteredError.value=false mockErrorFetchingNotify.mockClear() }) @@ -47,8 +47,8 @@ describe('stop notify after encountered error', () => { }) test('already encountered', async () => { - hasEncounteredError.set(true) - expect(hasEncounteredError.get()).toBe(true) + hasEncounteredError.value=true + expect(hasEncounteredError.value).toBe(true) await updateCourses() expect(mockErrorFetchingNotify).not.toHaveBeenCalled() }) diff --git a/tsconfig.json b/tsconfig.json index 3f92a05..c014ea4 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,4 +1,8 @@ { + "ts-node": { + // Do not forget to `npm i -D tsconfig-paths` + "require": ["tsconfig-paths/register"] + }, "compilerOptions": { /* Visit https://aka.ms/tsconfig to read more about this file */ @@ -11,7 +15,7 @@ // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ /* Language and Environment */ - "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + "target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */, // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ // "jsx": "preserve", /* Specify what JSX code is generated. */ // "experimentalDecorators": true, /* Enable experimental support for legacy experimental decorators. */ @@ -25,13 +29,13 @@ // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ /* Modules */ - "module": "commonjs", /* Specify what module code is generated. */ - "rootDir": "./", /* Specify the root folder within your source files. */ + "module": "commonjs" /* Specify what module code is generated. */, + "rootDir": "./" /* Specify the root folder within your source files. */, // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */ // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ "paths": { "@/*": ["./src/*"] - }, /* Specify a set of entries that re-map imports to additional lookup locations. */ + } /* Specify a set of entries that re-map imports to additional lookup locations. */, // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ // "types": [], /* Specify type package names to be included without being referenced in a source file. */ @@ -57,7 +61,7 @@ // "sourceMap": true, /* Create source map files for emitted JavaScript files. */ // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ - "outDir": "./build", /* Specify an output folder for all emitted files. */ + "outDir": "./build" /* Specify an output folder for all emitted files. */, // "removeComments": true, /* Disable emitting comments. */ // "noEmit": true, /* Disable emitting files from a compilation. */ // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ @@ -78,21 +82,21 @@ /* Interop Constraints */ // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ // "verbatimModuleSyntax": true, /* Do not transform or elide any imports or exports not marked as type-only, ensuring they are written in the output file's format based on the 'module' setting. */ - "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ - "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + "allowSyntheticDefaultImports": true /* Allow 'import x from y' when a module doesn't have a default export. */, + "esModuleInterop": true /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */, // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ - "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + "forceConsistentCasingInFileNames": true /* Ensure that casing is correct in imports. */, /* Type Checking */ - "strict": true, /* Enable all strict type-checking options. */ - "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ - "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + "strict": true /* Enable all strict type-checking options. */, + "noImplicitAny": true /* Enable error reporting for expressions and declarations with an implied 'any' type. */, + "strictNullChecks": true /* When type checking, take into account 'null' and 'undefined'. */, // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ - "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + "alwaysStrict": true /* Ensure 'use strict' is always emitted. */, // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ @@ -106,7 +110,7 @@ /* Completeness */ // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ - "skipLibCheck": true /* Skip type checking all .d.ts files. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ }, "exclude": ["tests"] } From 7242e738f9b84959aa951b815972b2bcf9c349a0 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Mon, 12 Aug 2024 16:44:15 +0700 Subject: [PATCH 5/8] prettify --- docker-compose-dev.yaml | 2 +- docker-compose-prod.yaml | 2 +- docker-compose.yaml | 4 +-- jest.config.js | 6 ++-- src/commands/update.ts | 13 ++++----- src/config/config.ts | 3 +- src/database/database.ts | 10 +++---- src/env/env.ts | 2 +- src/scraper/determineYearAndSemester.ts | 4 +-- src/scraper/extractAssignmentsFromCheerio.ts | 8 +++--- src/scraper/scrapeAssignmentsOfPage.ts | 10 +++---- src/scraper/updateAll.ts | 26 +++++++++-------- src/scraper/updateAssignments.ts | 10 +++---- src/server.ts | 2 +- src/start.ts | 2 +- src/utils/fetchAndCatch.ts | 6 ++-- src/utils/updateHandler.ts | 6 ++-- tests/tsconfig.json | 28 +++++++++---------- .../determineYearAndSemester.unit.test.ts | 6 ++-- ...extractAssignmentsFromCheerio.unit.test.ts | 10 +++---- tests/unit/updateAll.unit.test.ts | 4 +-- tests/unit/updateCourses.unit.test.ts | 12 ++++---- 22 files changed, 88 insertions(+), 88 deletions(-) diff --git a/docker-compose-dev.yaml b/docker-compose-dev.yaml index a96bae8..1989a80 100644 --- a/docker-compose-dev.yaml +++ b/docker-compose-dev.yaml @@ -11,4 +11,4 @@ services: volumes: - postgres-db-noti:/var/lib/postgresql/data volumes: - postgres-db-noti: \ No newline at end of file + postgres-db-noti: diff --git a/docker-compose-prod.yaml b/docker-compose-prod.yaml index 978715c..c95cb78 100644 --- a/docker-compose-prod.yaml +++ b/docker-compose-prod.yaml @@ -23,4 +23,4 @@ services: depends_on: - postgres volumes: - postgres-db-noti: \ No newline at end of file + postgres-db-noti: diff --git a/docker-compose.yaml b/docker-compose.yaml index a7f902c..01108e4 100644 --- a/docker-compose.yaml +++ b/docker-compose.yaml @@ -11,8 +11,8 @@ services: env_file: - ./.env healthcheck: - test: ["CMD", "curl", "-f", "http://localhost:${PORT}/health"] + test: ['CMD', 'curl', '-f', 'http://localhost:${PORT}/health'] interval: 1m30s timeout: 30s retries: 5 - start_period: 30s \ No newline at end of file + start_period: 30s diff --git a/jest.config.js b/jest.config.js index a26e829..eb72378 100644 --- a/jest.config.js +++ b/jest.config.js @@ -3,6 +3,6 @@ module.exports = { preset: 'ts-jest', testEnvironment: 'node', moduleNameMapper: { - "^@/(.*)$": "/src/$1" - } -}; \ No newline at end of file + '^@/(.*)$': '/src/$1', + }, +} diff --git a/src/commands/update.ts b/src/commands/update.ts index 0914a45..5219ddd 100644 --- a/src/commands/update.ts +++ b/src/commands/update.ts @@ -9,12 +9,11 @@ export default { interaction: ChatInputCommandInteraction, _calledChannel: NotificationChannel ) => { - const isUpToDate = await updateHandler(); - if(isUpToDate){ - await interaction.editReply('Assignments are up to date!') - } - else{ - await interaction.editReply('Done!') - } + const isUpToDate = await updateHandler() + if (isUpToDate) { + await interaction.editReply('Assignments are up to date!') + } else { + await interaction.editReply('Done!') + } }, } diff --git a/src/config/config.ts b/src/config/config.ts index 9efbc5f..727044e 100644 --- a/src/config/config.ts +++ b/src/config/config.ts @@ -8,7 +8,8 @@ export const MAX_DISCORD_MESSAGE_SIZE = 2000 export const NEW_ASSIGNMENTS_MESSAGE = '## New Assignments!!' export const NEW_ASSIGNMENTS_MESSAGE_SIZE = [...NEW_ASSIGNMENTS_MESSAGE].length export const COURSE_MESSAGE_PATTERN = '\n- %s' -export const ASSIGNMENT_MESSAGE_PATTERN = '\n - [%s](https://www.mycourseville.com/?q=courseville/worksheet/%d/%d)' +export const ASSIGNMENT_MESSAGE_PATTERN = + '\n - [%s](https://www.mycourseville.com/?q=courseville/worksheet/%d/%d)' export enum NotifyMessage { FetchingError = 'Error fetching, Might be rate limited or server is down', diff --git a/src/database/database.ts b/src/database/database.ts index 9a5a8a0..b30e216 100644 --- a/src/database/database.ts +++ b/src/database/database.ts @@ -5,7 +5,7 @@ import { NotificationChannel as PrismaNotificationChannel, } from '@prisma/client' import { assignmentsCache, coursesCache } from './cache' -import {targetSemester, targetYear} from '../config/config' +import { targetSemester, targetYear } from '../config/config' const prisma = new PrismaClient() @@ -69,10 +69,10 @@ namespace db { export async function getAllCoursesOfTargetSemester(): Promise { return await prisma.course.findMany({ - where:{ + where: { year: targetYear.value, - semester: targetSemester.value - } + semester: targetSemester.value, + }, }) } @@ -137,7 +137,7 @@ namespace db { export type NotificationChannel = PrismaNotificationChannel export type Course = PrismaCourse export type Assignment = PrismaAssignment -export type CourseWithAssignments = Course & { assignments: Array} +export type CourseWithAssignments = Course & { assignments: Array } export default db diff --git a/src/env/env.ts b/src/env/env.ts index 7da9690..8194234 100644 --- a/src/env/env.ts +++ b/src/env/env.ts @@ -44,7 +44,7 @@ if ( } if (!env.AUTO_DETERMINE_YEAR_AND_SEMESTER) { targetYear.value = yearAndSemesterEnv.YEAR! - targetSemester.value=yearAndSemesterEnv.SEMESTER! + targetSemester.value = yearAndSemesterEnv.SEMESTER! } export default env diff --git a/src/scraper/determineYearAndSemester.ts b/src/scraper/determineYearAndSemester.ts index abbe3f9..b1c60dc 100644 --- a/src/scraper/determineYearAndSemester.ts +++ b/src/scraper/determineYearAndSemester.ts @@ -8,6 +8,6 @@ export async function determineYearAndSemester($: cheerio.Root): Promise { return } const [_beforeSplit, currentYear, currentSemester] = split - targetYear.value=parseInt(currentYear) - targetSemester.value=parseInt(currentSemester) + targetYear.value = parseInt(currentYear) + targetSemester.value = parseInt(currentSemester) } diff --git a/src/scraper/extractAssignmentsFromCheerio.ts b/src/scraper/extractAssignmentsFromCheerio.ts index 856637c..994ebfa 100644 --- a/src/scraper/extractAssignmentsFromCheerio.ts +++ b/src/scraper/extractAssignmentsFromCheerio.ts @@ -9,13 +9,13 @@ export default async function extractAssignmentsFromCheerio( const assignments: Array = [] for (let i = 0; i < assignmentNameNodes.length; i++) { const ele = assignmentNameNodes[i] - const assignmentLink = $(ele).attr("href"); - const assignmentIdStr: string = assignmentLink!.match(/^.*\/(\d+)$/)![1]; - const assignmentId: number = parseInt(assignmentIdStr,10); + const assignmentLink = $(ele).attr('href') + const assignmentIdStr: string = assignmentLink!.match(/^.*\/(\d+)$/)![1] + const assignmentId: number = parseInt(assignmentIdStr, 10) const assignment: Assignment = { mcvCourseID: mcvID, assignmentName: $(ele).text(), - assignmentID: assignmentId + assignmentID: assignmentId, } const found = await db.assignmentExists(assignment) if (!found) { diff --git a/src/scraper/scrapeAssignmentsOfPage.ts b/src/scraper/scrapeAssignmentsOfPage.ts index 912038e..871d8a9 100644 --- a/src/scraper/scrapeAssignmentsOfPage.ts +++ b/src/scraper/scrapeAssignmentsOfPage.ts @@ -25,7 +25,7 @@ export default async function scrapeAssignmentsOfPage( const resultJson: LoadMoreAssignmentsResponse = await response?.json() if (resultJson.status == 0) { - return option.none; + return option.none } const $ = cheerio.load( @@ -33,11 +33,11 @@ export default async function scrapeAssignmentsOfPage( ) let assignments = await extractAssignmentsFromCheerio(mcvID, $) - + if (resultJson.all == undefined || resultJson.all !== true) { - const optionalResult = await scrapeAssignmentsOfPage(mcvID, next + 5); - if(option.isSome(optionalResult)){ - assignments = assignments.concat(optionalResult.value); + const optionalResult = await scrapeAssignmentsOfPage(mcvID, next + 5) + if (option.isSome(optionalResult)) { + assignments = assignments.concat(optionalResult.value) } } return option.some(assignments) diff --git a/src/scraper/updateAll.ts b/src/scraper/updateAll.ts index 917af68..6a49eed 100644 --- a/src/scraper/updateAll.ts +++ b/src/scraper/updateAll.ts @@ -5,15 +5,12 @@ import { NEW_ASSIGNMENTS_MESSAGE, NEW_ASSIGNMENTS_MESSAGE_SIZE, } from '@/config/config' -import db, { - Assignment, - Course, -} from '../database/database' +import db, { Assignment, Course } from '../database/database' import updateAssignments from './updateAssignments' import updateCourses from './updateCourses' import * as option from 'fp-ts/Option' import MutableWrapper from '@/utils/MutableWrapper' -import {format} from 'util' +import { format } from 'util' /** * @description update assignments of each course @@ -45,20 +42,25 @@ export async function updateAll(): Promise> { ) for (const [course, assignments] of coursesWithAssignments) { // const newCourseLine = `\n- ${course.title}` - const newCourseLine = format(COURSE_MESSAGE_PATTERN,course.title) + const newCourseLine = format(COURSE_MESSAGE_PATTERN, course.title) currentMessageSize.value += [...newCourseLine].length const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE if (hasExceeded) { - pushAndReinitialize(messages,currentMessage,currentMessageSize) + pushAndReinitialize(messages, currentMessage, currentMessageSize) } for (const assignment of assignments) { // const newAssignmentLine = `\n - [${assignment.assignmentName}](https://www.mycourseville.com/?q=courseville/worksheet/${course.mcvID}/${assignment.assignmentID})` - const newAssignmentLine = format(ASSIGNMENT_MESSAGE_PATTERN,assignment.assignmentName,course.mcvID,assignment.assignmentID) + const newAssignmentLine = format( + ASSIGNMENT_MESSAGE_PATTERN, + assignment.assignmentName, + course.mcvID, + assignment.assignmentID + ) currentMessageSize.value += [...newAssignmentLine].length const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE const isFirstAssignment = assignments[0] == assignment if (hasExceeded) { - pushAndReinitialize(messages,currentMessage,currentMessageSize) + pushAndReinitialize(messages, currentMessage, currentMessageSize) currentMessage.value += newCourseLine currentMessageSize.value += [...newCourseLine].length } else if (isFirstAssignment) { @@ -74,9 +76,9 @@ export async function updateAll(): Promise> { function pushAndReinitialize( messages: string[], currentMessage: MutableWrapper, - currentMessageSize: MutableWrapper, -){ + currentMessageSize: MutableWrapper +) { messages.push(currentMessage.value) currentMessage.value = NEW_ASSIGNMENTS_MESSAGE currentMessageSize.value = NEW_ASSIGNMENTS_MESSAGE_SIZE -} \ No newline at end of file +} diff --git a/src/scraper/updateAssignments.ts b/src/scraper/updateAssignments.ts index 4b51828..e59bb68 100644 --- a/src/scraper/updateAssignments.ts +++ b/src/scraper/updateAssignments.ts @@ -23,10 +23,10 @@ export default async function updateAssignments( let assignments = await extractAssignmentsFromCheerio(mcvID, $) - const optionalResult = await scrapeAssignmentsOfPage(mcvID,5); - if(option.isSome(optionalResult)){ - assignments = assignments.concat(optionalResult.value!); + const optionalResult = await scrapeAssignmentsOfPage(mcvID, 5) + if (option.isSome(optionalResult)) { + assignments = assignments.concat(optionalResult.value!) } - + return option.some(assignments) -} \ No newline at end of file +} diff --git a/src/server.ts b/src/server.ts index 21ac169..748a1e6 100644 --- a/src/server.ts +++ b/src/server.ts @@ -8,7 +8,7 @@ import formatDateToBangkok from './utils/formatDateToBangkok' import updateHandler from './utils/updateHandler' import MutableWrapper from './utils/MutableWrapper' import DiscordCommandHandler from './interfaces/DiscordCommandHandler' -import {isSome} from 'fp-ts/lib/Option' +import { isSome } from 'fp-ts/lib/Option' export const hasEncounteredError: MutableWrapper = new MutableWrapper( false diff --git a/src/start.ts b/src/start.ts index bc9a2e8..396afda 100644 --- a/src/start.ts +++ b/src/start.ts @@ -1,3 +1,3 @@ import { start } from './server' -start() \ No newline at end of file +start() diff --git a/src/utils/fetchAndCatch.ts b/src/utils/fetchAndCatch.ts index d585c1f..ab14e57 100644 --- a/src/utils/fetchAndCatch.ts +++ b/src/utils/fetchAndCatch.ts @@ -22,7 +22,7 @@ export default async function fetchAndCatch( if (hasEncounteredError.value) { return } - hasEncounteredError.value=true + hasEncounteredError.value = true await errorFetchingNotify(NotifyMessage.FetchingError) }) if (response == undefined) { @@ -33,10 +33,10 @@ export default async function fetchAndCatch( return option.none } console.log(await response.text()) - hasEncounteredError.value=true + hasEncounteredError.value = true await errorFetchingNotify(NotifyMessage.FetchingError) return option.none } - hasEncounteredError.value=false + hasEncounteredError.value = false return option.some(response) } diff --git a/src/utils/updateHandler.ts b/src/utils/updateHandler.ts index 3a6cbda..d1ef1a0 100644 --- a/src/utils/updateHandler.ts +++ b/src/utils/updateHandler.ts @@ -4,19 +4,19 @@ import env from '../env/env' import { updateAll } from '../scraper/updateAll' import { client } from '../server' import formatDateToBangkok from './formatDateToBangkok' -import {none, Option, some} from 'fp-ts/lib/Option' +import { none, Option, some } from 'fp-ts/lib/Option' /** * @description update assignments and send message to all notification channels * */ -export default async function updateHandler(): Promise>{ +export default async function updateHandler(): Promise> { if (env.INTERVAL_LOGGING) { console.log('new interval started at ' + formatDateToBangkok(new Date())) } let messages: Array = [] try { messages = await updateAll() - console.log("messages",messages) + console.log('messages', messages) } catch (e) { console.log(e) return none diff --git a/tests/tsconfig.json b/tests/tsconfig.json index c67f6d8..800cad3 100644 --- a/tests/tsconfig.json +++ b/tests/tsconfig.json @@ -11,7 +11,7 @@ // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ /* Language and Environment */ - "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + "target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */, // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ // "jsx": "preserve", /* Specify what JSX code is generated. */ // "experimentalDecorators": true, /* Enable experimental support for legacy experimental decorators. */ @@ -25,13 +25,13 @@ // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ /* Modules */ - "module": "commonjs", /* Specify what module code is generated. */ - "rootDir": "../", /* Specify the root folder within your source files. */ + "module": "commonjs" /* Specify what module code is generated. */, + "rootDir": "../" /* Specify the root folder within your source files. */, // "moduleResolution": "node10", /* Specify how TypeScript looks up a file from a given module specifier. */ // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ "paths": { "@/*": ["../src/*"] - }, /* Specify a set of entries that re-map imports to additional lookup locations. */ + } /* Specify a set of entries that re-map imports to additional lookup locations. */, // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ // "typeRoots": [], /* Specify multiple folders that act like './node_modules/@types'. */ // "types": [], /* Specify type package names to be included without being referenced in a source file. */ @@ -57,7 +57,7 @@ // "sourceMap": true, /* Create source map files for emitted JavaScript files. */ // "inlineSourceMap": true, /* Include sourcemap files inside the emitted JavaScript. */ // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ - "outDir": "./build", /* Specify an output folder for all emitted files. */ + "outDir": "./build" /* Specify an output folder for all emitted files. */, // "removeComments": true, /* Disable emitting comments. */ // "noEmit": true, /* Disable emitting files from a compilation. */ // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ @@ -78,21 +78,21 @@ /* Interop Constraints */ // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ // "verbatimModuleSyntax": true, /* Do not transform or elide any imports or exports not marked as type-only, ensuring they are written in the output file's format based on the 'module' setting. */ - "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ - "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + "allowSyntheticDefaultImports": true /* Allow 'import x from y' when a module doesn't have a default export. */, + "esModuleInterop": true /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */, // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ - "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + "forceConsistentCasingInFileNames": true /* Ensure that casing is correct in imports. */, /* Type Checking */ - "strict": true, /* Enable all strict type-checking options. */ - "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ - "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ + "strict": true /* Enable all strict type-checking options. */, + "noImplicitAny": true /* Enable error reporting for expressions and declarations with an implied 'any' type. */, + "strictNullChecks": true /* When type checking, take into account 'null' and 'undefined'. */, // "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ // "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ // "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ // "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ // "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ - "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ + "alwaysStrict": true /* Ensure 'use strict' is always emitted. */, // "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ // "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ // "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ @@ -106,6 +106,6 @@ /* Completeness */ // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ - "skipLibCheck": true /* Skip type checking all .d.ts files. */ - }, + "skipLibCheck": true /* Skip type checking all .d.ts files. */ + } } diff --git a/tests/unit/determineYearAndSemester.unit.test.ts b/tests/unit/determineYearAndSemester.unit.test.ts index d2d5e5d..63d3c3d 100644 --- a/tests/unit/determineYearAndSemester.unit.test.ts +++ b/tests/unit/determineYearAndSemester.unit.test.ts @@ -54,9 +54,9 @@ import responseToCheerio from '@/utils/responseToCheerio' let $: cheerio.Root describe('stop notify after encountered error', () => { beforeAll(async () => { - const result = await fetchAndCatch(`https://www.mycourseville.com/`,"GET"); - const optionalCheerioRoot = await responseToCheerio(result); - expect(option.isSome(optionalCheerioRoot)); + const result = await fetchAndCatch(`https://www.mycourseville.com/`, 'GET') + const optionalCheerioRoot = await responseToCheerio(result) + expect(option.isSome(optionalCheerioRoot)) if (option.isNone(optionalCheerioRoot)) { return } diff --git a/tests/unit/extractAssignmentsFromCheerio.unit.test.ts b/tests/unit/extractAssignmentsFromCheerio.unit.test.ts index 85311c7..38a84c7 100644 --- a/tests/unit/extractAssignmentsFromCheerio.unit.test.ts +++ b/tests/unit/extractAssignmentsFromCheerio.unit.test.ts @@ -1,7 +1,5 @@ -const mockSaveAssignment = jest.fn().mockImplementation(()=>{}) -const mockAssignmentExists = jest - .fn() - .mockImplementation(async () => false) +const mockSaveAssignment = jest.fn().mockImplementation(() => {}) +const mockAssignmentExists = jest.fn().mockImplementation(async () => false) jest.mock('@/database/database', () => { return { assignmentExists: mockAssignmentExists, @@ -34,10 +32,10 @@ describe('parse assignments', () => { ` - const $ = cheerio.load(html); + const $ = cheerio.load(html) // console.log($('tbody tr td:nth-child(2) a').toArray()) await extractAssignmentsFromCheerio(123, $) - expect(mockAssignmentExists).toHaveBeenCalledTimes(1); + expect(mockAssignmentExists).toHaveBeenCalledTimes(1) expect(mockSaveAssignment).toHaveBeenCalledTimes(1) expect(mockSaveAssignment).toHaveBeenCalledWith({ mcvCourseID: 123, diff --git a/tests/unit/updateAll.unit.test.ts b/tests/unit/updateAll.unit.test.ts index 95c3f98..c518d6c 100644 --- a/tests/unit/updateAll.unit.test.ts +++ b/tests/unit/updateAll.unit.test.ts @@ -100,7 +100,7 @@ describe('updateAll', () => { const expected: string[] = [ '## New Assignments!!' + '\n- How to Rickroll 101' + - `\n - [งานที่ 1](https://www.mycourseville.com/?q=courseville/worksheet/123/456)`+ + `\n - [งานที่ 1](https://www.mycourseville.com/?q=courseville/worksheet/123/456)` + '\n- How to Make Mcv bot 101' + `\n - [งานที่ 1 นะจ๊ะ](https://www.mycourseville.com/?q=courseville/worksheet/540/456)`, ] @@ -155,7 +155,7 @@ describe('updateAll', () => { '\n- How to Rickroll 101' + `\n - [${'ค'.repeat(850)}](https://www.mycourseville.com/?q=courseville/worksheet/123/150)`, ] - assertAndExpect(result,expected,updateAllSpy); + assertAndExpect(result, expected, updateAllSpy) }) it('exceed limit in the same course: edge case: 2000 chars after ข', async () => { diff --git a/tests/unit/updateCourses.unit.test.ts b/tests/unit/updateCourses.unit.test.ts index 5f90236..f9ee9a9 100644 --- a/tests/unit/updateCourses.unit.test.ts +++ b/tests/unit/updateCourses.unit.test.ts @@ -1,4 +1,4 @@ -global.fetch = jest.fn(); +global.fetch = jest.fn() jest.mock('@/env/env', () => { return { @@ -30,13 +30,13 @@ import updateCourses from '@/scraper/updateCourses' import { hasEncounteredError } from '@/server' describe('stop notify after encountered error', () => { - beforeAll(()=>{ - (global.fetch as jest.Mock).mockImplementation((async () => { + beforeAll(() => { + ;(global.fetch as jest.Mock).mockImplementation(async () => { throw new Error('') - })) + }) }) beforeEach(() => { - hasEncounteredError.value=false + hasEncounteredError.value = false mockErrorFetchingNotify.mockClear() }) @@ -47,7 +47,7 @@ describe('stop notify after encountered error', () => { }) test('already encountered', async () => { - hasEncounteredError.value=true + hasEncounteredError.value = true expect(hasEncounteredError.value).toBe(true) await updateCourses() expect(mockErrorFetchingNotify).not.toHaveBeenCalled() From af1f536ae28ed7c3eafdc492dfe8bbe33b47b432 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Mon, 12 Aug 2024 17:19:04 +0700 Subject: [PATCH 6/8] refactor updateAll; change logging commands to only logging names; make getCommands concurrent, remove unneccessary logging --- src/discord/getCommands.ts | 4 ++-- src/scraper/updateAll.ts | 14 ++++++++++---- src/server.ts | 2 +- src/utils/updateHandler.ts | 1 - 4 files changed, 13 insertions(+), 8 deletions(-) diff --git a/src/discord/getCommands.ts b/src/discord/getCommands.ts index 59623fe..956bff9 100644 --- a/src/discord/getCommands.ts +++ b/src/discord/getCommands.ts @@ -5,10 +5,10 @@ import DiscordCommandHandler from '../interfaces/DiscordCommandHandler' export async function getCommands(): Promise { const commands: DiscordCommandHandler = {} const commandFiles = fs.readdirSync(path.join(__dirname, '../commands')) - for (const file of commandFiles) { + await Promise.all(commandFiles.map(async (file)=>{ const { default: command } = await import(`../commands/${file}`) commands[command.name] = command - } + })) return commands } diff --git a/src/scraper/updateAll.ts b/src/scraper/updateAll.ts index 6a49eed..69e6863 100644 --- a/src/scraper/updateAll.ts +++ b/src/scraper/updateAll.ts @@ -43,11 +43,14 @@ export async function updateAll(): Promise> { for (const [course, assignments] of coursesWithAssignments) { // const newCourseLine = `\n- ${course.title}` const newCourseLine = format(COURSE_MESSAGE_PATTERN, course.title) - currentMessageSize.value += [...newCourseLine].length - const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE + const newAssignmentLineSize = [...newCourseLine].length + const hasExceeded = + currentMessageSize.value + newAssignmentLineSize > + MAX_DISCORD_MESSAGE_SIZE if (hasExceeded) { pushAndReinitialize(messages, currentMessage, currentMessageSize) } + currentMessageSize.value += [...newCourseLine].length for (const assignment of assignments) { // const newAssignmentLine = `\n - [${assignment.assignmentName}](https://www.mycourseville.com/?q=courseville/worksheet/${course.mcvID}/${assignment.assignmentID})` const newAssignmentLine = format( @@ -56,8 +59,10 @@ export async function updateAll(): Promise> { course.mcvID, assignment.assignmentID ) - currentMessageSize.value += [...newAssignmentLine].length - const hasExceeded = currentMessageSize.value > MAX_DISCORD_MESSAGE_SIZE + const newAssignmentLineSize = [...newAssignmentLine].length + const hasExceeded = + currentMessageSize.value + newAssignmentLineSize > + MAX_DISCORD_MESSAGE_SIZE const isFirstAssignment = assignments[0] == assignment if (hasExceeded) { pushAndReinitialize(messages, currentMessage, currentMessageSize) @@ -67,6 +72,7 @@ export async function updateAll(): Promise> { currentMessage.value += newCourseLine } currentMessage.value += newAssignmentLine + currentMessageSize.value += newAssignmentLineSize } } messages.push(currentMessage.value) diff --git a/src/server.ts b/src/server.ts index 748a1e6..627bd86 100644 --- a/src/server.ts +++ b/src/server.ts @@ -72,7 +72,7 @@ export async function start() { await registerCommands(commands) // let channels = await db.getAllChannels(); - console.log(commands) + console.log("found commands",Object.keys(commands)) client.login(env.DISCORD_TOKEN) } diff --git a/src/utils/updateHandler.ts b/src/utils/updateHandler.ts index d1ef1a0..d41e91d 100644 --- a/src/utils/updateHandler.ts +++ b/src/utils/updateHandler.ts @@ -16,7 +16,6 @@ export default async function updateHandler(): Promise> { let messages: Array = [] try { messages = await updateAll() - console.log('messages', messages) } catch (e) { console.log(e) return none From 37dae2846ef5e69dca2d020aa54327924dccfe3f Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Mon, 12 Aug 2024 17:22:15 +0700 Subject: [PATCH 7/8] mock env in updateAll --- tests/unit/updateAll.unit.test.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/tests/unit/updateAll.unit.test.ts b/tests/unit/updateAll.unit.test.ts index c518d6c..0c13cfb 100644 --- a/tests/unit/updateAll.unit.test.ts +++ b/tests/unit/updateAll.unit.test.ts @@ -3,6 +3,10 @@ const mockGetAllCoursesOfTargetSemester = jest.fn() const mockUpdateAssignments = jest.fn() jest.mock('@/scraper/updateCourses') +jest.mock('@/env/env', () => { + return {} +}) + jest.mock('@/database/database', () => { return { getAllCoursesOfTargetSemester: mockGetAllCoursesOfTargetSemester, From 9939d88485e7c2257252d331c4809fd66c8cfa65 Mon Sep 17 00:00:00 2001 From: NewBieCoderXD Date: Mon, 12 Aug 2024 17:24:16 +0700 Subject: [PATCH 8/8] fix style check --- src/discord/getCommands.ts | 10 ++++++---- src/server.ts | 2 +- 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/src/discord/getCommands.ts b/src/discord/getCommands.ts index 956bff9..c515319 100644 --- a/src/discord/getCommands.ts +++ b/src/discord/getCommands.ts @@ -5,10 +5,12 @@ import DiscordCommandHandler from '../interfaces/DiscordCommandHandler' export async function getCommands(): Promise { const commands: DiscordCommandHandler = {} const commandFiles = fs.readdirSync(path.join(__dirname, '../commands')) - await Promise.all(commandFiles.map(async (file)=>{ - const { default: command } = await import(`../commands/${file}`) - commands[command.name] = command - })) + await Promise.all( + commandFiles.map(async (file) => { + const { default: command } = await import(`../commands/${file}`) + commands[command.name] = command + }) + ) return commands } diff --git a/src/server.ts b/src/server.ts index 627bd86..7405cd1 100644 --- a/src/server.ts +++ b/src/server.ts @@ -72,7 +72,7 @@ export async function start() { await registerCommands(commands) // let channels = await db.getAllChannels(); - console.log("found commands",Object.keys(commands)) + console.log('found commands', Object.keys(commands)) client.login(env.DISCORD_TOKEN) }