From c4b4449fd4edd4f78c97f25bc3df20ff461ea897 Mon Sep 17 00:00:00 2001 From: Avior Date: Thu, 4 Nov 2021 10:02:26 +0100 Subject: [PATCH] feature: Implement new Server infrastructure with GraphQL. (#132) * Added new compiler to db Signed-off-by: Avior * Add compiled DB to artifacts Signed-off-by: Avior * Fixed space error Signed-off-by: Avior * Fixed? Signed-off-by: Avior * Update node.js.yml * Update node.js.yml * Made change so the db is no longer dependent on the SDK Signed-off-by: Avior * f Signed-off-by: Avior * Fixed artifact Signed-off-by: Avior * U Signed-off-by: Avior * \Changed folder Signed-off-by: Avior * Fixede? Signed-off-by: Avior * Try with everything * saved the file ;) * ignore compiler * Fixed prebuild being run again * Fixed public folder Signed-off-by: Avior * fixed graphql file Signed-off-by: Avior * fixed? Signed-off-by: Avior * Check tree because life is potato Signed-off-by: Avior * this is harder Signed-off-by: Avior * f Signed-off-by: Avior * Fixed? Signed-off-by: Avior * r Signed-off-by: Avior * fd Signed-off-by: Avior * added back context Signed-off-by: Avior * ah Signed-off-by: Avior * AAH Signed-off-by: Avior * AAAH Signed-off-by: Avior * ffffffffffffffffff Signed-off-by: Avior * fix: Changed the default builder Signed-off-by: Avior * Removed useless tree function Signed-off-by: Avior --- .dockerignore | 22 + .github/workflows/build.yml | 58 + .github/workflows/node.js.yml | 48 - .gitignore | 1 + Dockerfile | 44 + README.md | 8 +- meta/definitions/api.d.ts | 277 +++ meta/definitions/graphql.gql | 167 ++ package-lock.json | 316 ++- package.json | 10 +- server/.gitignore | 6 + server/README.md | 5 + server/compiler/compilerInterfaces.d.ts | 7 + server/compiler/endpoints/cards.ts | 30 + server/compiler/endpoints/series.ts | 33 + server/compiler/endpoints/sets.ts | 57 + server/compiler/index.ts | 63 + server/compiler/utils/cardUtil.ts | 153 ++ server/compiler/utils/serieUtil.ts | 66 + server/compiler/utils/setUtil.ts | 104 + server/compiler/utils/translationUtil.ts | 31 + server/compiler/utils/util.ts | 59 + server/package-lock.json | 2840 ++++++++++++++++++++++ server/package.json | 38 + server/public/favicon.ico | Bin 0 -> 15086 bytes server/public/v2/.gitkeep | 0 server/src/V2/Components/Card.ts | 112 + server/src/V2/Components/Serie.ts | 66 + server/src/V2/Components/Set.ts | 89 + server/src/V2/endpoints/jsonEndpoints.ts | 212 ++ server/src/V2/graphql/index.ts | 22 + server/src/V2/graphql/resolver.ts | 49 + server/src/index.ts | 35 + server/src/interfaces.d.ts | 11 + server/src/util.ts | 57 + server/tsconfig.json | 7 + tsconfig.json | 5 +- 37 files changed, 5038 insertions(+), 70 deletions(-) create mode 100644 .dockerignore create mode 100644 .github/workflows/build.yml delete mode 100644 .github/workflows/node.js.yml create mode 100644 Dockerfile create mode 100644 meta/definitions/api.d.ts create mode 100644 meta/definitions/graphql.gql create mode 100644 server/.gitignore create mode 100644 server/README.md create mode 100644 server/compiler/compilerInterfaces.d.ts create mode 100644 server/compiler/endpoints/cards.ts create mode 100644 server/compiler/endpoints/series.ts create mode 100644 server/compiler/endpoints/sets.ts create mode 100644 server/compiler/index.ts create mode 100644 server/compiler/utils/cardUtil.ts create mode 100644 server/compiler/utils/serieUtil.ts create mode 100644 server/compiler/utils/setUtil.ts create mode 100644 server/compiler/utils/translationUtil.ts create mode 100644 server/compiler/utils/util.ts create mode 100644 server/package-lock.json create mode 100644 server/package.json create mode 100644 server/public/favicon.ico create mode 100644 server/public/v2/.gitkeep create mode 100644 server/src/V2/Components/Card.ts create mode 100644 server/src/V2/Components/Serie.ts create mode 100644 server/src/V2/Components/Set.ts create mode 100644 server/src/V2/endpoints/jsonEndpoints.ts create mode 100644 server/src/V2/graphql/index.ts create mode 100644 server/src/V2/graphql/resolver.ts create mode 100644 server/src/index.ts create mode 100644 server/src/interfaces.d.ts create mode 100644 server/src/util.ts create mode 100644 server/tsconfig.json diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 000000000..600783217 --- /dev/null +++ b/.dockerignore @@ -0,0 +1,22 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +node_modules + + +# misc +.DS_Store +.env* + +# debug +npm-debug.log* + +dist/ +*.asc + +/data/**/*.js + + +# production +/server/dist +/server/generated diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml new file mode 100644 index 000000000..cf68b4657 --- /dev/null +++ b/.github/workflows/build.yml @@ -0,0 +1,58 @@ +name: Build + +# This workflow uses actions that are not certified by GitHub. +# They are provided by a third-party and are governed by +# separate terms of service, privacy policy, and support +# documentation. + +on: + push: + branches: [ master ] + # Publish semver tags as releases. + tags: [ 'v*.*.*' ] + pull_request: + branches: [ master ] + +env: + # Use docker.io for Docker Hub if empty + REGISTRY: ghcr.io + # github.repository as / + IMAGE_NAME: tcgdex/server + + +jobs: + build: + + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@v2 + + # Login against a Docker registry except on PR + # https://github.com/docker/login-action + - name: Log into registry ${{ env.REGISTRY }} + if: github.event_name != 'pull_request' + uses: docker/login-action@v1 + with: + registry: ${{ env.REGISTRY }} + username: ${{ github.actor }} + password: ${{ secrets.GITHUB_TOKEN }} + + # Extract metadata (tags, labels) for Docker + # https://github.com/docker/metadata-action + - name: Extract Docker metadata + id: meta + uses: docker/metadata-action@v3 + with: + images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} + + # Build and push Docker image with Buildx (don't push on PR) + # https://github.com/docker/build-push-action + - name: Build and push + uses: docker/build-push-action@v2 + with: + context: . + push: ${{ github.event_name != 'pull_request' }} + tags: ${{ steps.meta.outputs.tags }} + labels: ${{ steps.meta.outputs.labels }} diff --git a/.github/workflows/node.js.yml b/.github/workflows/node.js.yml deleted file mode 100644 index f88f7d488..000000000 --- a/.github/workflows/node.js.yml +++ /dev/null @@ -1,48 +0,0 @@ -# This workflow will do a clean install of node dependencies, build the source code and run tests across different versions of node -# For more information see: https://help.github.com/actions/language-and-framework-guides/using-nodejs-with-github-actions - -name: Compile - -on: - push: - branches: - - master - paths: - - 'data/**' - - 'meta/**' - pull_request: - paths: - - 'data/**' - - 'meta/**' - -jobs: - test: - name: Compilation Test - runs-on: ubuntu-latest - - steps: - - uses: actions/checkout@v2 - - - name: Setup NodeJS - uses: actions/setup-node@v2.4.1 - with: - node-version: 16.x - - - name: Install packages - run: npm ci - - - name: Build the database - run: npm run compile - - push: - name: Launch the compiler - needs: test - if: ${{ github.event_name == 'push' }} - runs-on: ubuntu-latest - steps: - - name: Compiler Dispatch - uses: peter-evans/repository-dispatch@v1 - with: - token: ${{ secrets.PAC }} - repository: tcgdex/compiler - event-type: card-database-update diff --git a/.gitignore b/.gitignore index 1c0c2791d..4207fda49 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ *.js node_modules/ +dist/ diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 000000000..3d86a5385 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,44 @@ +FROM node:alpine as BUILD_IMAGE + +WORKDIR /app + +ADD package.json package-lock.json ./ +ADD server/package.json server/package-lock.json ./server/ + +# install dependencies +RUN npm ci && \ +cd server && \ +npm ci + +# Add project files +ADD . . + +# build +RUN npm run compile && \ +cd server && \ +npm run compile && \ +npm run build + +# remove dev dependencies +RUN npm prune --production && \ +cd server && \ +npm prune --production + +# go to another VM +FROM node:alpine + +# go to folder +WORKDIR /app + +# copy from build image +COPY --from=BUILD_IMAGE /app/server/generated ./generated +COPY --from=BUILD_IMAGE /app/server/node_modules ./node_modules +COPY --from=BUILD_IMAGE /app/server/dist ./dist +COPY --from=BUILD_IMAGE /app/server/public ./public +COPY --from=BUILD_IMAGE /app/server/package.json ./package.json + +# Expose port +EXPOSE 3000 + +# run it ! +CMD ["npm", "run", "start"] diff --git a/README.md b/README.md index 8e5bc7729..904d6a24a 100644 --- a/README.md +++ b/README.md @@ -59,6 +59,12 @@ You want to ask something or you just want to discuss anything from an idea/impr Well you can join us on [Discord](https://discord.gg/NehYTAhsZE) ! +## Project Architecture + +- `/data`: Contains Cards Data architectured around `Series/Set/localId` folders +- `/meta`: Contains misc informations for the project (more details in the README.md in `/meta`) +- `/server`: The server that allow to communicate with the database + ## Contributing See [CONTRIBUTING.md](./CONTRIBUTING.md) @@ -98,4 +104,4 @@ Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/d -This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! \ No newline at end of file +This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! diff --git a/meta/definitions/api.d.ts b/meta/definitions/api.d.ts new file mode 100644 index 000000000..b89d52fe1 --- /dev/null +++ b/meta/definitions/api.d.ts @@ -0,0 +1,277 @@ +/** + * This file define how the API is architectured for the Compiler and the Javascript/Typescript SDK + */ + +/** + * /series endpoint + */ +export interface SerieResume { + id: string; + name: string; + logo?: string; +} + +/** + * /series/:id endpoint + */ +export interface Serie extends SerieResume { + sets: Array; +} + +interface variants { + normal?: boolean; + reverse?: boolean; + holo?: boolean; + firstEdition?: boolean; +} + +export interface SetResume { + id: string; + name: string; + logo?: string; + symbol?: string; + cardCount: { + /** + * total of number of cards + */ + total: number; + /** + * number of cards officialy (on the bottom of each cards) + */ + official: number; + }; +} + +/** + * /sets/:id + */ +export interface Set extends SetResume { + serie: SerieResume; + tcgOnline?: string; + variants?: variants; + releaseDate: string; + /** + * Designate if the set is usable in tournaments + * + * Note: this is specific to the set and if a + * card is banned from the set it will still be true + */ + legal: { + /** + * Ability to play in standard tournaments + */ + standard: boolean; + /** + * Ability to play in expanded tournaments + */ + expanded: boolean; + }; + cardCount: { + /** + * total of number of cards + */ + total: number; + /** + * number of cards officialy (on the bottom of each cards) + */ + official: number; + /** + * number of cards having a normal version + */ + normal: number; + /** + * number of cards having an reverse version + */ + reverse: number; + /** + * number of cards having an holo version + */ + holo: number; + /** + * Number of possible cards + */ + firstEd?: number; + }; + cards: Array; +} +export interface CardResume { + id: string; + localId: string; + /** + * Card Name (Including the suffix if next to card name) + */ + name: string; + image?: string; +} + +/** + * /cards/:id + * /sets/:set/:localId + */ +export interface Card extends CardResume { + /** + * Card illustrator + */ + illustrator?: string; + /** + * Card Rarity + * + * - None https://www.tcgdex.net/database/sm/smp/SM01 + * - Common https://www.tcgdex.net/database/xy/xy9/1 + * - Uncommon https://www.tcgdex.net/database/xy/xy9/2 + * - Rare https://www.tcgdex.net/database/xy/xy9/3 + * - Ultra Rare + * - Secret Rare + */ + rarity: string; + /** + * Card Category + * + * - Pokemon + * - Trainer + * - Energy + */ + category: string; + /** + * Card Variants (Override Set Variants) + */ + variants?: variants; + /** + * Card Set + */ + set: SetResume; + /** + * Pokemon only elements + */ + /** + * Pokemon Pokedex ID + */ + dexId?: Array; + /** + * Pokemon HP + */ + hp?: number; + /** + * Pokemon Types + * ex for multiple https://www.tcgdex.net/database/ex/ex13/17 + */ + types?: Array; + /** + * Pokemon Sub Evolution + */ + evolveFrom?: string; + /** + * Pokemon Weight + */ + weight?: string; + /** + * Pokemon Description + */ + description?: string; + /** + * Level of the Pokemon + * + * NOTE: can be equal to 'X' when the pokemon is a LEVEL-UP one + */ + level?: number | string; + /** + * Pokemon Stage + * + * - Basic https://www.tcgdex.net/database/xy/xy9/1 + * - BREAK https://www.tcgdex.net/database/xy/xy9/18 + * - LEVEL-UP https://www.tcgdex.net/database/dp/dp1/121 + * - MEGA https://www.tcgdex.net/database/xy/xy1/2 + * - RESTORED https://www.tcgdex.net/database/bw/bw5/53 + * - Stage1 https://www.tcgdex.net/database/xy/xy9/2 + * - Stage2 https://www.tcgdex.net/database/xy/xy9/3 + * - VMAX https://www.tcgdex.net/database/swsh/swsh1/50 + */ + stage?: string; + /** + * Card Suffix + * + * - EX https://www.tcgdex.net/database/ex/ex2/94 + * - GX https://www.tcgdex.net/database/sm/sm12/4 + * - V https://www.tcgdex.net/database/swsh/swsh1/1 + * - Legend https://www.tcgdex.net/database/hgss/hgss1/114 + * - Prime https://www.tcgdex.net/database/hgss/hgss2/85 + * - SP https://www.tcgdex.net/database/pl/pl1/7 + * - TAG TEAM-GX https://www.tcgdex.net/database/sm/sm12/226 + */ + suffix?: string; + /** + * Pokemon Held Item + * + * ex https://www.tcgdex.net/database/dp/dp2/75 + */ + item?: { + name: string; + effect: string; + }; + /** + * Pokemon Abilities + * + * multi abilities ex https://www.tcgdex.net/database/ex/ex15/10 + */ + abilities?: Array<{ + type: string; + name: string; + effect: string; + }>; + /** + * Pokemon Attacks + */ + attacks?: Array<{ + cost?: Array; + name: string; + effect?: string; + damage?: string | number; + }>; + /** + * Pokemon Weaknesses + */ + weaknesses?: Array<{ + type: string; + value?: string; + }>; + resistances?: Array<{ + type: string; + value?: string; + }>; + retreat?: number; + effect?: string; + trainerType?: string; + energyType?: string; + /** + * Define the rotation mark on cards >= Sword & Shield + */ + regulationMark?: string; + /** + * Card ability to be played in official tournaments + * + * Note: all cards are avaialable to play in unlimited tournaments + */ + legal: { + /** + * Ability to play in standard tournaments + */ + standard: boolean; + /** + * Ability to play in expanded tournaments + */ + expanded: boolean; + }; +} + +/** + * String Endpoint List + */ +export declare type StringEndpointList = Array; + +/** + * StringEndpoint + */ +export interface StringEndpoint { + name: string; + cards: Array; +} diff --git a/meta/definitions/graphql.gql b/meta/definitions/graphql.gql new file mode 100644 index 000000000..974235e58 --- /dev/null +++ b/meta/definitions/graphql.gql @@ -0,0 +1,167 @@ +################## +# Global # +################## + +# Locale Directive ex: {sets @locale(fr)} +directive @locale ( + lang: String! +) on FIELD + +# Queries to use on the DB +type Query { + cards(filters: CardsFilters, pagination: Pagination): [Card] + sets: [Set] + series: [Serie] + card( + id: ID!, + set: String + ): Card + set( + id: ID! + ): Set + serie( + id: ID! + ): Serie + +} + +# Pagination input +input Pagination { + page: Float! + count: Float! +} + +################## +# Card # +################## + +# Filters to be used with the Card query +input CardsFilters { + category: String + description: String + energyType: String + evolveFrom: String + hp: Float + id: ID + localId: String + dexId: Float + illustrator: String + image: String + level: Float + levelId: String + name: String + rarity: String + regulationMark: String + stage: String + suffix: String + trainerType: String + retreat: Float +} + +type Card { + abilities: [AbilitiesListItem] + attacks: [AttacksListItem] + category: String! + description: String + dexId: [Float] + energyType: String + evolveFrom: String + hp: Float + id: String! + illustrator: String + image: String + item: Item + legal: Legal! + level: Float + localId: String! + name: String! + rarity: String! + regulationMark: String + resistances: [[WeakResListItem]] + retreat: Float + set: Set! + stage: String + suffix: String + trainerType: String + types: [String] + variants: Variants + weaknesses: [[WeakResListItem]] +} + +type AbilitiesListItem { + effect: String + name: String + type: String +} + +type AttacksListItem { + cost: [String] + damage: String + effect: String + name: String! +} + +type Item { + effect: String! + name: String! +} + +type Legal { + expanded: Boolean + standard: Boolean +} + +type WeakResListItem { + type: String! + value: String +} + +type Variants { + firstEdition: Boolean! + holo: Boolean! + normal: Boolean! + reverse: Boolean! +} + +################## +# Set # +################## + +type Set { + cardCount: CardCount! + cards: [Card]! + id: String! + logo: String + name: String! + symbol: String + serie: Serie! +} + +type CardCount { + firstEd: Float + holo: Float + normal: Float + official: Float! + reverse: Float + total: Float! +} + +################## +# Serie # +################## + +type Serie { + id: String! + logo: String + name: String! + sets: [Set]! +} + +################## +# StringEndpoint # +################## + +type StringEndpoint { + cards: [Card]! + name: String! +} diff --git a/package-lock.json b/package-lock.json index 1aa9c1442..9da016ee9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,17 +1,20 @@ { - "name": "@tcgdex/db", + "name": "cards-database", "version": "2.0.0", "lockfileVersion": 2, "requires": true, "packages": { "": { - "name": "@tcgdex/db", - "version": "2.0.0", - "license": "MIT", "devDependencies": { + "@dzeio/object-util": "^1.2.0", + "@tcgdex/sdk": "^2.4.4", + "@types/glob": "^7.1.4", "@types/jscodeshift": "^0.11.2", "@types/node": "^16.11.6", + "@types/node-fetch": "2.5.12", + "glob": "^7.1.7", "jscodeshift": "^0.13.0", + "node-fetch": "2.6.5", "ts-node": "^10.4.0", "typescript": "^4.4.4" } @@ -1788,6 +1791,25 @@ "node": ">=12" } }, + "node_modules/@dzeio/object-util": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@dzeio/object-util/-/object-util-1.4.2.tgz", + "integrity": "sha512-dqBQo9aUNUi9XVAMJl3dZgA+wKAQgehdvmMX+04X29ovWCYcWuXvq+8HEzckQXMPQ/xYw+D/fV2S96QYaHzX1g==", + "dev": true + }, + "node_modules/@tcgdex/sdk": { + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/@tcgdex/sdk/-/sdk-2.4.5.tgz", + "integrity": "sha512-NOzMffhUdRdlrMrd6PCQ7A/bbwlkY9DiMbR8O+JWjRKg/q/E2IXKewYRCFzFvCXe/pEs+gbYj/Jhta49OzYxHg==", + "dev": true, + "dependencies": { + "isomorphic-unfetch": "^3.1.0", + "unfetch": "^4.2.0" + }, + "engines": { + "node": ">=12" + } + }, "node_modules/@tsconfig/node10": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.8.tgz", @@ -1812,6 +1834,16 @@ "integrity": "sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA==", "dev": true }, + "node_modules/@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "dependencies": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, "node_modules/@types/jscodeshift": { "version": "0.11.2", "resolved": "https://registry.npmjs.org/@types/jscodeshift/-/jscodeshift-0.11.2.tgz", @@ -1822,12 +1854,28 @@ "recast": "^0.20.3" } }, + "node_modules/@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, "node_modules/@types/node": { "version": "16.11.6", "resolved": "https://registry.npmjs.org/@types/node/-/node-16.11.6.tgz", "integrity": "sha512-ua7PgUoeQFjmWPcoo9khiPum3Pd60k4/2ZGXt18sm2Slk0W0xZTqt5Y0Ny1NyBiN1EVQ/+FaF9NcY4Qe6rwk5w==", "dev": true }, + "node_modules/@types/node-fetch": { + "version": "2.5.12", + "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.12.tgz", + "integrity": "sha512-MKgC4dlq4kKNa/mYrwpKfzQMB5X3ee5U6fSprkKpToBqBmX4nFZL9cW5jl6sWn+xpRJ7ypWh2yyqqr8UUCstSw==", + "dev": true, + "dependencies": { + "@types/node": "*", + "form-data": "^3.0.0" + } + }, "node_modules/acorn": { "version": "8.5.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.5.0.tgz", @@ -1924,6 +1972,12 @@ "node": ">=4" } }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, "node_modules/atob": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", @@ -2215,6 +2269,18 @@ "node": ">=0.1.90" } }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/commondir": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", @@ -2395,6 +2461,15 @@ "node": ">=0.10.0" } }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/diff": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", @@ -2571,6 +2646,20 @@ "node": ">=0.10.0" } }, + "node_modules/form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dev": true, + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/fragment-cache": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", @@ -2628,9 +2717,9 @@ } }, "node_modules/glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", "dev": true, "dependencies": { "fs.realpath": "^1.0.0", @@ -2885,6 +2974,16 @@ "node": ">=0.10.0" } }, + "node_modules/isomorphic-unfetch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/isomorphic-unfetch/-/isomorphic-unfetch-3.1.0.tgz", + "integrity": "sha512-geDJjpoZ8N0kWexiwkX8F9NkTsXhetLPVbZFQ+JTW239QNOwvB0gniuR1Wc6f0AMTn7/mFGyXvHTifrCp/GH8Q==", + "dev": true, + "dependencies": { + "node-fetch": "^2.6.1", + "unfetch": "^4.2.0" + } + }, "node_modules/js-tokens": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", @@ -3091,6 +3190,27 @@ "node": ">=0.10.0" } }, + "node_modules/mime-db": { + "version": "1.50.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz", + "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.33", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz", + "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==", + "dev": true, + "dependencies": { + "mime-db": "1.50.0" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/minimatch": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", @@ -3224,6 +3344,18 @@ "node": ">= 0.10.5" } }, + "node_modules/node-fetch": { + "version": "2.6.5", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.5.tgz", + "integrity": "sha512-mmlIVHJEu5rnIxgEgez6b9GgWXbkZj5YZ7fx+2r94a2E+Uirsp6HsPTPlomfdHtpt/B0cdKviwkoaM6pyvUOpQ==", + "dev": true, + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + } + }, "node_modules/node-modules-regexp": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/node-modules-regexp/-/node-modules-regexp-1.0.0.tgz", @@ -3990,6 +4122,12 @@ "node": ">=0.10.0" } }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=", + "dev": true + }, "node_modules/ts-node": { "version": "10.4.0", "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.4.0.tgz", @@ -4050,6 +4188,12 @@ "node": ">=4.2.0" } }, + "node_modules/unfetch": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/unfetch/-/unfetch-4.2.0.tgz", + "integrity": "sha512-F9p7yYCn6cIW9El1zi0HI6vqpeIvBsr3dSuRO6Xuppb1u5rXpCPmMvLSyECLhybr9isec8Ohl0hPekMVrEinDA==", + "dev": true + }, "node_modules/unicode-canonical-property-names-ecmascript": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz", @@ -4173,6 +4317,22 @@ "node": ">=0.10.0" } }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=", + "dev": true + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", + "dev": true, + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, "node_modules/wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", @@ -5465,6 +5625,22 @@ "@cspotcode/source-map-consumer": "0.8.0" } }, + "@dzeio/object-util": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@dzeio/object-util/-/object-util-1.4.2.tgz", + "integrity": "sha512-dqBQo9aUNUi9XVAMJl3dZgA+wKAQgehdvmMX+04X29ovWCYcWuXvq+8HEzckQXMPQ/xYw+D/fV2S96QYaHzX1g==", + "dev": true + }, + "@tcgdex/sdk": { + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/@tcgdex/sdk/-/sdk-2.4.5.tgz", + "integrity": "sha512-NOzMffhUdRdlrMrd6PCQ7A/bbwlkY9DiMbR8O+JWjRKg/q/E2IXKewYRCFzFvCXe/pEs+gbYj/Jhta49OzYxHg==", + "dev": true, + "requires": { + "isomorphic-unfetch": "^3.1.0", + "unfetch": "^4.2.0" + } + }, "@tsconfig/node10": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.8.tgz", @@ -5489,6 +5665,16 @@ "integrity": "sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA==", "dev": true }, + "@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "requires": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, "@types/jscodeshift": { "version": "0.11.2", "resolved": "https://registry.npmjs.org/@types/jscodeshift/-/jscodeshift-0.11.2.tgz", @@ -5499,12 +5685,28 @@ "recast": "^0.20.3" } }, + "@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, "@types/node": { "version": "16.11.6", "resolved": "https://registry.npmjs.org/@types/node/-/node-16.11.6.tgz", "integrity": "sha512-ua7PgUoeQFjmWPcoo9khiPum3Pd60k4/2ZGXt18sm2Slk0W0xZTqt5Y0Ny1NyBiN1EVQ/+FaF9NcY4Qe6rwk5w==", "dev": true }, + "@types/node-fetch": { + "version": "2.5.12", + "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.12.tgz", + "integrity": "sha512-MKgC4dlq4kKNa/mYrwpKfzQMB5X3ee5U6fSprkKpToBqBmX4nFZL9cW5jl6sWn+xpRJ7ypWh2yyqqr8UUCstSw==", + "dev": true, + "requires": { + "@types/node": "*", + "form-data": "^3.0.0" + } + }, "acorn": { "version": "8.5.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.5.0.tgz", @@ -5571,6 +5773,12 @@ "tslib": "^2.0.1" } }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, "atob": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", @@ -5803,6 +6011,15 @@ "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", "dev": true }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "requires": { + "delayed-stream": "~1.0.0" + } + }, "commondir": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", @@ -5954,6 +6171,12 @@ } } }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true + }, "diff": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", @@ -6086,6 +6309,17 @@ "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", "dev": true }, + "form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } + }, "fragment-cache": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", @@ -6131,9 +6365,9 @@ "dev": true }, "glob": { - "version": "7.1.6", - "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", - "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", "dev": true, "requires": { "fs.realpath": "^1.0.0", @@ -6329,6 +6563,16 @@ "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", "dev": true }, + "isomorphic-unfetch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/isomorphic-unfetch/-/isomorphic-unfetch-3.1.0.tgz", + "integrity": "sha512-geDJjpoZ8N0kWexiwkX8F9NkTsXhetLPVbZFQ+JTW239QNOwvB0gniuR1Wc6f0AMTn7/mFGyXvHTifrCp/GH8Q==", + "dev": true, + "requires": { + "node-fetch": "^2.6.1", + "unfetch": "^4.2.0" + } + }, "js-tokens": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", @@ -6491,6 +6735,21 @@ } } }, + "mime-db": { + "version": "1.50.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz", + "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A==", + "dev": true + }, + "mime-types": { + "version": "2.1.33", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz", + "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==", + "dev": true, + "requires": { + "mime-db": "1.50.0" + } + }, "minimatch": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", @@ -6604,6 +6863,15 @@ "minimatch": "^3.0.2" } }, + "node-fetch": { + "version": "2.6.5", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.5.tgz", + "integrity": "sha512-mmlIVHJEu5rnIxgEgez6b9GgWXbkZj5YZ7fx+2r94a2E+Uirsp6HsPTPlomfdHtpt/B0cdKviwkoaM6pyvUOpQ==", + "dev": true, + "requires": { + "whatwg-url": "^5.0.0" + } + }, "node-modules-regexp": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/node-modules-regexp/-/node-modules-regexp-1.0.0.tgz", @@ -7218,6 +7486,12 @@ "repeat-string": "^1.6.1" } }, + "tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=", + "dev": true + }, "ts-node": { "version": "10.4.0", "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.4.0.tgz", @@ -7250,6 +7524,12 @@ "integrity": "sha512-DqGhF5IKoBl8WNf8C1gu8q0xZSInh9j1kJJMqT3a94w1JzVaBU4EXOSMrz9yDqMT0xt3selp83fuFMQ0uzv6qA==", "dev": true }, + "unfetch": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/unfetch/-/unfetch-4.2.0.tgz", + "integrity": "sha512-F9p7yYCn6cIW9El1zi0HI6vqpeIvBsr3dSuRO6Xuppb1u5rXpCPmMvLSyECLhybr9isec8Ohl0hPekMVrEinDA==", + "dev": true + }, "unicode-canonical-property-names-ecmascript": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.0.tgz", @@ -7346,6 +7626,22 @@ "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", "dev": true }, + "webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=", + "dev": true + }, + "whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", + "dev": true, + "requires": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, "wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", diff --git a/package.json b/package.json index 24363ce1f..72108b105 100644 --- a/package.json +++ b/package.json @@ -1,18 +1,10 @@ { - "name": "@tcgdex/db", - "version": "2.0.0", - "description": "Database of pokemon TCG", - "repository": "https://github.com/tcgdex/cards-database.git", - "author": "Avior ", - "license": "MIT", - "private": false, + "private": true, "scripts": { "compile": "tsc --project tsconfig.json" }, "devDependencies": { - "@types/jscodeshift": "^0.11.2", "@types/node": "^16.11.6", - "jscodeshift": "^0.13.0", "ts-node": "^10.4.0", "typescript": "^4.4.4" } diff --git a/server/.gitignore b/server/.gitignore new file mode 100644 index 000000000..b5128cd15 --- /dev/null +++ b/server/.gitignore @@ -0,0 +1,6 @@ +/node_modules +/dist +/generated +/public/**/graphql.gql +/public/**/api.d.ts +/public/**/openapi.yaml diff --git a/server/README.md b/server/README.md new file mode 100644 index 000000000..b236ffc15 --- /dev/null +++ b/server/README.md @@ -0,0 +1,5 @@ +# TCGdex/server + +The server running the TCGdex API. + +More informations at https://github.com/tcgdex/cards-database#pokemon-tcgdex diff --git a/server/compiler/compilerInterfaces.d.ts b/server/compiler/compilerInterfaces.d.ts new file mode 100644 index 000000000..cc9fb1f84 --- /dev/null +++ b/server/compiler/compilerInterfaces.d.ts @@ -0,0 +1,7 @@ +// eslint-disable-next-line @typescript-eslint/ban-types +export interface Endpoint { + index(common: C): Promise + item(common: C): Promise | undefined> + sub?(common: C, item: string): Promise | undefined> + common?(): Promise +} diff --git a/server/compiler/endpoints/cards.ts b/server/compiler/endpoints/cards.ts new file mode 100644 index 000000000..68e9e5748 --- /dev/null +++ b/server/compiler/endpoints/cards.ts @@ -0,0 +1,30 @@ +import { Card as CardSingle, CardResume } from '../../../meta/definitions/api' +import { Card, Languages } from '../../../interfaces' +import { Endpoint } from '../compilerInterfaces' +import { cardToCardSimple, cardToCardSingle, getCards } from '../utils/cardUtil' + +type CardList = Array + +export default class implements Endpoint, Array<[string, Card]>> { + + public constructor( + private lang: keyof Languages + ) {} + + public async index(common: Array<[string, Card]>): Promise { + return Promise.all(common.map((c) => cardToCardSimple(c[0], c[1], this.lang))) + } + + public async item(common: Array<[string, Card]>): Promise> { + const items: Record = {} + for await (const card of common) { + items[`${card[1].set.id}-${card[0]}`] = await cardToCardSingle(card[0], card[1], this.lang) + } + return items + } + + public async common(): Promise> { + return getCards(this.lang) + } + +} diff --git a/server/compiler/endpoints/series.ts b/server/compiler/endpoints/series.ts new file mode 100644 index 000000000..08db16b5f --- /dev/null +++ b/server/compiler/endpoints/series.ts @@ -0,0 +1,33 @@ +import { Serie as SerieSingle, SerieResume } from '../../../meta/definitions/api' +import { Languages, Serie } from '../../../interfaces' +import { Endpoint } from '../compilerInterfaces' +import { getSeries, serieToSerieSimple, serieToSerieSingle } from '../utils/serieUtil' + +type SerieList = Array + +export default class implements Endpoint, Array> { + + public constructor( + private lang: keyof Languages + ) {} + + public async index(common: Array): Promise> { + return Promise.all(common.map((s) => serieToSerieSimple(s, this.lang))) + } + + public async item(common: Array): Promise> { + const items: Record = {} + for await (const val of common) { + const gen = await serieToSerieSingle(val, this.lang) + const name = val.name[this.lang] as string + items[name] = gen + items[val.id] = gen + } + return items + } + + public async common(): Promise> { + return getSeries(this.lang) + } + +} diff --git a/server/compiler/endpoints/sets.ts b/server/compiler/endpoints/sets.ts new file mode 100644 index 000000000..185f87112 --- /dev/null +++ b/server/compiler/endpoints/sets.ts @@ -0,0 +1,57 @@ +import { Set as SetSingle, Card as CardSingle, SetResume } from '../../../meta/definitions/api' +import { getSets, isSetAvailable, setToSetSimple, setToSetSingle } from '../utils/setUtil' +import { Languages, Set } from '../../../interfaces' +import { Endpoint } from '../compilerInterfaces' +import { cardToCardSingle, getCards } from '../utils/cardUtil' + +type SetList = Array + +export default class implements Endpoint> { + + public constructor( + private lang: keyof Languages + ) {} + + public async index(common: Array): Promise { + const sets = common + .sort((a, b) => a.releaseDate > b.releaseDate ? 1 : -1) + + const tmp: SetList = await Promise.all(sets.map((el) => setToSetSimple(el, this.lang))) + + return tmp + } + + public async item(common: Array): Promise> { + const sets = await Promise.all(common + .map((set) => setToSetSingle(set, this.lang))) + const res: Record = {} + + for (const set of sets) { + res[set.name] = set + res[set.id] = set + } + + return res + } + + public async common(): Promise> { + return getSets(undefined, this.lang) + } + + public async sub(common: Array, item: string): Promise | undefined> { + const set = common.find((s) => s.name[this.lang] === item || s.id === item) + + if (!set || !isSetAvailable(set, this.lang)) { + return undefined + } + + const lit = await getCards(this.lang, set) + const list: Record = {} + for await (const card of lit) { + list[card[0]] = await cardToCardSingle(card[0], card[1], this.lang) + } + + return list + } + +} diff --git a/server/compiler/index.ts b/server/compiler/index.ts new file mode 100644 index 000000000..823ce3aa8 --- /dev/null +++ b/server/compiler/index.ts @@ -0,0 +1,63 @@ +/* eslint-disable max-statements */ +import { Endpoint } from './compilerInterfaces' +import { promises as fs } from 'fs' +import { fetchRemoteFile } from './utils/util' + +const LANGS = ['en', 'fr', 'es', 'it', 'pt', 'de'] + +const DIST_FOLDER = './generated' + +;(async () => { + const paths = (await fs.readdir('./compiler/endpoints')).filter((p) => p.endsWith('.ts')) + + console.log('Prefetching pictures') + await fetchRemoteFile('https://assets.tcgdex.net/datas.json') + + // Delete dist folder to be sure to have a clean base + try { + await fs.rm(DIST_FOLDER, {recursive: true}) + } catch {} + + + console.log('Let\'s GO !') + + // Process each languages + for await (const lang of LANGS) { + console.log('Processing', lang) + // loop through """endpoints""" + for await (const file of paths) { + + // final folder path + const folder = `${DIST_FOLDER}/${lang}` + + // Make the folder + await fs.mkdir(folder, {recursive: true}) + + // Import the """Endpoint""" + const Ep = (await import(`./endpoints/${file}`)).default + + const endpoint = new Ep(lang) as Endpoint + + console.log(file, 'Running Common') + let common: any | null = null + + if (endpoint.common) { + common = await endpoint.common() + } + + console.log(file, 'Running Item') + const item = await endpoint.item(common) + + // Write to file + await fs.writeFile(`${folder}/${file.replace('.ts', '')}.json`, JSON.stringify(item)) + + console.log(file, 'Finished Item') + } + } + + // Finally copy definitions files to the public folder :D + for await (const file of await fs.readdir('../meta/definitions')) { + await fs.copyFile('../meta/definitions/' + file, './public/v2/' + file) + } + +})() diff --git a/server/compiler/utils/cardUtil.ts b/server/compiler/utils/cardUtil.ts new file mode 100644 index 000000000..ee370b401 --- /dev/null +++ b/server/compiler/utils/cardUtil.ts @@ -0,0 +1,153 @@ +/* eslint-disable sort-keys */ +import { setToSetSimple } from './setUtil' +import { cardIsLegal, DB_PATH, fetchRemoteFile, smartGlob } from './util' +import { Set, SupportedLanguages, Card, Types } from '../../../interfaces' +import { Card as CardSingle, CardResume } from '../../../meta/definitions/api' +import translate from './translationUtil' + +export async function getCardPictures(cardId: string, card: Card, lang: SupportedLanguages): Promise { + try { + const file = await fetchRemoteFile('https://assets.tcgdex.net/datas.json') + const fileExists = Boolean(file[lang]?.[card.set.serie.id]?.[card.set.id]?.[cardId]) + if (fileExists) { + return `https://assets.tcgdex.net/${lang}/${card.set.serie.id}/${card.set.id}/${cardId}` + } + } catch { + return undefined + } + return undefined +} + +export async function cardToCardSimple(id: string, card: Card, lang: SupportedLanguages): Promise { + const cardName = card.name[lang] + if (!cardName) { + throw new Error(`Card (${card.set.id}-${id}) has no name in (${lang})`) + } + const img = await getCardPictures(id, card, lang) + return { + id: `${card.set.id}-${id}`, + image: img, + localId: id, + name: cardName + } +} + +// eslint-disable-next-line max-lines-per-function +export async function cardToCardSingle(localId: string, card: Card, lang: SupportedLanguages): Promise { + const image = await getCardPictures(localId, card, lang) + + if (!card.name[lang]) { + throw new Error(`Card (${localId}) dont exist in (${lang})`) + } + + return { + category: translate('category', card.category, lang) as any, + id: `${card.set.id}-${localId}`, + illustrator: card.illustrator, + image, + localId, + name: card.name[lang] as string, + + rarity: translate('rarity', card.rarity, lang) as any, + set: await setToSetSimple(card.set, lang), + variants: { + firstEdition: typeof card.variants?.firstEdition === 'boolean' ? card.variants.firstEdition : false, + holo: typeof card.variants?.holo === 'boolean' ? card.variants.holo : true, + normal: typeof card.variants?.normal === 'boolean' ? card.variants.normal : true, + reverse: typeof card.variants?.reverse === 'boolean' ? card.variants.reverse : true + }, + + + dexId: card.dexId, + hp: card.hp, + types: card.types?.map((t) => translate('types', t, lang)) as Array, + evolveFrom: card.evolveFrom && card.evolveFrom[lang], + weight: card.weight, + description: card.description ? card.description[lang] as string : undefined, + level: card.level, + stage: translate('stage', card.stage, lang) as any, + suffix: translate('suffix', card.suffix, lang) as any, + item: card.item ? { + name: card.item.name[lang] as string, + effect: card.item.effect[lang] as string + } : undefined, + + abilities: card.abilities?.map((el) => ({ + type: translate('abilityType', el.type, lang) as any, + name: el.name[lang] as string, + effect: el.effect[lang] as string + })), + + attacks: card.attacks?.map((el) => ({ + cost: el.cost?.map((t) => translate('types', t, lang)) as Array, + name: el.name[lang] as string, + effect: el.effect ? el.effect[lang] : undefined, + damage: el.damage + })), + weaknesses: card.weaknesses?.map((el) => ({ + type: translate('types', el.type, lang) as Types, + value: el.value + })), + + resistances: card.resistances?.map((el) => ({ + type: translate('types', el.type, lang) as Types, + value: el.value + })), + + retreat: card.retreat, + + effect: card.effect ? card.effect[lang] : undefined, + + trainerType: translate('trainerType', card.trainerType, lang) as any, + energyType: translate('energyType', card.energyType, lang) as any, + regulationMark: card.regulationMark, + + legal: { + standard: cardIsLegal('standard', card, localId), + expanded: cardIsLegal('expanded', card, localId) + } + + } +} + +/** + * + * @param setName the setname of the card + * @param id the local id of the card + * @returns [the local id, the Card object] + */ +export async function getCard(serie: string, setName: string, id: string): Promise { + return (await import(`../../${DB_PATH}/data/${serie}/${setName}/${id}.js`)).default +} + +export async function getCards(lang: SupportedLanguages, set?: Set): Promise> { + const cards = await smartGlob(`${DB_PATH}/data/${(set && set.serie.name.en) ?? '*'}/${(set && set.name.en) ?? '*'}/*.js`) + const list: Array<[string, Card]> = [] + for (const path of cards) { + const id = path.substring(path.lastIndexOf('/') + 1, path.lastIndexOf('.')) + const setName = (set && set.name.en) ?? (() => { + const part1 = path.substr(0, path.lastIndexOf(id) - 1) + return part1.substr(part1.lastIndexOf('/') + 1) + })() + const serieName = (set && set.serie.name.en) ?? (() => { + const part1 = path.substr(0, path.lastIndexOf(setName) - 1) + return part1.substr(part1.lastIndexOf('/') + 1) + })() + // console.log(path, id, setName) + const c = await getCard(serieName, setName, id) + if (!c.name[lang]) { + continue + } + list.push([id, c]) + } + + // Sort by id when possible + return list.sort(([a], [b]) => { + const ra = parseInt(a, 10) + const rb = parseInt(b, 10) + if (!isNaN(ra) && !isNaN(rb)) { + return ra - rb + } + return a >= b ? 1 : -1 + }) +} diff --git a/server/compiler/utils/serieUtil.ts b/server/compiler/utils/serieUtil.ts new file mode 100644 index 000000000..78002e038 --- /dev/null +++ b/server/compiler/utils/serieUtil.ts @@ -0,0 +1,66 @@ +import { DB_PATH, smartGlob } from './util' +import { setToSetSimple, getSets } from './setUtil' +import { Serie, SupportedLanguages, Set } from '../../../interfaces' +import { Serie as SerieSingle, SerieResume } from '../../../meta/definitions/api' + +export async function getSerie(name: string): Promise { + return (await import(`../../${DB_PATH}/data/${name}.js`)).default +} + +export async function isSerieAvailable(serie: Serie, lang: SupportedLanguages): Promise { + if (!serie.name[lang]) { + return false + } + const sets = await getSets(serie.name.en, lang) + return sets.length > 0 +} + +export async function getSeries(lang: SupportedLanguages): Promise> { + let series: Array = (await Promise.all((await smartGlob(`${DB_PATH}/data/*.js`)) + // Find Serie's name + .map((it) => it.substring(it.lastIndexOf('/') + 1, it.length - 3)) + // Fetch the Serie + .map((it) => getSerie(it)))) + // Filter the serie if no name's exists in the selected lang + .filter((serie) => Boolean(serie.name[lang])) + + // Filter available series + const isAvailable = await Promise.all(series.map((serie) => isSerieAvailable(serie, lang))) + series = series.filter((_, index) => isAvailable[index]) + + // Sort series by the first set release date + const tmp: Array<[Serie, Set | undefined]> = await Promise.all(series.map(async (it) => [ + it, + (await getSets(it.name.en, lang)) + .reduce((p, c) => p ? p.releaseDate < c.releaseDate ? p : c : c, undefined) as Set + ] as [Serie, Set])) + + return tmp.sort((a, b) => (a[1] ? a[1].releaseDate : '0') > (b[1] ? b[1].releaseDate : '0') ? 1 : -1).map((it) => it[0]) +} + +export async function serieToSerieSimple(serie: Serie, lang: SupportedLanguages): Promise { + const setsTmp = await getSets(serie.name.en, lang) + const sets = await Promise.all(setsTmp + .sort((a, b) => a.releaseDate > b.releaseDate ? 1 : -1) + .map((el) => setToSetSimple(el, lang))) + const logo = sets.find((set) => set.logo)?.logo + return { + id: serie.id, + logo, + name: serie.name[lang] as string + } +} + +export async function serieToSerieSingle(serie: Serie, lang: SupportedLanguages): Promise { + const setsTmp = await getSets(serie.name.en, lang) + const sets = await Promise.all(setsTmp + .sort((a, b) => a.releaseDate > b.releaseDate ? 1 : -1) + .map((el) => setToSetSimple(el, lang))) + const logo = sets.find((set) => set.logo)?.logo + return { + id: serie.id, + logo, + name: serie.name[lang] as string, + sets + } +} diff --git a/server/compiler/utils/setUtil.ts b/server/compiler/utils/setUtil.ts new file mode 100644 index 000000000..58cb5db36 --- /dev/null +++ b/server/compiler/utils/setUtil.ts @@ -0,0 +1,104 @@ +import { Set, SupportedLanguages } from '../../../interfaces' +import { DB_PATH, fetchRemoteFile, setIsLegal, smartGlob } from './util' +import { cardToCardSimple, getCards } from './cardUtil' +import { SetResume, Set as SetSingle } from '../../../meta/definitions/api' + +interface t { + [key: string]: Set +} + +const setCache: t = {} + +export function isSetAvailable(set: Set, lang: SupportedLanguages): boolean { + return lang in set.name && lang in set.serie.name +} + +/** + * Return the set + * @param name the name of the set (don't include.js/.ts) + */ +export async function getSet(name: string, serie = '*'): Promise { + if (!setCache[name]) { + try { + const [path] = await smartGlob(`${DB_PATH}/data/${serie}/${name}.js`) + setCache[name] = (await import('../../' + path)).default + } catch (error) { + console.error(error) + console.error(`Error trying to import importing (${`db/data/${serie}/${name}.js`})`) + process.exit(1) + } + } + return setCache[name] +} + +// Dont use cache as it wont necessary have them all +export async function getSets(serie = '*', lang: SupportedLanguages): Promise> { + // list sets names + const rawSets = (await smartGlob(`${DB_PATH}/data/${serie}/*.js`)).map((set) => set.substring(set.lastIndexOf('/') + 1, set.lastIndexOf('.'))) + // Fetch sets + const sets = (await Promise.all(rawSets.map((set) => getSet(set, serie)))) + // Filter sets + .filter((set) => isSetAvailable(set, lang)) + // Sort sets by release date + .sort((a, b) => a.releaseDate > b.releaseDate ? 1 : -1) + return sets +} + +export async function getSetPictures(set: Set, lang: SupportedLanguages): Promise<[string | undefined, string | undefined]> { + try { + const file = await fetchRemoteFile('https://assets.tcgdex.net/datas.json') + const logoExists = file[lang]?.[set.serie.id]?.[set.id]?.logo ? `https://assets.tcgdex.net/${lang}/${set.serie.id}/${set.id}/logo` : undefined + const symbolExists = file.univ?.[set.serie.id]?.[set.id]?.symbol ? `https://assets.tcgdex.net/univ/${set.serie.id}/${set.id}/symbol` : undefined + return [ + logoExists, + symbolExists + ] + } catch { + return [undefined, undefined] + } +} + +export async function setToSetSimple(set: Set, lang: SupportedLanguages): Promise { + const cards = await getCards(lang, set) + const pics = await getSetPictures(set, lang) + return { + cardCount: { + official: set.cardCount.official, + total: Math.max(set.cardCount.official, cards.length) + }, + id: set.id, + logo: pics[0], + name: set.name[lang] as string, + symbol: pics[1] + } +} + +export async function setToSetSingle(set: Set, lang: SupportedLanguages): Promise { + const cards = await getCards(lang, set) + const pics = await getSetPictures(set, lang) + return { + cardCount: { + firstEd: cards.reduce((count, card) => count + (card[1].variants?.firstEdition ? 1 : 0), 0), + holo: cards.reduce((count, card) => count + (card[1].variants?.holo ? 1 : 0), 0), + normal: cards.reduce((count, card) => count + (card[1].variants?.normal ? 1 : 0), 0), + official: set.cardCount.official, + reverse: cards.reduce((count, card) => count + (card[1].variants?.reverse ? 1 : 0), 0), + total: Math.max(set.cardCount.official, cards.length) + }, + cards: await Promise.all(cards.map(([id, card]) => cardToCardSimple(id, card, lang))), + id: set.id, + legal: { + expanded: setIsLegal('expanded', set), + standard: setIsLegal('standard', set) + }, + logo: pics[0], + name: set.name[lang] as string, + releaseDate: set.releaseDate, + serie: { + id: set.serie.id, + name: set.serie.name[lang] as string + }, + symbol: pics[1], + tcgOnline: set.tcgOnline + } +} diff --git a/server/compiler/utils/translationUtil.ts b/server/compiler/utils/translationUtil.ts new file mode 100644 index 000000000..95293ba81 --- /dev/null +++ b/server/compiler/utils/translationUtil.ts @@ -0,0 +1,31 @@ +import { SupportedLanguages } from '../../../interfaces' +import es from '../../../meta/translations/es.json' +import it from '../../../meta/translations/it.json' +import pt from '../../../meta/translations/pt.json' +import de from '../../../meta/translations/de.json' +import fr from '../../../meta/translations/fr.json' + +type translatable = 'types' | 'rarity' | 'stage' | 'category' | 'suffix' | 'abilityType' | 'trainerType' | 'energyType' + +const translations: Record>> = { + es, + fr, + it, + pt, + de +} + +export default function translate(item: translatable, key: string | undefined, lang: SupportedLanguages): string | undefined { + if (!key) { + return key + } + // Temporary trenslations are in english while they are being worked on + if (lang === 'en' || !Object.keys(translations).includes(lang)) { + return key + } + const res = translations[lang]?.[item]?.[key] + if (!res) { + throw new Error(`Could not find translation for ${lang}.${item}.${key}`) + } + return res +} diff --git a/server/compiler/utils/util.ts b/server/compiler/utils/util.ts new file mode 100644 index 000000000..ba0b1c736 --- /dev/null +++ b/server/compiler/utils/util.ts @@ -0,0 +1,59 @@ +import { Card, Set } from '../../../interfaces' +import glob from 'glob' +import fetch from 'node-fetch' +import * as legals from '../../../meta/legals' + +interface fileCacheInterface { + [key: string]: any +} + +export const DB_PATH = "../../" + +const fileCache: fileCacheInterface = {} + +export async function fetchRemoteFile(url: string): Promise { + if (!fileCache[url]) { + const resp = await fetch(url, { + timeout: 60 * 1000 + }) + fileCache[url] = resp.json() + } + return fileCache[url] +} + +const globCache: Record> = {} + +export async function smartGlob(query: string): Promise> { + if (!globCache[query]) { + globCache[query] = await new Promise((res) => { + glob(query, (_, matches) => res(matches)) + }) + } + return globCache[query] +} + +export function cardIsLegal(type: 'standard' | 'expanded', card: Card, localId: string): boolean { + const legal = legals[type] + if ( + legal.includes.series.includes(card.set.serie.id) || + legal.includes.sets.includes(card.set.id) || + card.regulationMark && legal.includes.regulationMark.includes(card.regulationMark) + ) { + return !( + legal.excludes.sets.includes(card.set.id) || + legal.excludes.cards.includes(`${card.set.id}-${localId}`) + ) + } + return false +} + +export function setIsLegal(type: 'standard' | 'expanded', set: Set): boolean { + const legal = legals[type] + if ( + legal.includes.series.includes(set.serie.id) || + legal.includes.sets.includes(set.id) + ) { + return !legal.excludes.sets.includes(set.id) + } + return false +} diff --git a/server/package-lock.json b/server/package-lock.json new file mode 100644 index 000000000..98dca3440 --- /dev/null +++ b/server/package-lock.json @@ -0,0 +1,2840 @@ +{ + "name": "@tcgdex/server", + "version": "1.0.0", + "lockfileVersion": 2, + "requires": true, + "packages": { + "": { + "name": "@tcgdex/server", + "version": "1.0.0", + "license": "MIT", + "dependencies": { + "@dzeio/config": "^1.1.4", + "@dzeio/object-util": "^1.4.2", + "@tcgdex/sdk": "^2.4.5", + "apicache": "^1.6.3", + "express": "^4.17.1", + "express-graphql": "^0.12.0", + "graphql": "^15.7.0", + "js2xmlparser": "^4.0.1" + }, + "devDependencies": { + "@types/apicache": "^1.6.0", + "@types/express": "^4.17.13", + "@types/glob": "^7.2.0", + "@types/node": "^16.11.6", + "@types/node-fetch": "^2.5.12", + "fs-extra": "^10.0.0", + "glob": "^7.2.0", + "node-fetch": "^2.6.6", + "ts-node": "^10.4.0", + "ts-node-dev": "^1.1.8", + "typescript": "^4.4.4" + } + }, + "node_modules/@cspotcode/source-map-consumer": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-consumer/-/source-map-consumer-0.8.0.tgz", + "integrity": "sha512-41qniHzTU8yAGbCp04ohlmSrZf8bkf/iJsl3V0dRGsQN/5GFfx+LbCSsCpp2gqrqjTVg/K6O8ycoV35JIwAzAg==", + "dev": true, + "engines": { + "node": ">= 12" + } + }, + "node_modules/@cspotcode/source-map-support": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.7.0.tgz", + "integrity": "sha512-X4xqRHqN8ACt2aHVe51OxeA2HjbcL4MqFqXkrmQszJ1NOUuUu5u6Vqx/0lZSVNku7velL5FC/s5uEAj1lsBMhA==", + "dev": true, + "dependencies": { + "@cspotcode/source-map-consumer": "0.8.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@dzeio/config": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@dzeio/config/-/config-1.1.4.tgz", + "integrity": "sha512-asVWPtUXmlIf9YNXS/zwbTvb39AT0FMkzxeO6vo5uxlv8tKoCGV1ULa4LGVuhbL+A9Heyq+vag1HcqA4jSY1gw==", + "peerDependencies": { + "@typescript-eslint/parser": "^4.28.2", + "next": "^11.1.2", + "next-pre-css": "^1.0.0" + }, + "peerDependenciesMeta": { + "@typescript-eslint/parser": { + "optional": true + }, + "next": { + "optional": true + }, + "next-pre-css": { + "optional": true + } + } + }, + "node_modules/@dzeio/object-util": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@dzeio/object-util/-/object-util-1.4.2.tgz", + "integrity": "sha512-dqBQo9aUNUi9XVAMJl3dZgA+wKAQgehdvmMX+04X29ovWCYcWuXvq+8HEzckQXMPQ/xYw+D/fV2S96QYaHzX1g==" + }, + "node_modules/@tcgdex/sdk": { + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/@tcgdex/sdk/-/sdk-2.4.5.tgz", + "integrity": "sha512-NOzMffhUdRdlrMrd6PCQ7A/bbwlkY9DiMbR8O+JWjRKg/q/E2IXKewYRCFzFvCXe/pEs+gbYj/Jhta49OzYxHg==", + "dependencies": { + "isomorphic-unfetch": "^3.1.0", + "unfetch": "^4.2.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/@tsconfig/node10": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.8.tgz", + "integrity": "sha512-6XFfSQmMgq0CFLY1MslA/CPUfhIL919M1rMsa5lP2P097N2Wd1sSX0tx1u4olM16fLNhtHZpRhedZJphNJqmZg==", + "dev": true + }, + "node_modules/@tsconfig/node12": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.9.tgz", + "integrity": "sha512-/yBMcem+fbvhSREH+s14YJi18sp7J9jpuhYByADT2rypfajMZZN4WQ6zBGgBKp53NKmqI36wFYDb3yaMPurITw==", + "dev": true + }, + "node_modules/@tsconfig/node14": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.1.tgz", + "integrity": "sha512-509r2+yARFfHHE7T6Puu2jjkoycftovhXRqW328PDXTVGKihlb1P8Z9mMZH04ebyajfRY7dedfGynlrFHJUQCg==", + "dev": true + }, + "node_modules/@tsconfig/node16": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.2.tgz", + "integrity": "sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA==", + "dev": true + }, + "node_modules/@types/apicache": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@types/apicache/-/apicache-1.6.0.tgz", + "integrity": "sha512-tFl/w1qvBrdpa/A4VasojDXgrKhaxk8UvIgenkgHmiAXl5MXxPbmveuiHhQBR6NdDAIWBSQ3UFiR9wadQVUTuQ==", + "dev": true, + "dependencies": { + "@types/redis": "*" + } + }, + "node_modules/@types/body-parser": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.1.tgz", + "integrity": "sha512-a6bTJ21vFOGIkwM0kzh9Yr89ziVxq4vYH2fQ6N8AeipEzai/cFK6aGMArIkUeIdRIgpwQa+2bXiLuUJCpSf2Cg==", + "dev": true, + "dependencies": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "node_modules/@types/connect": { + "version": "3.4.35", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz", + "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/express": { + "version": "4.17.13", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.13.tgz", + "integrity": "sha512-6bSZTPaTIACxn48l50SR+axgrqm6qXFIxrdAKaG6PaJk3+zuUr35hBlgT7vOmJcum+OEaIBLtHV/qloEAFITeA==", + "dev": true, + "dependencies": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.18", + "@types/qs": "*", + "@types/serve-static": "*" + } + }, + "node_modules/@types/express-serve-static-core": { + "version": "4.17.24", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.24.tgz", + "integrity": "sha512-3UJuW+Qxhzwjq3xhwXm2onQcFHn76frIYVbTu+kn24LFxI+dEhdfISDFovPB8VpEgW8oQCTpRuCe+0zJxB7NEA==", + "dev": true, + "dependencies": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*" + } + }, + "node_modules/@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "dependencies": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "node_modules/@types/mime": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.2.tgz", + "integrity": "sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==", + "dev": true + }, + "node_modules/@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, + "node_modules/@types/node": { + "version": "16.11.6", + "resolved": "https://registry.npmjs.org/@types/node/-/node-16.11.6.tgz", + "integrity": "sha512-ua7PgUoeQFjmWPcoo9khiPum3Pd60k4/2ZGXt18sm2Slk0W0xZTqt5Y0Ny1NyBiN1EVQ/+FaF9NcY4Qe6rwk5w==", + "dev": true + }, + "node_modules/@types/node-fetch": { + "version": "2.5.12", + "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.12.tgz", + "integrity": "sha512-MKgC4dlq4kKNa/mYrwpKfzQMB5X3ee5U6fSprkKpToBqBmX4nFZL9cW5jl6sWn+xpRJ7ypWh2yyqqr8UUCstSw==", + "dev": true, + "dependencies": { + "@types/node": "*", + "form-data": "^3.0.0" + } + }, + "node_modules/@types/qs": { + "version": "6.9.7", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz", + "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==", + "dev": true + }, + "node_modules/@types/range-parser": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz", + "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==", + "dev": true + }, + "node_modules/@types/redis": { + "version": "2.8.32", + "resolved": "https://registry.npmjs.org/@types/redis/-/redis-2.8.32.tgz", + "integrity": "sha512-7jkMKxcGq9p242exlbsVzuJb57KqHRhNl4dHoQu2Y5v9bCAbtIXXH0R3HleSQW4CTOqpHIYUW3t6tpUj4BVQ+w==", + "dev": true, + "dependencies": { + "@types/node": "*" + } + }, + "node_modules/@types/serve-static": { + "version": "1.13.10", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.10.tgz", + "integrity": "sha512-nCkHGI4w7ZgAdNkrEu0bv+4xNV/XDqW+DydknebMOQwkpDGx8G+HTlj7R7ABI8i8nKxVw0wtKPi1D+lPOkh4YQ==", + "dev": true, + "dependencies": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "node_modules/@types/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-FKjsOVbC6B7bdSB5CuzyHCkK69I=", + "dev": true + }, + "node_modules/@types/strip-json-comments": { + "version": "0.0.30", + "resolved": "https://registry.npmjs.org/@types/strip-json-comments/-/strip-json-comments-0.0.30.tgz", + "integrity": "sha512-7NQmHra/JILCd1QqpSzl8+mJRc8ZHz3uDm8YV1Ks9IhK0epEiTw8aIErbvH9PI+6XbqhyIQy3462nEsn7UVzjQ==", + "dev": true + }, + "node_modules/accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "dependencies": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "8.5.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.5.0.tgz", + "integrity": "sha512-yXbYeFy+jUuYd3/CDcg2NkIYE991XYX/bje7LmjJigUciaeO1JR4XxXgCIV1/Zc/dRuFEyw1L0pbA+qynJkW5Q==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/acorn-walk": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", + "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "dev": true, + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/apicache": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/apicache/-/apicache-1.6.3.tgz", + "integrity": "sha512-jS3VfUFpQ9BesFQZcdd1vVYg3ZsO2kGPmTJHqycIYPAQs54r74CRiyj8DuzJpwzLwIfCBYzh4dy9Jt8xYbo27w==", + "engines": { + "node": ">=8" + } + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "node_modules/balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "node_modules/binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "dependencies": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "engines": { + "node": ">= 0.8" + } + }, + "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/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "node_modules/brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "node_modules/bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/chokidar": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", + "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", + "dev": true, + "dependencies": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.2" + } + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", + "dev": true + }, + "node_modules/content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "dependencies": { + "safe-buffer": "5.1.2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "node_modules/create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/dynamic-dedupe": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/dynamic-dedupe/-/dynamic-dedupe-0.3.0.tgz", + "integrity": "sha1-BuRMIj9eTpTXjvnbI6ZRXOL5YqE=", + "dev": true, + "dependencies": { + "xtend": "^4.0.0" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "dependencies": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/express-graphql": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/express-graphql/-/express-graphql-0.12.0.tgz", + "integrity": "sha512-DwYaJQy0amdy3pgNtiTDuGGM2BLdj+YO2SgbKoLliCfuHv3VVTt7vNG/ZqK2hRYjtYHE2t2KB705EU94mE64zg==", + "dependencies": { + "accepts": "^1.3.7", + "content-type": "^1.0.4", + "http-errors": "1.8.0", + "raw-body": "^2.4.1" + }, + "engines": { + "node": ">= 10.x" + }, + "peerDependencies": { + "graphql": "^14.7.0 || ^15.3.0" + } + }, + "node_modules/express-graphql/node_modules/http-errors": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.0.tgz", + "integrity": "sha512-4I8r0C5JDhT5VkvI47QktDW75rNlGVsUf/8hzjCC/wkWI/jdTRmBb9aI7erSG82r1bjKY3F6k28WnsVxB1C73A==", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express-graphql/node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/express-graphql/node_modules/raw-body": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.1.tgz", + "integrity": "sha512-9WmIKF6mkvA0SLmA2Knm9+qj89e+j1zqgyn8aXGd7+nAduPoqgI9lO57SAZNn/Byzo5P7JhXTyg9PzaJbH73bA==", + "dependencies": { + "bytes": "3.1.0", + "http-errors": "1.7.3", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/express-graphql/node_modules/raw-body/node_modules/http-errors": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", + "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/express-graphql/node_modules/raw-body/node_modules/setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "node_modules/express-graphql/node_modules/setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" + }, + "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": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + }, + "engines": { + "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": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "node_modules/form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dev": true, + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fs-extra": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.0.0.tgz", + "integrity": "sha512-C5owb14u9eJwizKGdchcDUQeFtlSHHthBk8pbX9Vc1PFZrLombudjDnNns88aYslCyF6IY5SUw3Roz6xShcEIQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + }, + "engines": { + "node": ">=12" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "node_modules/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "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/graceful-fs": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "dev": true + }, + "node_modules/graphql": { + "version": "15.7.0", + "resolved": "https://registry.npmjs.org/graphql/-/graphql-15.7.0.tgz", + "integrity": "sha512-1jvUsS5mSzcgXLTQNQyrP7eKkBZW+HUnmx2LYSnfvkyseVpij8wwO/sFBGgxbkZ+zzFwYQxrHsOana5oMXmMxg==", + "engines": { + "node": ">= 10.x" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "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/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-core-module": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.0.tgz", + "integrity": "sha512-vd15qHsaqrRL7dtH6QNuy0ndJmRDrS9HAM1CAiSifNUFv4x1a0CCVsj18hJ1mShxIG6T2i1sO78MkP56r0nYRw==", + "dev": true, + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/isomorphic-unfetch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/isomorphic-unfetch/-/isomorphic-unfetch-3.1.0.tgz", + "integrity": "sha512-geDJjpoZ8N0kWexiwkX8F9NkTsXhetLPVbZFQ+JTW239QNOwvB0gniuR1Wc6f0AMTn7/mFGyXvHTifrCp/GH8Q==", + "dependencies": { + "node-fetch": "^2.6.1", + "unfetch": "^4.2.0" + } + }, + "node_modules/js2xmlparser": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-4.0.1.tgz", + "integrity": "sha512-KrPTolcw6RocpYjdC7pL7v62e55q7qOMHvLX1UCLc5AAS8qeJ6nukarEJAF2KL2PZxlbGueEbINqZR2bDe/gUw==", + "dependencies": { + "xmlcreate": "^2.0.3" + } + }, + "node_modules/jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "dependencies": { + "universalify": "^2.0.0" + }, + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.50.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz", + "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.33", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz", + "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==", + "dependencies": { + "mime-db": "1.50.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", + "dev": true + }, + "node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/node-fetch": { + "version": "2.6.6", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.6.tgz", + "integrity": "sha512-Z8/6vRlTUChSdIgMa51jxQ4lrw/Jy5SOW10ObaA47/RElsAN2c5Pn8bTgFGWn/ibwzXTE8qwr1Yzx28vsecXEA==", + "dependencies": { + "whatwg-url": "^5.0.0" + }, + "engines": { + "node": "4.x || >=6.0.0" + } + }, + "node_modules/normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "node_modules/path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "node_modules/picomatch": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "dev": true, + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "dependencies": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/resolve": { + "version": "1.20.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz", + "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==", + "dev": true, + "dependencies": { + "is-core-module": "^2.2.0", + "path-parse": "^1.0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "node_modules/send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "dependencies": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "engines": { + "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": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + }, + "node_modules/serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "dependencies": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.20", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.20.tgz", + "integrity": "sha512-n1lZZ8Ve4ksRqizaBQgxXDgKwttHDhyfQjA6YZZn8+AroHbsIz+JjwxQDxbp+7y5OYCI8t1Yk7etjD9CRd2hIw==", + "dev": true, + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==", + "engines": { + "node": ">=0.6" + } + }, + "node_modules/tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=" + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true, + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/ts-node": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.4.0.tgz", + "integrity": "sha512-g0FlPvvCXSIO1JDF6S232P5jPYqBkRL9qly81ZgAOSU7rwI0stphCgd2kLiCrU9DjQCrJMWEqcNSjQL02s6d8A==", + "dev": true, + "dependencies": { + "@cspotcode/source-map-support": "0.7.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-cwd": "dist/bin-cwd.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "peerDependencies": { + "@swc/core": ">=1.2.50", + "@swc/wasm": ">=1.2.50", + "@types/node": "*", + "typescript": ">=2.7" + }, + "peerDependenciesMeta": { + "@swc/core": { + "optional": true + }, + "@swc/wasm": { + "optional": true + } + } + }, + "node_modules/ts-node-dev": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/ts-node-dev/-/ts-node-dev-1.1.8.tgz", + "integrity": "sha512-Q/m3vEwzYwLZKmV6/0VlFxcZzVV/xcgOt+Tx/VjaaRHyiBcFlV0541yrT09QjzzCxlDZ34OzKjrFAynlmtflEg==", + "dev": true, + "dependencies": { + "chokidar": "^3.5.1", + "dynamic-dedupe": "^0.3.0", + "minimist": "^1.2.5", + "mkdirp": "^1.0.4", + "resolve": "^1.0.0", + "rimraf": "^2.6.1", + "source-map-support": "^0.5.12", + "tree-kill": "^1.2.2", + "ts-node": "^9.0.0", + "tsconfig": "^7.0.0" + }, + "bin": { + "ts-node-dev": "lib/bin.js", + "tsnd": "lib/bin.js" + }, + "engines": { + "node": ">=0.8.0" + }, + "peerDependencies": { + "node-notifier": "*", + "typescript": "*" + }, + "peerDependenciesMeta": { + "node-notifier": { + "optional": true + } + } + }, + "node_modules/ts-node-dev/node_modules/ts-node": { + "version": "9.1.1", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-9.1.1.tgz", + "integrity": "sha512-hPlt7ZACERQGf03M253ytLY3dHbGNGrAq9qIHWUY9XHYl1z7wYngSr3OQ5xmui8o2AaxsONxIzjafLUiWBo1Fg==", + "dev": true, + "dependencies": { + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "source-map-support": "^0.5.17", + "yn": "3.1.1" + }, + "bin": { + "ts-node": "dist/bin.js", + "ts-node-script": "dist/bin-script.js", + "ts-node-transpile-only": "dist/bin-transpile.js", + "ts-script": "dist/bin-script-deprecated.js" + }, + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "typescript": ">=2.7" + } + }, + "node_modules/tsconfig": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-7.0.0.tgz", + "integrity": "sha512-vZXmzPrL+EmC4T/4rVlT2jNVMWCi/O4DIiSj3UHg1OE5kCKbk4mfrXc6dZksLgRM/TZlKnousKH9bbTazUWRRw==", + "dev": true, + "dependencies": { + "@types/strip-bom": "^3.0.0", + "@types/strip-json-comments": "0.0.30", + "strip-bom": "^3.0.0", + "strip-json-comments": "^2.0.0" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typescript": { + "version": "4.4.4", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.4.4.tgz", + "integrity": "sha512-DqGhF5IKoBl8WNf8C1gu8q0xZSInh9j1kJJMqT3a94w1JzVaBU4EXOSMrz9yDqMT0xt3selp83fuFMQ0uzv6qA==", + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/unfetch": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/unfetch/-/unfetch-4.2.0.tgz", + "integrity": "sha512-F9p7yYCn6cIW9El1zi0HI6vqpeIvBsr3dSuRO6Xuppb1u5rXpCPmMvLSyECLhybr9isec8Ohl0hPekMVrEinDA==" + }, + "node_modules/universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true, + "engines": { + "node": ">= 10.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=", + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=" + }, + "node_modules/whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", + "dependencies": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + }, + "node_modules/xmlcreate": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/xmlcreate/-/xmlcreate-2.0.3.tgz", + "integrity": "sha512-HgS+X6zAztGa9zIK3Y3LXuJes33Lz9x+YyTxgrkIdabu2vqcGOWwdfCpf1hWLRrd553wd4QCDf6BBO6FfdsRiQ==" + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "engines": { + "node": ">=6" + } + } + }, + "dependencies": { + "@cspotcode/source-map-consumer": { + "version": "0.8.0", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-consumer/-/source-map-consumer-0.8.0.tgz", + "integrity": "sha512-41qniHzTU8yAGbCp04ohlmSrZf8bkf/iJsl3V0dRGsQN/5GFfx+LbCSsCpp2gqrqjTVg/K6O8ycoV35JIwAzAg==", + "dev": true + }, + "@cspotcode/source-map-support": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/@cspotcode/source-map-support/-/source-map-support-0.7.0.tgz", + "integrity": "sha512-X4xqRHqN8ACt2aHVe51OxeA2HjbcL4MqFqXkrmQszJ1NOUuUu5u6Vqx/0lZSVNku7velL5FC/s5uEAj1lsBMhA==", + "dev": true, + "requires": { + "@cspotcode/source-map-consumer": "0.8.0" + } + }, + "@dzeio/config": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/@dzeio/config/-/config-1.1.4.tgz", + "integrity": "sha512-asVWPtUXmlIf9YNXS/zwbTvb39AT0FMkzxeO6vo5uxlv8tKoCGV1ULa4LGVuhbL+A9Heyq+vag1HcqA4jSY1gw==", + "requires": {} + }, + "@dzeio/object-util": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/@dzeio/object-util/-/object-util-1.4.2.tgz", + "integrity": "sha512-dqBQo9aUNUi9XVAMJl3dZgA+wKAQgehdvmMX+04X29ovWCYcWuXvq+8HEzckQXMPQ/xYw+D/fV2S96QYaHzX1g==" + }, + "@tcgdex/sdk": { + "version": "2.4.5", + "resolved": "https://registry.npmjs.org/@tcgdex/sdk/-/sdk-2.4.5.tgz", + "integrity": "sha512-NOzMffhUdRdlrMrd6PCQ7A/bbwlkY9DiMbR8O+JWjRKg/q/E2IXKewYRCFzFvCXe/pEs+gbYj/Jhta49OzYxHg==", + "requires": { + "isomorphic-unfetch": "^3.1.0", + "unfetch": "^4.2.0" + } + }, + "@tsconfig/node10": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/@tsconfig/node10/-/node10-1.0.8.tgz", + "integrity": "sha512-6XFfSQmMgq0CFLY1MslA/CPUfhIL919M1rMsa5lP2P097N2Wd1sSX0tx1u4olM16fLNhtHZpRhedZJphNJqmZg==", + "dev": true + }, + "@tsconfig/node12": { + "version": "1.0.9", + "resolved": "https://registry.npmjs.org/@tsconfig/node12/-/node12-1.0.9.tgz", + "integrity": "sha512-/yBMcem+fbvhSREH+s14YJi18sp7J9jpuhYByADT2rypfajMZZN4WQ6zBGgBKp53NKmqI36wFYDb3yaMPurITw==", + "dev": true + }, + "@tsconfig/node14": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/@tsconfig/node14/-/node14-1.0.1.tgz", + "integrity": "sha512-509r2+yARFfHHE7T6Puu2jjkoycftovhXRqW328PDXTVGKihlb1P8Z9mMZH04ebyajfRY7dedfGynlrFHJUQCg==", + "dev": true + }, + "@tsconfig/node16": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@tsconfig/node16/-/node16-1.0.2.tgz", + "integrity": "sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA==", + "dev": true + }, + "@types/apicache": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/@types/apicache/-/apicache-1.6.0.tgz", + "integrity": "sha512-tFl/w1qvBrdpa/A4VasojDXgrKhaxk8UvIgenkgHmiAXl5MXxPbmveuiHhQBR6NdDAIWBSQ3UFiR9wadQVUTuQ==", + "dev": true, + "requires": { + "@types/redis": "*" + } + }, + "@types/body-parser": { + "version": "1.19.1", + "resolved": "https://registry.npmjs.org/@types/body-parser/-/body-parser-1.19.1.tgz", + "integrity": "sha512-a6bTJ21vFOGIkwM0kzh9Yr89ziVxq4vYH2fQ6N8AeipEzai/cFK6aGMArIkUeIdRIgpwQa+2bXiLuUJCpSf2Cg==", + "dev": true, + "requires": { + "@types/connect": "*", + "@types/node": "*" + } + }, + "@types/connect": { + "version": "3.4.35", + "resolved": "https://registry.npmjs.org/@types/connect/-/connect-3.4.35.tgz", + "integrity": "sha512-cdeYyv4KWoEgpBISTxWvqYsVy444DOqehiF3fM3ne10AmJ62RSyNkUnxMJXHQWRQQX2eR94m5y1IZyDwBjV9FQ==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/express": { + "version": "4.17.13", + "resolved": "https://registry.npmjs.org/@types/express/-/express-4.17.13.tgz", + "integrity": "sha512-6bSZTPaTIACxn48l50SR+axgrqm6qXFIxrdAKaG6PaJk3+zuUr35hBlgT7vOmJcum+OEaIBLtHV/qloEAFITeA==", + "dev": true, + "requires": { + "@types/body-parser": "*", + "@types/express-serve-static-core": "^4.17.18", + "@types/qs": "*", + "@types/serve-static": "*" + } + }, + "@types/express-serve-static-core": { + "version": "4.17.24", + "resolved": "https://registry.npmjs.org/@types/express-serve-static-core/-/express-serve-static-core-4.17.24.tgz", + "integrity": "sha512-3UJuW+Qxhzwjq3xhwXm2onQcFHn76frIYVbTu+kn24LFxI+dEhdfISDFovPB8VpEgW8oQCTpRuCe+0zJxB7NEA==", + "dev": true, + "requires": { + "@types/node": "*", + "@types/qs": "*", + "@types/range-parser": "*" + } + }, + "@types/glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-ZUxbzKl0IfJILTS6t7ip5fQQM/J3TJYubDm3nMbgubNNYS62eXeUpoLUC8/7fJNiFYHTrGPQn7hspDUzIHX3UA==", + "dev": true, + "requires": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "@types/mime": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@types/mime/-/mime-1.3.2.tgz", + "integrity": "sha512-YATxVxgRqNH6nHEIsvg6k2Boc1JHI9ZbH5iWFFv/MTkchz3b1ieGDa5T0a9RznNdI0KhVbdbWSN+KWWrQZRxTw==", + "dev": true + }, + "@types/minimatch": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.5.tgz", + "integrity": "sha512-Klz949h02Gz2uZCMGwDUSDS1YBlTdDDgbWHi+81l29tQALUtvz4rAYi5uoVhE5Lagoq6DeqAUlbrHvW/mXDgdQ==", + "dev": true + }, + "@types/node": { + "version": "16.11.6", + "resolved": "https://registry.npmjs.org/@types/node/-/node-16.11.6.tgz", + "integrity": "sha512-ua7PgUoeQFjmWPcoo9khiPum3Pd60k4/2ZGXt18sm2Slk0W0xZTqt5Y0Ny1NyBiN1EVQ/+FaF9NcY4Qe6rwk5w==", + "dev": true + }, + "@types/node-fetch": { + "version": "2.5.12", + "resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.5.12.tgz", + "integrity": "sha512-MKgC4dlq4kKNa/mYrwpKfzQMB5X3ee5U6fSprkKpToBqBmX4nFZL9cW5jl6sWn+xpRJ7ypWh2yyqqr8UUCstSw==", + "dev": true, + "requires": { + "@types/node": "*", + "form-data": "^3.0.0" + } + }, + "@types/qs": { + "version": "6.9.7", + "resolved": "https://registry.npmjs.org/@types/qs/-/qs-6.9.7.tgz", + "integrity": "sha512-FGa1F62FT09qcrueBA6qYTrJPVDzah9a+493+o2PCXsesWHIn27G98TsSMs3WPNbZIEj4+VJf6saSFpvD+3Zsw==", + "dev": true + }, + "@types/range-parser": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/@types/range-parser/-/range-parser-1.2.4.tgz", + "integrity": "sha512-EEhsLsD6UsDM1yFhAvy0Cjr6VwmpMWqFBCb9w07wVugF7w9nfajxLuVmngTIpgS6svCnm6Vaw+MZhoDCKnOfsw==", + "dev": true + }, + "@types/redis": { + "version": "2.8.32", + "resolved": "https://registry.npmjs.org/@types/redis/-/redis-2.8.32.tgz", + "integrity": "sha512-7jkMKxcGq9p242exlbsVzuJb57KqHRhNl4dHoQu2Y5v9bCAbtIXXH0R3HleSQW4CTOqpHIYUW3t6tpUj4BVQ+w==", + "dev": true, + "requires": { + "@types/node": "*" + } + }, + "@types/serve-static": { + "version": "1.13.10", + "resolved": "https://registry.npmjs.org/@types/serve-static/-/serve-static-1.13.10.tgz", + "integrity": "sha512-nCkHGI4w7ZgAdNkrEu0bv+4xNV/XDqW+DydknebMOQwkpDGx8G+HTlj7R7ABI8i8nKxVw0wtKPi1D+lPOkh4YQ==", + "dev": true, + "requires": { + "@types/mime": "^1", + "@types/node": "*" + } + }, + "@types/strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/@types/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-FKjsOVbC6B7bdSB5CuzyHCkK69I=", + "dev": true + }, + "@types/strip-json-comments": { + "version": "0.0.30", + "resolved": "https://registry.npmjs.org/@types/strip-json-comments/-/strip-json-comments-0.0.30.tgz", + "integrity": "sha512-7NQmHra/JILCd1QqpSzl8+mJRc8ZHz3uDm8YV1Ks9IhK0epEiTw8aIErbvH9PI+6XbqhyIQy3462nEsn7UVzjQ==", + "dev": true + }, + "accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "requires": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + } + }, + "acorn": { + "version": "8.5.0", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.5.0.tgz", + "integrity": "sha512-yXbYeFy+jUuYd3/CDcg2NkIYE991XYX/bje7LmjJigUciaeO1JR4XxXgCIV1/Zc/dRuFEyw1L0pbA+qynJkW5Q==", + "dev": true + }, + "acorn-walk": { + "version": "8.2.0", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.2.0.tgz", + "integrity": "sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA==", + "dev": true + }, + "anymatch": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.2.tgz", + "integrity": "sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg==", + "dev": true, + "requires": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + } + }, + "apicache": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/apicache/-/apicache-1.6.3.tgz", + "integrity": "sha512-jS3VfUFpQ9BesFQZcdd1vVYg3ZsO2kGPmTJHqycIYPAQs54r74CRiyj8DuzJpwzLwIfCBYzh4dy9Jt8xYbo27w==" + }, + "arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, + "asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "balanced-match": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", + "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", + "dev": true + }, + "binary-extensions": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.2.0.tgz", + "integrity": "sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA==", + "dev": true + }, + "body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "requires": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "brace-expansion": { + "version": "1.1.11", + "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", + "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", + "dev": true, + "requires": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dev": true, + "requires": { + "fill-range": "^7.0.1" + } + }, + "buffer-from": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", + "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", + "dev": true + }, + "bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" + }, + "chokidar": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.5.2.tgz", + "integrity": "sha512-ekGhOnNVPgT77r4K/U3GDhu+FQ2S8TnK/s2KbIGXi0SZWuwkZ2QNyfWdZW+TVfn84DpEP7rLeCt2UI6bJ8GwbQ==", + "dev": true, + "requires": { + "anymatch": "~3.1.2", + "braces": "~3.0.2", + "fsevents": "~2.3.2", + "glob-parent": "~5.1.2", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.6.0" + } + }, + "combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "requires": { + "delayed-stream": "~1.0.0" + } + }, + "concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", + "dev": true + }, + "content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" + }, + "cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "create-require": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz", + "integrity": "sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==", + "dev": true + }, + "delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true + }, + "dynamic-dedupe": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/dynamic-dedupe/-/dynamic-dedupe-0.3.0.tgz", + "integrity": "sha1-BuRMIj9eTpTXjvnbI6ZRXOL5YqE=", + "dev": true, + "requires": { + "xtend": "^4.0.0" + } + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "requires": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "express-graphql": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/express-graphql/-/express-graphql-0.12.0.tgz", + "integrity": "sha512-DwYaJQy0amdy3pgNtiTDuGGM2BLdj+YO2SgbKoLliCfuHv3VVTt7vNG/ZqK2hRYjtYHE2t2KB705EU94mE64zg==", + "requires": { + "accepts": "^1.3.7", + "content-type": "^1.0.4", + "http-errors": "1.8.0", + "raw-body": "^2.4.1" + }, + "dependencies": { + "http-errors": { + "version": "1.8.0", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.8.0.tgz", + "integrity": "sha512-4I8r0C5JDhT5VkvI47QktDW75rNlGVsUf/8hzjCC/wkWI/jdTRmBb9aI7erSG82r1bjKY3F6k28WnsVxB1C73A==", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.2.0", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "raw-body": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.1.tgz", + "integrity": "sha512-9WmIKF6mkvA0SLmA2Knm9+qj89e+j1zqgyn8aXGd7+nAduPoqgI9lO57SAZNn/Byzo5P7JhXTyg9PzaJbH73bA==", + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.3", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "dependencies": { + "http-errors": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", + "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.4", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + } + } + }, + "setprototypeof": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz", + "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==" + } + } + }, + "fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dev": true, + "requires": { + "to-regex-range": "^5.0.1" + } + }, + "finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "form-data": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-3.0.1.tgz", + "integrity": "sha512-RHkBKtLWUVwd7SqRIvCZMEvAMoGUp0XU+seQiZejj0COz3RI3hWP4sCv3gZWWLjJTd7rGwcsF5eKZGii0r/hbg==", + "dev": true, + "requires": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + } + }, + "forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==" + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "fs-extra": { + "version": "10.0.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-10.0.0.tgz", + "integrity": "sha512-C5owb14u9eJwizKGdchcDUQeFtlSHHthBk8pbX9Vc1PFZrLombudjDnNns88aYslCyF6IY5SUw3Roz6xShcEIQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.2.0", + "jsonfile": "^6.0.1", + "universalify": "^2.0.0" + } + }, + "fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", + "dev": true + }, + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + }, + "function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "dev": true + }, + "glob": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.2.0.tgz", + "integrity": "sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==", + "dev": true, + "requires": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + } + }, + "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, + "requires": { + "is-glob": "^4.0.1" + } + }, + "graceful-fs": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.8.tgz", + "integrity": "sha512-qkIilPUYcNhJpd33n0GBXTB1MMPp14TxEsEs0pTrsSVucApsYzW5V+Q8Qxhik6KU3evy+qkAAowTByymK0avdg==", + "dev": true + }, + "graphql": { + "version": "15.7.0", + "resolved": "https://registry.npmjs.org/graphql/-/graphql-15.7.0.tgz", + "integrity": "sha512-1jvUsS5mSzcgXLTQNQyrP7eKkBZW+HUnmx2LYSnfvkyseVpij8wwO/sFBGgxbkZ+zzFwYQxrHsOana5oMXmMxg==" + }, + "has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dev": true, + "requires": { + "function-bind": "^1.1.1" + } + }, + "http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dev": true, + "requires": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" + }, + "is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dev": true, + "requires": { + "binary-extensions": "^2.0.0" + } + }, + "is-core-module": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.8.0.tgz", + "integrity": "sha512-vd15qHsaqrRL7dtH6QNuy0ndJmRDrS9HAM1CAiSifNUFv4x1a0CCVsj18hJ1mShxIG6T2i1sO78MkP56r0nYRw==", + "dev": true, + "requires": { + "has": "^1.0.3" + } + }, + "is-extglob": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", + "dev": true + }, + "is-glob": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz", + "integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==", + "dev": true, + "requires": { + "is-extglob": "^2.1.1" + } + }, + "is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "dev": true + }, + "isomorphic-unfetch": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/isomorphic-unfetch/-/isomorphic-unfetch-3.1.0.tgz", + "integrity": "sha512-geDJjpoZ8N0kWexiwkX8F9NkTsXhetLPVbZFQ+JTW239QNOwvB0gniuR1Wc6f0AMTn7/mFGyXvHTifrCp/GH8Q==", + "requires": { + "node-fetch": "^2.6.1", + "unfetch": "^4.2.0" + } + }, + "js2xmlparser": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/js2xmlparser/-/js2xmlparser-4.0.1.tgz", + "integrity": "sha512-KrPTolcw6RocpYjdC7pL7v62e55q7qOMHvLX1UCLc5AAS8qeJ6nukarEJAF2KL2PZxlbGueEbINqZR2bDe/gUw==", + "requires": { + "xmlcreate": "^2.0.3" + } + }, + "jsonfile": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-6.1.0.tgz", + "integrity": "sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.6", + "universalify": "^2.0.0" + } + }, + "make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" + }, + "mime-db": { + "version": "1.50.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.50.0.tgz", + "integrity": "sha512-9tMZCDlYHqeERXEHO9f/hKfNXhre5dK2eE/krIvUjZbS2KPcqGDfNShIWS1uW9XOTKQKqK6qbeOci18rbfW77A==" + }, + "mime-types": { + "version": "2.1.33", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.33.tgz", + "integrity": "sha512-plLElXp7pRDd0bNZHw+nMd52vRYjLwQjygaNg7ddJ2uJtTlmnTCjWuPKxVu6//AdaRuME84SvLW91sIkBqGT0g==", + "requires": { + "mime-db": "1.50.0" + } + }, + "minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dev": true, + "requires": { + "brace-expansion": "^1.1.7" + } + }, + "minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", + "dev": true + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" + }, + "node-fetch": { + "version": "2.6.6", + "resolved": "https://registry.npmjs.org/node-fetch/-/node-fetch-2.6.6.tgz", + "integrity": "sha512-Z8/6vRlTUChSdIgMa51jxQ4lrw/Jy5SOW10ObaA47/RElsAN2c5Pn8bTgFGWn/ibwzXTE8qwr1Yzx28vsecXEA==", + "requires": { + "whatwg-url": "^5.0.0" + } + }, + "normalize-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", + "dev": true + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "requires": { + "ee-first": "1.1.1" + } + }, + "once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dev": true, + "requires": { + "wrappy": "1" + } + }, + "parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" + }, + "path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "dev": true + }, + "path-parse": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", + "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", + "dev": true + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "picomatch": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "dev": true + }, + "proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "requires": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + } + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" + }, + "range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" + }, + "raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "readdirp": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", + "integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==", + "dev": true, + "requires": { + "picomatch": "^2.2.1" + } + }, + "resolve": { + "version": "1.20.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.20.0.tgz", + "integrity": "sha512-wENBPt4ySzg4ybFQW2TT1zMQucPK95HSh/nq2CFTZVOGut2+pQvSsgtda4d26YrYcr067wjbmzOG8byDPBX63A==", + "dev": true, + "requires": { + "is-core-module": "^2.2.0", + "path-parse": "^1.0.6" + } + }, + "rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "requires": { + "glob": "^7.1.3" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "dependencies": { + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + }, + "dependencies": { + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + } + } + }, + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + } + } + }, + "serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + } + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + }, + "source-map-support": { + "version": "0.5.20", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.20.tgz", + "integrity": "sha512-n1lZZ8Ve4ksRqizaBQgxXDgKwttHDhyfQjA6YZZn8+AroHbsIz+JjwxQDxbp+7y5OYCI8t1Yk7etjD9CRd2hIw==", + "dev": true, + "requires": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" + }, + "strip-bom": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", + "integrity": "sha1-IzTBjpx1n3vdVv3vfprj1YjmjtM=", + "dev": true + }, + "strip-json-comments": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz", + "integrity": "sha1-PFMZQukIwml8DsNEhYwobHygpgo=", + "dev": true + }, + "to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dev": true, + "requires": { + "is-number": "^7.0.0" + } + }, + "toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" + }, + "tr46": { + "version": "0.0.3", + "resolved": "https://registry.npmjs.org/tr46/-/tr46-0.0.3.tgz", + "integrity": "sha1-gYT9NH2snNwYWZLzpmIuFLnZq2o=" + }, + "tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true + }, + "ts-node": { + "version": "10.4.0", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.4.0.tgz", + "integrity": "sha512-g0FlPvvCXSIO1JDF6S232P5jPYqBkRL9qly81ZgAOSU7rwI0stphCgd2kLiCrU9DjQCrJMWEqcNSjQL02s6d8A==", + "dev": true, + "requires": { + "@cspotcode/source-map-support": "0.7.0", + "@tsconfig/node10": "^1.0.7", + "@tsconfig/node12": "^1.0.7", + "@tsconfig/node14": "^1.0.0", + "@tsconfig/node16": "^1.0.2", + "acorn": "^8.4.1", + "acorn-walk": "^8.1.1", + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "yn": "3.1.1" + } + }, + "ts-node-dev": { + "version": "1.1.8", + "resolved": "https://registry.npmjs.org/ts-node-dev/-/ts-node-dev-1.1.8.tgz", + "integrity": "sha512-Q/m3vEwzYwLZKmV6/0VlFxcZzVV/xcgOt+Tx/VjaaRHyiBcFlV0541yrT09QjzzCxlDZ34OzKjrFAynlmtflEg==", + "dev": true, + "requires": { + "chokidar": "^3.5.1", + "dynamic-dedupe": "^0.3.0", + "minimist": "^1.2.5", + "mkdirp": "^1.0.4", + "resolve": "^1.0.0", + "rimraf": "^2.6.1", + "source-map-support": "^0.5.12", + "tree-kill": "^1.2.2", + "ts-node": "^9.0.0", + "tsconfig": "^7.0.0" + }, + "dependencies": { + "ts-node": { + "version": "9.1.1", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-9.1.1.tgz", + "integrity": "sha512-hPlt7ZACERQGf03M253ytLY3dHbGNGrAq9qIHWUY9XHYl1z7wYngSr3OQ5xmui8o2AaxsONxIzjafLUiWBo1Fg==", + "dev": true, + "requires": { + "arg": "^4.1.0", + "create-require": "^1.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "source-map-support": "^0.5.17", + "yn": "3.1.1" + } + } + } + }, + "tsconfig": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/tsconfig/-/tsconfig-7.0.0.tgz", + "integrity": "sha512-vZXmzPrL+EmC4T/4rVlT2jNVMWCi/O4DIiSj3UHg1OE5kCKbk4mfrXc6dZksLgRM/TZlKnousKH9bbTazUWRRw==", + "dev": true, + "requires": { + "@types/strip-bom": "^3.0.0", + "@types/strip-json-comments": "0.0.30", + "strip-bom": "^3.0.0", + "strip-json-comments": "^2.0.0" + } + }, + "type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + } + }, + "typescript": { + "version": "4.4.4", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.4.4.tgz", + "integrity": "sha512-DqGhF5IKoBl8WNf8C1gu8q0xZSInh9j1kJJMqT3a94w1JzVaBU4EXOSMrz9yDqMT0xt3selp83fuFMQ0uzv6qA==", + "dev": true + }, + "unfetch": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/unfetch/-/unfetch-4.2.0.tgz", + "integrity": "sha512-F9p7yYCn6cIW9El1zi0HI6vqpeIvBsr3dSuRO6Xuppb1u5rXpCPmMvLSyECLhybr9isec8Ohl0hPekMVrEinDA==" + }, + "universalify": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", + "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "dev": true + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" + }, + "webidl-conversions": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-3.0.1.tgz", + "integrity": "sha1-JFNCdeKnvGvnvIZhHMFq4KVlSHE=" + }, + "whatwg-url": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-5.0.0.tgz", + "integrity": "sha1-lmRU6HZUYuN2RNNib2dCzotwll0=", + "requires": { + "tr46": "~0.0.3", + "webidl-conversions": "^3.0.0" + } + }, + "wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", + "dev": true + }, + "xmlcreate": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/xmlcreate/-/xmlcreate-2.0.3.tgz", + "integrity": "sha512-HgS+X6zAztGa9zIK3Y3LXuJes33Lz9x+YyTxgrkIdabu2vqcGOWwdfCpf1hWLRrd553wd4QCDf6BBO6FfdsRiQ==" + }, + "xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true + }, + "yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true + } + } +} diff --git a/server/package.json b/server/package.json new file mode 100644 index 000000000..343e7b2ef --- /dev/null +++ b/server/package.json @@ -0,0 +1,38 @@ +{ + "name": "@tcgdex/server", + "private": true, + "version": "1.0.0", + "description": "", + "main": "index.js", + "scripts": { + "compile": "ts-node compiler/index.ts", + "dev": "ts-node-dev -T src/index.ts", + "build": "tsc --project tsconfig.json", + "start": "node dist/index.js" + }, + "author": "", + "license": "MIT", + "dependencies": { + "@dzeio/config": "^1.1.4", + "@dzeio/object-util": "^1.4.2", + "@tcgdex/sdk": "^2.4.5", + "apicache": "^1.6.3", + "express": "^4.17.1", + "express-graphql": "^0.12.0", + "graphql": "^15.7.0", + "js2xmlparser": "^4.0.1" + }, + "devDependencies": { + "@types/apicache": "^1.6.0", + "@types/express": "^4.17.13", + "@types/glob": "^7.2.0", + "@types/node": "^16.11.6", + "@types/node-fetch": "^2.5.12", + "fs-extra": "^10.0.0", + "glob": "^7.2.0", + "node-fetch": "^2.6.6", + "ts-node": "^10.4.0", + "ts-node-dev": "^1.1.8", + "typescript": "^4.4.4" + } +} diff --git a/server/public/favicon.ico b/server/public/favicon.ico new file mode 100644 index 0000000000000000000000000000000000000000..7f1d2cd6f536c2107f42117f44915367f3cf262d GIT binary patch literal 15086 zcmeI3e{5D)8OJX{LzbBgVocU#86=FzpNNT?n5`R>fr&HD6&;MW^asBz#Tn)_aoM_= zV5Z?mvW;jY!~eEnlt|Pi5SFsx*l1Z9P}V}r-gaeFilb0ETO9B4`JDGWm($C=?~nE^ zH0qmtdG0;Gp6~NL&pGG5w=Y33C72q_n-_?k6}&np2)-5s!K_*CetHm`k+SJxsbBlK zAlM`g=1Lps608(ZZa?>A%>V!W-%wz~iQzKA%Ys+3fcmn@sEFfk#OA`w*zw3%2Be`T2~Ot&Q{5FE*D}v>CU*Y+S}t*Kqt_GlUm;`ktUA0o2iE z_E>yObf4dR@X77Okqxw&XdaV!eAW&cqI z?Hxy>pqzTzO_n)EUpT?_*s)_*g_}u2XJ@BT(4KzuzZH6u3!>-4@W{ruySv-;_xE1| z0|NubhbwEAvK2RTKF~}!Z%)!%Is06*T@wenRmfZrZ?-)ssSY3biqo9jh|@V6_ekBJ z^NB(BcFx}e0`~fxSl9f?duy6f7D@S11n!z!Sz|y~;mp~U>kPK_>)#(r*B$6F;I-Dy z*tJQ%M06b2J@cEi9$0t6o#(;4_^*AZ-*VslPS;K0jt(vdS0wFAZVq^kE04iODeERa zU;F*XPu;-3h&dBCU(NZfg}4}nJ29AC+K+Rs#&I51#`zG(FISmxZ_K5Awwxav9JD-W z!#<^LwY-3}GHMQ7?vpc(S-{%pSN=(PpG|wtg{@n+n)>?s5ZqH<0C&3%I3rbW;YH35 z3UaR9Y`H_;C-Hcr=}pVIyu92*QB_rGcJ11g*f(xW_9BmeIl30YKW@)N`)Dq3kG^QW z^!w!ou8ob22HmmT1L|oP?H|Y8l{G7W_Eh1{+RXM{`*o=w)IA?PqriXM`3qMxSDk6Y zTmf{4J9V^+>mOH!FXf+)bun&%f9>n1L!YXjYUt;-UQ>tm=&juSbpmf4NBiVs^Wp8c z`Qed^8*6W;7O`M}ME{eBRxfk+O)sSqAxZM$mG@J6^xKJa~!9A}@&g%*9rNScFO zYhbK={eHgo!kKlcdrv|37R|w^ych25IR)|7_Bscg2mV@~$T?uo@y}S{GMhCicH>w; zUr_tU?k8mIzv_855$k~QbS;&&Y~vNPj=AL-%15^OrVf5+tKdFcpJMX$$CCY)?=_s+yzlkFc|Ixs#uwqw zyP~ACuflIT8Lx#q`lGe4J=Z+rO?v&&w9x#YjPr0SIT`y20?!7|81B4;& z2-JG*ZMpn2Ff-G>ajNSdSUVg9rHnTa1T%v`?t(yeN4X1vY1b{)@-wHv&;ZN*6jv`c zCAWN1_Fq>n*R~zj4t)JyH(+sLKYFj@+V(Q8`fa3@__ZJHB-WUe<*WP+9c?^msifYx zc*FRe9#bIJ(#Xm96t1zmt^%3K}Gv8W$n6hBWp zf8o3NuC6X4fB!IjeSIP1J#6d6ix(T|Qw$Ok(_HbpUogrqJ}+Imln8hT=EZTyk%Klb zYv1gB;g^THUUWK#zKJ)RU%xJi!-lqzdu0uNCix%x+_mn=#cK@*Qs&i`>oF65A-w!GcK7QNiwTqqeE8LshTA1V3qbJIg`!MGwlT^V|B@ zb!EeUSFN#{3;uHjyjXuLH(mE1HzsAoici3GTsz;tyys{{Z~Vor{{_0JKY>G|53D|zs`ZLL39oZ^aFfojI#VeZSImOpJi_ zOxbw-YJc(1bCF>!IJ+MoT0ha&#q2+uEiYz#6U{$qeBK&lK2uH8GxAnKZ%vb21vAZO z!5#9-WLiqzVu+n0uNXkkI1XJa;#x%yTW7#of*aL!(GLHCSG+i)D z5Wc67|7%vrNAh8Y{LSy%8%~VyTQ9N0;aR5RP$sAnJnj(wueF8~BX?2OaANr989Q>9 z_ERh6x2X^O$o<_~r#5x+{VyuCEFUY+D)R*|$@i*O`TgT5aPHi>?ecv1b-(}mV?%OY z;X4EP9Q&}hhW4GEr|q-mg$ozLXCUd{L%oh6oQZ#rU|)U9<$<5P`|-%(w)SiE_50~F z6jsJa!9)7owxMnK8?p`-;(y)SeeJKbcFFI=GJa{xHP2vJe~inQFaJUM-)()j4Ih$m z4&i&x!FSD(mS)=?yJ^!V`@0pamNVAr&Qlkp?>&rR=Sn!>|3~$|GVAJ|GHUtF3vuk& zu_G*}4oKg9RyX*If8wuv_^7EW?K7($yku&YoVM-gOMC7q&(!seWVFw4Km+_&J=tsf z>sec~q;G^Vv0COV@$bQ}0{?ZpKE6!bx`o4u&uH>@PtJ~STK`sm<{$&gY91PRtZv~* zTg0CmFP)-Z=Re6-+_<3zh-gzOA79DeKNUZ rb240uNqsW!HhC{9koTu)d8gVWJB;i}L8Ab>(ph#M>S=e4_JaQah%8?t literal 0 HcmV?d00001 diff --git a/server/public/v2/.gitkeep b/server/public/v2/.gitkeep new file mode 100644 index 000000000..e69de29bb diff --git a/server/src/V2/Components/Card.ts b/server/src/V2/Components/Card.ts new file mode 100644 index 000000000..19b918149 --- /dev/null +++ b/server/src/V2/Components/Card.ts @@ -0,0 +1,112 @@ +import { objectLoop } from '@dzeio/object-util' +import { Card as SDKCard, CardResume, SupportedLanguages } from '@tcgdex/sdk' +import Set from './Set' +import { Pagination } from '../../interfaces' + +type LocalCard = Omit & {set: () => Set} + +interface variants { + normal?: boolean; + reverse?: boolean; + holo?: boolean; + firstEdition?: boolean; +} + +export default class Card implements LocalCard { + illustrator?: string | undefined + rarity!: string + category!: string + variants?: variants | undefined + dexId?: number[] | undefined + hp?: number | undefined + types?: string[] | undefined + evolveFrom?: string | undefined + weight?: string | undefined + description?: string | undefined + level?: string | number | undefined + stage?: string | undefined + suffix?: string | undefined + item?: { name: string; effect: string } | undefined + abilities?: { type: string; name: string; effect: string }[] | undefined + attacks?: { cost?: string[] | undefined; name: string; effect?: string | undefined; damage?: string | number | undefined }[] | undefined + weaknesses?: { type: string; value?: string | undefined }[] | undefined + resistances?: { type: string; value?: string | undefined }[] | undefined + retreat?: number | undefined + effect?: string | undefined + trainerType?: string | undefined + energyType?: string | undefined + regulationMark?: string | undefined + legal!: { standard: boolean; expanded: boolean } + id!: string + localId!: string + name!: string + image?: string | undefined + + public constructor( + private lang: SupportedLanguages, + private card: SDKCard + ) { + objectLoop(card, (it, key) => { + if (key === 'set') { + return + } + this[key as 'id'] = it + }) + } + + + public set(): Set { + return Set.findOne(this.lang, {id: this.card.set.id}) as Set + } + + public static find(lang: SupportedLanguages, params: Partial> = {}, pagination?: Pagination) { + let list : Array = (require(`../../../generated/${lang}/cards.json`) as Array) + .filter((c) => objectLoop(params, (it, key) => { + if (typeof it === "string") { + return c[key as 'localId'].toLowerCase().includes(it.toLowerCase()) + } + return c[key as 'localId'].includes(it) + })) + if (pagination) { + list = list + .splice(pagination.count * pagination.page - 1, pagination.count) + } + return list.map((it) => new Card(lang, it)) + } + + public static raw(lang: SupportedLanguages): Array { + return require(`../../generated/${lang}/cards.json`) + } + + public static findOne(lang: SupportedLanguages, params: Partial> = {}) { + const res = (require(`../../generated/${lang}/cards.json`) as Array).find((c) => { + return objectLoop(params, (it, key) => { + if (key === 'set') { + return c['set'].id.includes(it) || c['set'].name.includes(it) + } + if (typeof it === "string") { + return c[key as 'localId'].toLowerCase().includes(it.toLowerCase()) + } + return c[key as 'localId'].includes(it) + }) + }) + if (!res) { + return undefined + } + return new Card(lang, res) + } + + public resume(): CardResume { + return { + id: this.id, + localId: this.localId, + name: this.name, + image: this.image + } + } + + public full(): SDKCard { + return this.card + } + +} diff --git a/server/src/V2/Components/Serie.ts b/server/src/V2/Components/Serie.ts new file mode 100644 index 000000000..571ddc93b --- /dev/null +++ b/server/src/V2/Components/Serie.ts @@ -0,0 +1,66 @@ +import { objectLoop } from '@dzeio/object-util' +import { Serie as SDKSerie, SerieResume, SupportedLanguages } from '@tcgdex/sdk' +import Set from './Set' +import { Pagination } from '../../interfaces' + +type LocalSerie = Omit & {sets: () => Array} + +export default class Serie implements LocalSerie { + + id!: string + name!: string + logo?: string | undefined + + public constructor( + private lang: SupportedLanguages, + private serie: SDKSerie + ) { + objectLoop(serie, (it, key) => { + if (key === 'sets') { + return + } + this[key as 'id'] = it + }) + } + + public sets(): Array { + return this.serie.sets.map((s) => Set.findOne(this.lang, {id: s.id}) as Set) + } + + public static find(lang: SupportedLanguages, params: Partial> = {}, pagination?: Pagination) { + let list = (require(`../../generated/${lang}/series.json`) as Array) + .filter((c) => objectLoop(params, (it, key) => { + return c[key as 'id'].includes(it) + })) + if (pagination) { + list = list + .splice(pagination.count * pagination.page - 1, pagination.count) + } + return list.map((it) => new Serie(lang, it)) + } + + public static findOne(lang: SupportedLanguages, params: Partial> = {}): Serie | undefined { + const res = (require(`../../../generated/${lang}/series.json`) as Array) + .find((c) => { + return objectLoop(params, (it, key) => { + return c[key as 'id'].includes(it) + }) + }) + if (!res) { + return undefined + } + return new Serie(lang, res) + } + + public resume(): SerieResume { + return { + id: this.id, + name: this.name, + logo: this.logo + } + } + + public full(): SDKSerie { + return this.serie + } +} diff --git a/server/src/V2/Components/Set.ts b/server/src/V2/Components/Set.ts new file mode 100644 index 000000000..5d68b387a --- /dev/null +++ b/server/src/V2/Components/Set.ts @@ -0,0 +1,89 @@ +import { objectLoop } from '@dzeio/object-util' +import { Set as SDKSet, SetResume, SupportedLanguages } from '@tcgdex/sdk' +import Card from './Card' +import { Pagination } from '../../interfaces' +import Serie from './Serie' + +interface variants { + normal?: boolean; + reverse?: boolean; + holo?: boolean; + firstEdition?: boolean; +} + +type LocalSet = {serie: () => Serie, cards: () => Array} & Omit + +export default class Set implements LocalSet { + + public constructor( + private lang: SupportedLanguages, + private set: SDKSet + ) { + objectLoop(set, (it, key) => { + if (key === 'serie' || key === 'cards') { + return + } + this[key as 'id'] = it + }) + } + + tcgOnline?: string | undefined + variants?: variants | undefined + releaseDate!: string + legal!: { standard: boolean; expanded: boolean } + cardCount!: { total: number; official: number; normal: number; reverse: number; holo: number; firstEd?: number | undefined } + id!: string + name!: string + logo?: string | undefined + symbol?: string | undefined + + public serie(): Serie { + return Serie.findOne(this.lang, {id: this.set.serie.id}) as Serie + } + + public cards(): Array { + return this.set.cards.map((s) => Card.findOne(this.lang, {id: s.id}) as Card) + } + + public static find(lang: SupportedLanguages, params: Partial> = {}, pagination?: Pagination) { + let list = (require(`../../../generated/${lang}/sets.json`) as Array) + .filter((c) => objectLoop(params, (it, key) => { + return c[key as 'id'].includes(it) + })) + if (pagination) { + list = list + .splice(pagination.count * pagination.page - 1, pagination.count) + } + return list.map((it) => new Set(lang, it)) + } + + public static findOne(lang: SupportedLanguages, params: Partial> = {}) { + const res = (require(`../../../generated/${lang}/sets.json`) as Array).find((c) => { + return objectLoop(params, (it, key) => { + return c[key as 'id'].includes(it) + }) + }) + if (!res) { + return undefined + } + return new Set(lang, res) + } + + public resume(): SetResume { + return { + id: this.id, + name: this.name, + logo: this.logo, + symbol: this.symbol, + cardCount: { + total: this.cardCount.total, + official: this.cardCount.official + } + } + } + + public full(): SDKSet { + return this.set + } + +} diff --git a/server/src/V2/endpoints/jsonEndpoints.ts b/server/src/V2/endpoints/jsonEndpoints.ts new file mode 100644 index 000000000..3c7bcadb6 --- /dev/null +++ b/server/src/V2/endpoints/jsonEndpoints.ts @@ -0,0 +1,212 @@ +import { objectKeys, objectSize } from '@dzeio/object-util' +import { Card as SDKCard } from '@tcgdex/sdk' +import Card from '../Components/Card' +import Serie from '../Components/Serie' +import Set from '../Components/Set' +import express from 'express' +import apicache from 'apicache' +import { betterSorter, checkLanguage, sendError, unique } from '../../util' + +const server = express.Router() + +const endpointToField: Record = { + "categories": 'category', + 'energy-types': 'energyType', + "hp": 'hp', + 'illustrators': 'illustrator', + "rarities": 'rarity', + 'regulation-marks': 'regulationMark', + "retreats": 'retreat', + "stages": "stage", + "suffixes": "suffix", + "trainer-types": "trainerType", + + // fields that need special care + 'dex-ids': 'dexId', + "sets": "set", + "types": "types", + "variants": "variants", +} + +// server +// .get('/cache/performance', (req, res) => { +// res.json(apicache.getPerformance()) +// }) + +// // add route to display cache index +// .get('/cache/index', (req, res) => { +// res.json(apicache.getIndex()) +// }) + +server + .use(apicache.middleware('1 day', undefined, {})) + + + + /** + * Listing Endpoint + * ex: /v2/en/cards + */ + .get('/:lang/:endpoint', (req, res): void => { + let { lang, endpoint } = req.params + + if (endpoint.endsWith('.json')) { + endpoint = endpoint.replace('.json', '') + } + + if (!checkLanguage(lang)) { + return sendError('LanguageNotFoundError', res, lang) + } + + let result: any + + switch (endpoint) { + case 'cards': + // to be quicker we directly return the raw file + if (objectSize(req.query) === 0) { + result = Card.raw(lang) + return + } else { + result = Card + .find(lang, req.query) + .map((c) => c.resume()) + } + break + + case 'sets': + result = Set + .find(lang, req.query) + .map((c) => c.resume()) + break + case 'series': + result = Serie + .find(lang, req.query) + .map((c) => c.resume()) + break + case 'categories': + case "energy-types": + case "hp": + case "illustrators": + case "rarities": + case "regulation-marks": + case "retreats": + case "series": + case "stages": + case "suffixes": + case "trainer-types": + result = unique( + Card.raw(lang) + .map((c) => c[endpointToField[endpoint]] as string) + .filter((c) => c) + ).sort(betterSorter) + break + case "types": + case "dex-ids": + result = unique( + Card.raw(lang) + .map((c) => c[endpointToField[endpoint]] as Array) + .filter((c) => c) + .reduce((p, c) => [...p, ...c], [] as Array) + ).sort(betterSorter) + break + case "variants": + result = unique( + Card.raw(lang) + .map((c) => objectKeys(c.variants ?? {}) as Array) + .filter((c) => c) + .reduce((p, c) => [...p, ...c], [] as Array) + ).sort() + break + default: + sendError('EndpointNotFoundError', res, endpoint) + return + } + + if (!result) { + sendError('NotFoundError', res) + } + res.json(result) + }) + + /** + * Listing Endpoint + * ex: /v2/en/cards/base1-1 + */ + .get('/:lang/:endpoint/:id', (req, res) => { + let { id, lang, endpoint } = req.params + + if (id.endsWith('.json')) { + id = id.replace('.json', '') + } + + id = id.toLowerCase() + + if (!checkLanguage(lang)) { + return sendError('LanguageNotFoundError', res, lang) + } + + let result: any | undefined + switch (endpoint) { + case 'cards': + result = Card.findOne(lang, {id})?.full() + if (!result) { + result = Card.findOne(lang, {name: id})?.full() + } + break + + case 'sets': + result = Set.findOne(lang, {id})?.full() + if (!result) { + result = Set.findOne(lang, {name: id})?.full() + } + break + + case 'series': + result = Serie.findOne(lang, {id})?.full() + if (!result) { + result = Serie.findOne(lang, {name: id})?.full() + } + break + default: + result = Card.find(lang, {[endpointToField[endpoint]]: id}) + } + if (!result) { + return res.status(404).send({error: "Endpoint or id not found"}) + } + return res.send(result) + + }) + + /** + * sub id Endpoint (for the set endpoint only currently) + * ex: /v2/en/sets/base1/1 + */ + .get('/:lang/:endpoint/:id/:subid', (req, res) => { + let { id, lang, endpoint, subid } = req.params + + if (subid.endsWith('.json')) { + subid = subid.replace('.json', '') + } + + id = id.toLowerCase() + subid = subid.toLowerCase() + + if (!checkLanguage(lang)) { + return sendError('LanguageNotFoundError', res, lang) + } + + let result: any | undefined + + switch (endpoint) { + case 'sets': + result = Card + .findOne(lang, {localId: subid, set: id})?.full() + break + } + if (!result) { + return sendError('NotFoundError', res) + } + return res.send(result) + }) + +export default server diff --git a/server/src/V2/graphql/index.ts b/server/src/V2/graphql/index.ts new file mode 100644 index 000000000..d04b4c30a --- /dev/null +++ b/server/src/V2/graphql/index.ts @@ -0,0 +1,22 @@ +import express from 'express' +import { graphqlHTTP } from 'express-graphql' +import { buildSchema } from 'graphql' +import resolver from './resolver' +import fs from 'fs' + +// Init Express Router +const router = express.Router() +/** + * Drawbacks + * Attack.damage is a string instead of possibly being a number or a string + */ +const schema = buildSchema(fs.readFileSync('./public/v2/graphql.gql').toString()) + +// Add graphql to the route +router.use(graphqlHTTP({ + schema, + rootValue: resolver, + graphiql: true +})) + +export default router diff --git a/server/src/V2/graphql/resolver.ts b/server/src/V2/graphql/resolver.ts new file mode 100644 index 000000000..e1dbf42ed --- /dev/null +++ b/server/src/V2/graphql/resolver.ts @@ -0,0 +1,49 @@ +import Card from '../Components/Card' +import { Options } from '../../interfaces' +import Serie from '../Components/Serie' +import Set from '../Components/Set' +import { SupportedLanguages } from '@tcgdex/sdk' +import { checkLanguage } from '../../util' + +const middleware = = Record>(fn: (lang: SupportedLanguages, query: Q) => any) => ( + data: Q, + _: any, + e: any +) => { + let lang = e?.fieldNodes?.[0]?.directives?.[0]?.arguments?.[0]?.value?.value + if (!lang) { + lang = 'en' + } + if (!checkLanguage(lang)) { + return undefined + } + return fn(lang, data) +} + +export default { + // Cards Endpoints + cards: middleware>((lang, query) => { + return Card.find(lang, query.filters ?? {}, query.pagination) + }), + card: middleware<{set?: string, id: string}>((lang, query) => { + const toSearch = query.set ? 'localId' : 'id' + return Card.findOne(lang, {[toSearch]: query.id}) + }), + + // Set Endpoints + set: middleware<{id: string}>((lang, query) => { + return Set.findOne(lang, {id: query.id}) ?? Set.findOne(lang, {name: query.id}) + }), + sets: middleware>((lang, query) => { + return Set.find(lang, query.filters ?? {}, query.pagination) + }), + + // Serie Endpoints + serie: middleware<{id: string}>((lang, query) => { + return Serie.findOne(lang, {id: query.id}) ?? Serie.findOne(lang, {name: query.id}) + }), + series: middleware>((lang, query) => { + return Serie.find(lang, query.filters ?? {}, query.pagination) + }), + +}; diff --git a/server/src/index.ts b/server/src/index.ts new file mode 100644 index 000000000..972653b9b --- /dev/null +++ b/server/src/index.ts @@ -0,0 +1,35 @@ +import express from 'express' +import graphql from './V2/graphql' +import jsonEndpoints from './V2/endpoints/jsonEndpoints' + +// Current API version +const VERSION = 2 + +// Init Express server +const server = express() + +// Set global headers +server.use((_, res, next) => { + res + .setHeader('Access-Control-Allow-Origin', '*') + .setHeader('Access-Control-Allow-Methods', 'GET,POST,OPTIONS') + .setHeader('Access-Control-Allow-Headers', 'DNT,User-Agent,X-Requested-With,If-Modified-Since,Cache-Control,Content-Type,Range') + .setHeader('Access-Control-Expose-Headers', 'Content-Length,Content-Range') + next() +}) + +server.get('/', (_, res) => { + res.redirect('https://www.tcgdex.net/docs?ref=api.tcgdex.net') +}) + +server.use(express.static('./public')) + +// Setup GraphQL +server.use(`/v${VERSION}/graphql`, graphql) + +// Setup JSON endpoints +server.use(`/v${VERSION}`, jsonEndpoints) + +// Start server +server.listen(3000) +console.log(`🚀 Server ready at localhost:3000`); diff --git a/server/src/interfaces.d.ts b/server/src/interfaces.d.ts new file mode 100644 index 000000000..6d1677d42 --- /dev/null +++ b/server/src/interfaces.d.ts @@ -0,0 +1,11 @@ +import { SupportedLanguages } from '@tcgdex/sdk' + +export interface Pagination { + page: number + count: number +} + +export interface Options { + pagination?: Pagination + filters?: Partial> +} diff --git a/server/src/util.ts b/server/src/util.ts new file mode 100644 index 000000000..d60394a27 --- /dev/null +++ b/server/src/util.ts @@ -0,0 +1,57 @@ +import { SupportedLanguages } from '@tcgdex/sdk' +import { Response } from 'express' +import fs from 'fs' + +export function checkLanguage(str: string): str is SupportedLanguages { + return ['en', 'fr', 'es', 'it', 'pt', 'de'].includes(str) +} + +export function unique(arr: Array): Array { + return arr.reduce((p, c) => p.includes(c) ? p : [...p, c], [] as Array) +} + +export function sendError(error: 'UnknownError' | 'NotFoundError' | 'LanguageNotFoundError' | 'EndpointNotFoundError', res: Response, v?: any) { + let message = '' + let status = 404 + switch (error) { + case 'LanguageNotFoundError': + message = `Language not found (${v})` + break + case 'EndpointNotFoundError': + message = `Endpoint not found (${v})` + break + case 'NotFoundError': + message = 'The resource you are searching does not exists' + break + case 'UnknownError': + default: + message = `an unknown error occured (${v})` + status = 500 + break + } + res.status(status).json({ + message + }).end() + +} + +export function betterSorter(a: string, b: string) { + const ra = parseInt(a, 10) + const rb = parseInt(b, 10) + if (!isNaN(ra) && !isNaN(rb)) { + return ra - rb + } + return a >= b ? 1 : -1 +} + +export function tree(path: string, padding = 0) { + const folder = fs.readdirSync(path) + for (const file of folder) { + const filePath = path + '/' + file + console.log(filePath.padStart(padding, '-')) + try { + fs.lstatSync(filePath).isDirectory() + tree(filePath) + } catch {} + } +} diff --git a/server/tsconfig.json b/server/tsconfig.json new file mode 100644 index 000000000..d0b5512b4 --- /dev/null +++ b/server/tsconfig.json @@ -0,0 +1,7 @@ +{ + "extends": "./node_modules/@dzeio/config/tsconfig.base", + "compilerOptions": { + "outDir": "dist" + }, + "include": ["src"] +} diff --git a/tsconfig.json b/tsconfig.json index fa25d424b..9e655ad13 100644 --- a/tsconfig.json +++ b/tsconfig.json @@ -1,9 +1,10 @@ { - "include": ["data", "interfaces.d.ts", "meta"], + "include": ["data", "interfaces.d.ts"], "compilerOptions": { "target": "es5", "module": "commonjs", "esModuleInterop": true, - "forceConsistentCasingInFileNames": true + "forceConsistentCasingInFileNames": true, + "resolveJsonModule": true } }