├── .docker └── compose │ ├── .env │ ├── docker-compose.yml │ ├── services.yml │ └── update_and_restart.sh ├── .github └── workflows │ ├── app.yml │ └── browser.yml ├── .gitignore ├── .nginx └── example.com.conf ├── .vscode ├── extensions.json └── settings.json ├── CHANGELOG.md ├── README.md ├── app ├── .editorconfig ├── .env.example ├── .eslintignore ├── .eslintrc.json ├── .gitignore ├── .prettierignore ├── .prettierrc ├── Dockerfile ├── Readme.md ├── bin │ └── docker-build.sh ├── copy-files.ts ├── jest.config.js ├── nodemon.json ├── package-lock.json ├── package.json ├── src │ ├── app.ts │ ├── config.ts │ ├── dev.ts │ ├── endpoints │ │ └── v1 │ │ │ ├── fetch-dev.ts │ │ │ └── fetch.ts │ ├── index.ts │ ├── lib │ │ ├── blocked.ts │ │ ├── browser-manager.ts │ │ ├── google.ts │ │ ├── redis.ts │ │ └── server.ts │ ├── middleware │ │ ├── authByApiKey.ts │ │ └── index.ts │ └── zod-schema │ │ ├── api.ts │ │ └── index.ts └── tsconfig.json ├── browser ├── Dockerfile ├── bin │ └── docker-build.sh ├── entrypoint.sh ├── package-lock.json ├── package.json └── worker.js ├── package-lock.json └── package.json /.docker/compose/.env: -------------------------------------------------------------------------------- 1 | REDIS_URL=redis://redis:6379 2 | 3 | # Api key 4 | # app/src/middleware/authByApiKey.ts 5 | # generator https://tokengenerator.dev/ 6 | SECRET_API_KEY=your_api_key -------------------------------------------------------------------------------- /.docker/compose/docker-compose.yml: -------------------------------------------------------------------------------- 1 | 2 | services: 3 | redis: 4 | image: redis:alpine 5 | volumes: 6 | - ./redis-data:/data 7 | restart: unless-stopped 8 | ports: 9 | - "127.0.0.1:6380:6379" 10 | app: 11 | depends_on: 12 | - redis 13 | extends: 14 | file: services.yml 15 | service: app 16 | ports: 17 | - "127.0.0.1:3100:3000" 18 | 19 | browser-node-1: 20 | depends_on: 21 | - redis 22 | environment: 23 | - HOST_NAME=browser-node-1 24 | ports: 25 | - "127.0.0.1:9701:8080" 26 | extends: 27 | file: services.yml 28 | service: browser-node 29 | 30 | browser-node-2: 31 | depends_on: 32 | - redis 33 | environment: 34 | - HOST_NAME=browser-node-2 35 | ports: 36 | - "127.0.0.1:9702:8080" 37 | extends: 38 | file: services.yml 39 | service: browser-node 40 | 41 | browser-node-3: 42 | depends_on: 43 | - redis 44 | environment: 45 | - HOST_NAME=browser-node-3 46 | ports: 47 | - "127.0.0.1:9703:8080" 48 | extends: 49 | file: services.yml 50 | service: browser-node 51 | 52 | browser-node-4: 53 | depends_on: 54 | - redis 55 | environment: 56 | - HOST_NAME=browser-node-4 57 | ports: 58 | - "127.0.0.1:9704:8080" 59 | extends: 60 | file: services.yml 61 | service: browser-node -------------------------------------------------------------------------------- /.docker/compose/services.yml: -------------------------------------------------------------------------------- 1 | services: 2 | browser-node: 3 | image: seadfeng/playwright-browser-worker 4 | env_file: .env 5 | restart: unless-stopped 6 | deploy: 7 | resources: 8 | limits: 9 | memory: 2GB 10 | reservations: 11 | memory: 1GB 12 | app: 13 | image: seadfeng/playwright-browser-app 14 | env_file: .env 15 | restart: unless-stopped -------------------------------------------------------------------------------- /.docker/compose/update_and_restart.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | 3 | # Pull the latest images 4 | docker compose pull 5 | 6 | docker compose up -d --no-deps --build 7 | -------------------------------------------------------------------------------- /.github/workflows/app.yml: -------------------------------------------------------------------------------- 1 | name: "Express App Deploy" 2 | on: 3 | release: 4 | types: 5 | - published 6 | paths: 7 | - 'app/**' 8 | - 'app/package.json' 9 | - '.github/workflows/app.yml' 10 | jobs: 11 | deploy: 12 | runs-on: ubuntu-latest 13 | environment: 14 | name: production 15 | url: ${{ vars.DEPOLY_URL }} 16 | env: 17 | SSH_PRIVATE_KEY: ${{ secrets.SSH_PRIVATE_KEY }} 18 | HOST: ${{ secrets.HOST }} 19 | SSH_DP_USER: ${{ secrets.SSH_DP_USER }} 20 | steps: 21 | - name: Checkout Repository 22 | uses: actions/checkout@v3 23 | 24 | - name: Set up QEMU 25 | uses: docker/setup-qemu-action@v3 26 | 27 | - name: Setup Docker Buildx 28 | id: docker_buildx 29 | uses: docker/setup-buildx-action@v3 30 | 31 | - name: Docker Build browser-app 32 | run: | 33 | echo ${{ secrets.DOCKER_LOGIN }} | docker login -u ${{ github.actor }} --password-stdin 34 | docker buildx create --use 35 | docker buildx inspect --bootstrap 36 | 37 | VERSION=$(node -p "require('./app/package.json').version") 38 | APP=$(node -p "require('./app/package.json').name") 39 | 40 | echo "Build: ${APP}.${VERSION}" 41 | 42 | docker buildx build \ 43 | --push \ 44 | --platform linux/amd64,linux/arm64/v8 \ 45 | --build-arg VERSION=$VERSION \ 46 | -t ${{ github.actor }}/playwright-${APP}:$VERSION \ 47 | -t ${{ github.actor }}/playwright-${APP}:latest \ 48 | -f ./app/Dockerfile \ 49 | ./app 50 | 51 | echo "${{ github.actor }}/${APP}:${VERSION}" 52 | echo "${{ github.actor }}/${APP}:latest" 53 | 54 | - name: Configure SSH FOR self_host 55 | run: | 56 | mkdir -p ~/.ssh/ 57 | echo "$SSH_PRIVATE_KEY" > ~/.ssh/id_rsa 58 | chmod 600 ~/.ssh/id_rsa 59 | ssh-keyscan github.com >> ~/.ssh/known_hosts 60 | eval "$(ssh-agent)" 61 | ssh-add ~/.ssh/id_rsa 62 | cat >>~/.ssh/config < ~/.ssh/id_rsa 57 | chmod 600 ~/.ssh/id_rsa 58 | ssh-keyscan github.com >> ~/.ssh/known_hosts 59 | eval "$(ssh-agent)" 60 | ssh-add ~/.ssh/id_rsa 61 | cat >>~/.ssh/config < { 25 | console.log("Directory copied successfully!"); 26 | }) 27 | .catch((err) => { 28 | console.error("Error copying directory:", err); 29 | }); 30 | -------------------------------------------------------------------------------- /app/jest.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | preset: "ts-jest", 3 | testEnvironment: "node", 4 | moduleNameMapper: { 5 | "^@tests/(.*)$": "/src/__tests__/$1", 6 | "^@/(.*)$": "/src/$1", 7 | }, 8 | }; 9 | -------------------------------------------------------------------------------- /app/nodemon.json: -------------------------------------------------------------------------------- 1 | { 2 | "watch": [ 3 | "src" 4 | ], 5 | "ext": "ts", 6 | "ignore": [ 7 | "src/**/*.spec.ts" 8 | ], 9 | "exec": "tsx src/index.ts" 10 | } -------------------------------------------------------------------------------- /app/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "browser-app", 3 | "version": "1.0.5", 4 | "scripts": { 5 | "dev": "npx nodemon", 6 | "start": "npx nodemon", 7 | "docker:build": "sh bin/docker-build.sh", 8 | "build": "npx tsup src --clean --format cjs --env.NODE_ENV production && npx tsx copy-files.ts", 9 | "test": "dotenv -c -- npx jest" 10 | }, 11 | "dependencies": { 12 | "dotenv": "^16.4.7", 13 | "express": "^4.18.2", 14 | "header-generator": "^2.1.62", 15 | "https-proxy-agent": "^7.0.6", 16 | "ioredis": "^5.3.2", 17 | "morgan": "^1.10.0", 18 | "node-fetch": "^3.3.2", 19 | "playwright": "^1.50.0", 20 | "response-time": "^2.3.3", 21 | "zod": "^3.24.1" 22 | }, 23 | "devDependencies": { 24 | "@types/debug": "^4.1.12", 25 | "@types/dotenv": "^8.2.3", 26 | "@types/express": "^4.17.21", 27 | "@types/jest": "^29.5.12", 28 | "@types/morgan": "^1.9.9", 29 | "@types/node": "^20.14.11", 30 | "@types/response-time": "^2.3.8", 31 | "@typescript-eslint/eslint-plugin": "^7.16.1", 32 | "@typescript-eslint/parser": "^7.16.1", 33 | "eslint": "^8.57.0", 34 | "eslint-config-prettier": "^9.1.0", 35 | "eslint-plugin-prettier": "^5.2.1", 36 | "fs-extra": "^11.2.0", 37 | "jest": "^29.7.0", 38 | "nodemon": "^3.1.9", 39 | "prettier": "^3.3.3", 40 | "ts-jest": "^29.2.3", 41 | "tsup": "^8.3.6", 42 | "tsx": "^4.19.3" 43 | } 44 | } -------------------------------------------------------------------------------- /app/src/app.ts: -------------------------------------------------------------------------------- 1 | import { fetchTask } from "endpoints/v1/fetch"; 2 | import { fetchTaskDev } from "endpoints/v1/fetch-dev"; 3 | import express, { Application } from "express"; 4 | import { authByApiKey } from "middleware"; 5 | import logger from "morgan"; 6 | import responseTime from "response-time"; 7 | 8 | const app: Application = express(); 9 | 10 | app.use(responseTime()); 11 | app.use(logger("dev")); 12 | app.use(authByApiKey); 13 | app.use(express.json()); 14 | app.use(express.urlencoded({ extended: true })); 15 | 16 | app.post("/api/v1/fetch", fetchTask); 17 | if (process.env.NODE_ENV !== "production") { 18 | app.post("/api/v1/fetch-dev", fetchTaskDev); 19 | } 20 | 21 | export default app; 22 | -------------------------------------------------------------------------------- /app/src/config.ts: -------------------------------------------------------------------------------- 1 | export const TRACKING_HOSTS = [ 2 | "google-analytics.com", 3 | "doubleclick.net", 4 | "facebook.net", 5 | "amplitude.com", 6 | "segment.com", 7 | "mixpanel.com", 8 | "hotjar.com", 9 | "newrelic.com", 10 | ]; 11 | -------------------------------------------------------------------------------- /app/src/dev.ts: -------------------------------------------------------------------------------- 1 | // for local development 2 | export const BROWSERS = { 3 | "browser-node-1": "127.0.0.1:9701", 4 | "browser-node-2": "127.0.0.1:9702", 5 | "browser-node-3": "127.0.0.1:9703", 6 | "browser-node-4": "127.0.0.1:9704", 7 | }; 8 | -------------------------------------------------------------------------------- /app/src/endpoints/v1/fetch-dev.ts: -------------------------------------------------------------------------------- 1 | import { Request, Response } from "express"; 2 | import { executeTask } from "lib/server"; 3 | import { chromium } from "playwright"; 4 | import { fetchParamsSchema } from "zod-schema"; 5 | 6 | export const fetchTaskDev = async (req: Request, res: Response) => { 7 | try { 8 | const { url, locale = "en-US", proxy } = fetchParamsSchema.parse(req.body); 9 | let status = 500; 10 | let html = ""; 11 | const browser = await chromium.launch({ 12 | args: [ 13 | "--no-sandbox", // Disable sandbox for container environments 14 | "--disable-setuid-sandbox", // Disable setuid sandbox 15 | "--disable-dev-shm-usage", // Overcome limited shared memory in containers 16 | "--disable-gpu", // Disable GPU hardware acceleration 17 | 18 | // Anti-detection settings 19 | "--disable-blink-features=AutomationControlled", // Hide automation flags 20 | // '--disable-features=IsolateOrigins,site-per-process', // Disable site isolation 21 | 22 | // Additional performance tweaks 23 | "--disable-web-security", // Disable web security for testing 24 | "--disable-notifications", // Disable browser notifications 25 | "--ignore-certificate-errors", // Ignore SSL/TLS errors 26 | "--window-size=1920,1080", // Set standard window size 27 | ], 28 | headless: false, // Run in headless mode 29 | }); 30 | try { 31 | const result = await executeTask({ url, browser, locale, proxy }); 32 | 33 | if (result.headers) { 34 | Object.entries(result.headers).forEach(([key, value]) => { 35 | res.setHeader(key, value); 36 | }); 37 | } 38 | 39 | status = result.status; 40 | html = result.html; 41 | } catch (error) { 42 | console.error("Task execution failed:", error); 43 | status = 500; 44 | html = "Task execution failed" + ((error as Error)?.message ?? ""); 45 | } finally { 46 | console.info("browser close"); 47 | await browser.close(); 48 | } 49 | return res.status(status).send(html); 50 | } catch (error) { 51 | console.error(error); 52 | const text = "error:" + ((error as Error)?.message ?? ""); 53 | return res.status(500).send(text); 54 | } 55 | }; 56 | -------------------------------------------------------------------------------- /app/src/endpoints/v1/fetch.ts: -------------------------------------------------------------------------------- 1 | import { Request, Response } from "express"; 2 | import { BrowserManager } from "lib/browser-manager"; 3 | import { executeTask } from "lib/server"; 4 | import { fetchParamsSchema } from "zod-schema"; 5 | 6 | export const fetchTask = async (req: Request, res: Response) => { 7 | try { 8 | const { url, locale = "en-US", proxy } = fetchParamsSchema.parse(req.body); 9 | await BrowserManager.init(); 10 | let status = 500; 11 | let html = ""; 12 | try { 13 | const browser = await BrowserManager.getBrowser(); 14 | 15 | let retry = 0; 16 | 17 | while (retry <= 3) { 18 | const result = await executeTask({ 19 | url, 20 | browser, 21 | locale, 22 | proxy, 23 | }); 24 | if (result.status === 429) { 25 | retry += 1; 26 | console.warn("Retry... ", retry); 27 | continue; 28 | } 29 | if (result.status !== 200) { 30 | throw new Error(`unkown error, result.status: ${result.status}`); 31 | } 32 | 33 | retry = 10; 34 | if (result.headers) { 35 | Object.entries(result.headers).forEach(([key, value]) => { 36 | res.setHeader(key, value); 37 | }); 38 | } 39 | 40 | status = result.status; 41 | html = result.html; 42 | } 43 | } catch (error) { 44 | console.error("Task execution failed:", error); 45 | status = 500; 46 | html = "Task execution failed.\n" + ((error as Error)?.message ?? ""); 47 | } 48 | 49 | return res.status(status).send(html); 50 | } catch (error) { 51 | console.error(error); 52 | const text = "error:" + ((error as Error)?.message ?? ""); 53 | return res.status(500).send(text); 54 | } 55 | }; 56 | -------------------------------------------------------------------------------- /app/src/index.ts: -------------------------------------------------------------------------------- 1 | import debug from "debug"; 2 | import "dotenv/config"; 3 | import http from "http"; 4 | import { AddressInfo } from "net"; 5 | import app from "./app"; 6 | 7 | const serverDebug = debug("starter:server"); 8 | let serverStarted = false; 9 | /** 10 | * Get port from environment and store in Express. 11 | */ 12 | const DEFAULT_PORT = 3000; 13 | const port = normalizePort(process.env.PORT || DEFAULT_PORT.toString()) as 14 | | string 15 | | number; 16 | const host = process.env.HOST || "0.0.0.0"; 17 | app.set("port", port); 18 | app.set("host", host); 19 | 20 | /** 21 | * Create HTTP server. 22 | */ 23 | const server: http.Server = http.createServer(app); 24 | 25 | /** 26 | * Listen on provided port, on all network interfaces. 27 | */ 28 | startServer(port); 29 | 30 | function startServer(port: string | number) { 31 | server.listen(port); 32 | server.on("error", (error) => onError(error, port)); 33 | server.on("listening", onListening); 34 | } 35 | 36 | /** 37 | * Normalize a port into a number, string, or false. 38 | */ 39 | function normalizePort(val: string | number): string | number | boolean { 40 | const parsedPort: number = parseInt(val as string, 10); 41 | 42 | if (isNaN(parsedPort)) { 43 | // named pipe 44 | return val; 45 | } 46 | 47 | if (parsedPort >= 0) { 48 | // port number 49 | return parsedPort; 50 | } 51 | 52 | return false; 53 | } 54 | 55 | /** 56 | * Event listener for HTTP server "error" event. 57 | */ 58 | function onError(error: NodeJS.ErrnoException, port: string | number): void { 59 | if (error.syscall !== "listen") { 60 | throw error; 61 | } 62 | 63 | const bind: string = 64 | typeof port === "string" ? "Pipe " + port : "Port " + port; 65 | 66 | // handle specific listen errors with friendly messages 67 | switch (error.code) { 68 | case "EACCES": { 69 | console.error(bind + " requires elevated privileges"); 70 | process.exit(1); 71 | } 72 | case "EADDRINUSE": { 73 | console.error(bind + " is already in use"); 74 | const nextPort = (typeof port === "number" ? port : parseInt(port)) + 1; 75 | console.log( 76 | `Port ${port} is in use, attempting to use next port: ${nextPort}`, 77 | ); 78 | startServer(nextPort); 79 | break; 80 | } 81 | default: 82 | throw error; 83 | } 84 | } 85 | 86 | /** 87 | * Event listener for HTTP server "listening" event. 88 | */ 89 | function onListening(): void { 90 | if (!serverStarted) { 91 | serverStarted = true; 92 | const addr = server.address(); 93 | if (addr && typeof addr === "object") { 94 | const bind: string = "port " + (addr as AddressInfo).port; 95 | const url = `http://127.0.0.1:${(addr as AddressInfo).port}`; 96 | console.log(`Listening on ${bind}`); 97 | console.info(`Server running at ${url}`); 98 | serverDebug("Listening on " + bind); 99 | } 100 | } 101 | } 102 | -------------------------------------------------------------------------------- /app/src/lib/blocked.ts: -------------------------------------------------------------------------------- 1 | import { TRACKING_HOSTS } from "config"; 2 | import { Page } from "playwright"; 3 | 4 | export const needBlocked = async (page: Page) => { 5 | await page.route("**/*", (route) => { 6 | const request = route.request(); 7 | const url = new URL(request.url()); 8 | const resourceType = request.resourceType(); 9 | 10 | const blockedTypes = ["image", "media", "stylesheet", "font"]; 11 | 12 | const matchHost = (host: string, hostList: string[]) => 13 | hostList.some( 14 | (allowedHost) => 15 | host === allowedHost || host.endsWith(`.${allowedHost}`), 16 | ); 17 | 18 | if (blockedTypes.includes(resourceType)) { 19 | route.abort("blockedbyclient"); 20 | return; 21 | } 22 | 23 | if (resourceType === "script" && matchHost(url.hostname, TRACKING_HOSTS)) { 24 | route.abort("blockedbyclient"); 25 | return; 26 | } 27 | route.continue(); 28 | }); 29 | }; 30 | 31 | export const navigationBlocked = async (page: Page) => { 32 | await page.route("**/*", async (route) => { 33 | if ( 34 | route.request().isNavigationRequest() && 35 | route.request().url().includes("sei") 36 | ) { 37 | console.log("Blocked navigation to:", route.request().url()); 38 | await route.abort(); 39 | } else { 40 | await route.continue(); 41 | } 42 | }); 43 | }; 44 | -------------------------------------------------------------------------------- /app/src/lib/browser-manager.ts: -------------------------------------------------------------------------------- 1 | const BROWSER_CURRENT = "browser_current"; 2 | const BROWSER_STATUS = "browser_status"; 3 | 4 | import { BROWSERS } from "dev"; 5 | import { Browser, chromium } from "playwright"; 6 | import { redis } from "./redis"; 7 | 8 | export class BrowserManager { 9 | private static browserConnections: Map = new Map(); 10 | private static browserIds: string[] = []; 11 | public static total: number = 0; 12 | 13 | static async count(): Promise { 14 | if (this.total === 0) { 15 | console.log("set total"); 16 | const [allStatus] = await Promise.all([ 17 | redis.hgetall(BROWSER_STATUS), 18 | redis.set(BROWSER_CURRENT, 0), 19 | ]); 20 | const keys = Object.keys(allStatus); 21 | this.total = keys.length; 22 | } 23 | return this.total; 24 | } 25 | 26 | static async init(): Promise { 27 | await this.count(); 28 | if (this.total > 0) { 29 | console.log("total", this.total); 30 | const keys = Object.keys(await redis.hgetall(BROWSER_STATUS)); 31 | this.browserIds = keys; 32 | } else { 33 | throw new Error("browser not found"); 34 | } 35 | return this.browserIds; 36 | } 37 | 38 | static async nextBrowserId(): Promise { 39 | const [current] = await Promise.all([redis.incr(BROWSER_CURRENT)]); 40 | const index = current % this.total; 41 | 42 | return this.browserIds[index]; 43 | } 44 | 45 | static async getBrowser(): Promise { 46 | const browserId = await this.nextBrowserId(); 47 | 48 | const getWsEndpoint = (wsEndpoint: string) => { 49 | const hostName = browserId.split("_")[1] as keyof typeof BROWSERS; 50 | console.log("hostName", hostName); 51 | console.log("wsEndpoint", wsEndpoint); 52 | return process.env.NODE_ENV !== "production" 53 | ? wsEndpoint.replace(/\/.*:8080/, `/${BROWSERS[hostName]}`) 54 | : wsEndpoint; 55 | }; 56 | 57 | const createConnection = async (browserId: string) => { 58 | const wsEndpoint = await redis.get(`ws:${browserId}`); 59 | if (!wsEndpoint) { 60 | throw new Error(`No websocket endpoint found for browser ${browserId}`); 61 | } 62 | 63 | const browser = await chromium.connect({ 64 | wsEndpoint: getWsEndpoint(wsEndpoint), 65 | }); 66 | this.browserConnections.set(browserId, browser); 67 | return browser; 68 | }; 69 | 70 | if (!this.browserConnections.has(browserId)) { 71 | return createConnection(browserId); 72 | } 73 | 74 | const browser = this.browserConnections.get(browserId)!; 75 | 76 | if (!browser.isConnected) { 77 | this.browserConnections.delete(browserId); 78 | console.warn("ReConnected:", browserId); 79 | return createConnection(browserId); 80 | } else { 81 | console.warn("Found browserId:", browserId); 82 | return browser; 83 | } 84 | } 85 | } 86 | -------------------------------------------------------------------------------- /app/src/lib/google.ts: -------------------------------------------------------------------------------- 1 | import { Page } from "playwright"; 2 | 3 | /** 4 | * 5 | * Recaptcha Try to pass 6 | * 7 | * @param page 8 | * @returns 9 | */ 10 | export const googleRecaptcha = async (page: Page) => { 11 | console.info("Recaptcha", "Try to pass"); 12 | const iframe = page.locator('iframe[title="reCAPTCHA"]'); 13 | const frameLocator = iframe.contentFrame(); 14 | const checkbox = frameLocator.locator(".recaptcha-checkbox"); 15 | 16 | if ((await checkbox.count()) > 0) { 17 | console.log("recaptcha checkbox found"); 18 | await checkbox.waitFor({ state: "visible", timeout: 8000 }); 19 | await checkbox.scrollIntoViewIfNeeded(); 20 | 21 | await checkbox.click(); 22 | console.log("checkbox click"); 23 | await frameLocator 24 | .locator(".recaptcha-checkbox-loading") 25 | .waitFor({ state: "visible", timeout: 8000 }); 26 | await frameLocator 27 | .locator(".recaptcha-checkbox-loading") 28 | .waitFor({ state: "detached", timeout: 8000 }); 29 | 30 | const imageselect = page.locator(".g-recaptcha-bubble-arrow"); // iframe #rc-imageselect 31 | console.log("imageselect count", await imageselect.count()); 32 | if ((await imageselect.count()) > 0) { 33 | return true; 34 | } else { 35 | return false; 36 | } 37 | } 38 | }; 39 | -------------------------------------------------------------------------------- /app/src/lib/redis.ts: -------------------------------------------------------------------------------- 1 | import Redis from "ioredis"; 2 | 3 | export const redis = new Redis(process.env.REDIS_URL!); 4 | -------------------------------------------------------------------------------- /app/src/lib/server.ts: -------------------------------------------------------------------------------- 1 | import { HeaderGenerator } from "header-generator"; 2 | import { IncomingMessage, ServerResponse } from "http"; 3 | import { Browser, BrowserContextOptions, Cookie } from "playwright"; 4 | import { googleRecaptcha } from "./google"; 5 | 6 | interface Headers { 7 | [key: string]: string; 8 | } 9 | 10 | interface ExecuteTaskOptions { 11 | url: string; 12 | browser: Browser; 13 | locale?: string; 14 | proxy?: string; 15 | } 16 | 17 | interface TaskResult { 18 | status: number; 19 | html: string; 20 | cookies?: Cookie[]; 21 | executionTime?: string; 22 | headers?: Headers; 23 | url?: string; 24 | } 25 | 26 | function sanitizeHeaders(headers: Headers): Headers { 27 | const sanitized: Headers = {}; 28 | for (const [key, value] of Object.entries(headers)) { 29 | try { 30 | // 创建一个模拟的 IncomingMessage 对象 31 | const req = Object.create(IncomingMessage.prototype); 32 | const testResponse = new ServerResponse(req as IncomingMessage); 33 | testResponse.setHeader(key, value); 34 | const ignoredHeaders = ["content-encoding", "transfer-encoding"]; 35 | 36 | if (!ignoredHeaders.includes(key)) { 37 | sanitized[key] = value; 38 | } 39 | } catch (e) { 40 | // console.warn(`Skipping invalid header ${key}: ${value}`); 41 | } 42 | } 43 | return sanitized; 44 | } 45 | 46 | async function executeTask({ 47 | url, 48 | browser, 49 | locale = "en-US", 50 | proxy, 51 | }: ExecuteTaskOptions): Promise { 52 | const startTime = Date.now(); 53 | try { 54 | const headerGenerator = new HeaderGenerator(); 55 | const headers = headerGenerator.getHeaders({ 56 | operatingSystems: ["macos"], 57 | browsers: [ 58 | { name: "firefox", minVersion: 80 }, 59 | { name: "chrome", minVersion: 87 }, 60 | "safari", 61 | ], 62 | devices: ["desktop"], 63 | locales: [locale], 64 | }); 65 | 66 | const contextOptions: BrowserContextOptions = { 67 | locale, 68 | viewport: { width: 1920, height: 1080 }, 69 | // deviceScaleFactor: 1, 70 | userAgent: headers["user-agent"], 71 | }; 72 | 73 | if (proxy) { 74 | const { password, username, protocol, host } = new URL(proxy); 75 | const server = `${protocol}//${host}`; 76 | contextOptions.proxy = { server }; 77 | if (username) 78 | contextOptions.proxy.username = decodeURIComponent(username); 79 | if (password) 80 | contextOptions.proxy.password = decodeURIComponent(password); 81 | } 82 | 83 | const context = await browser.newContext(contextOptions); 84 | const page = await context.newPage(); 85 | // await needBlocked(page); 86 | // await navigationBlocked(page); 87 | 88 | try { 89 | const response = await page.goto(url, { 90 | waitUntil: "domcontentloaded", 91 | }); 92 | 93 | if (!response) { 94 | throw new Error("No response received"); 95 | } 96 | 97 | const [status, html, recaptchaCount] = await Promise.all([ 98 | response.status(), 99 | page.content(), 100 | page.locator("#recaptcha").count(), 101 | ]); 102 | 103 | const responseHeaders = sanitizeHeaders(response.headers()); 104 | const cookies = await context.cookies(); 105 | 106 | if (recaptchaCount > 0 && (await googleRecaptcha(page))) { 107 | const [html] = await Promise.all([page.content()]); 108 | 109 | const endTime = Date.now(); 110 | const executionTime = endTime - startTime; 111 | return { 112 | status, 113 | html: `\n\n${html}`, 114 | executionTime: `${executionTime / 1000} s`, 115 | url: page.url(), 116 | cookies, 117 | headers, 118 | }; 119 | } 120 | 121 | const endTime = Date.now(); 122 | const executionTime = endTime - startTime; 123 | 124 | console.info( 125 | "executeTask status", 126 | status, 127 | "execution time:", 128 | executionTime / 1000, 129 | "s", 130 | ); 131 | 132 | return { 133 | status, 134 | html, 135 | executionTime: `${executionTime / 1000} s`, 136 | headers: responseHeaders, 137 | cookies, 138 | url: page.url(), 139 | }; 140 | } finally { 141 | await page.close(); 142 | await context.close(); 143 | } 144 | } catch (error) { 145 | const message = (error as Error)?.message ?? "unknown error"; 146 | console.error(message); 147 | return { 148 | status: 500, 149 | html: message, 150 | }; 151 | } 152 | } 153 | 154 | export { executeTask, type ExecuteTaskOptions, type TaskResult }; 155 | -------------------------------------------------------------------------------- /app/src/middleware/authByApiKey.ts: -------------------------------------------------------------------------------- 1 | import { NextFunction, Request, Response } from "express"; 2 | 3 | export const authByApiKey = ( 4 | req: Request, 5 | res: Response, 6 | next: NextFunction, 7 | ) => { 8 | let { token } = req.query as { token: string }; // ?token=** 9 | const apiKeyHeader = req.headers["x-api-key"] as string; 10 | const apiKey = process.env.SECRET_API_KEY!; 11 | 12 | if (!token) { 13 | token = apiKeyHeader; 14 | } 15 | 16 | if (!token) { 17 | return res 18 | .status(403) 19 | .json({ error: "Unauthorized", message: "Missing x-api-key" }); 20 | } 21 | if (apiKey !== token) { 22 | return res 23 | .status(403) 24 | .json({ error: "Unauthorized", message: "x-api-key not match" }); 25 | } 26 | delete req.headers["x-api-key"]; 27 | next(); 28 | }; 29 | -------------------------------------------------------------------------------- /app/src/middleware/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./authByApiKey"; 2 | -------------------------------------------------------------------------------- /app/src/zod-schema/api.ts: -------------------------------------------------------------------------------- 1 | import { z } from "zod"; 2 | 3 | export const fetchParamsSchema = z.object({ 4 | url: z.string().url(), 5 | locale: z.string().optional(), 6 | proxy: z.string().optional(), 7 | }); 8 | export type FetchParams = z.infer; 9 | -------------------------------------------------------------------------------- /app/src/zod-schema/index.ts: -------------------------------------------------------------------------------- 1 | export * from "./api"; 2 | -------------------------------------------------------------------------------- /app/tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "allowJs": true, 4 | "forceConsistentCasingInFileNames": true, 5 | "declaration": true, 6 | "noEmit": true, 7 | "sourceMap": true, 8 | "esModuleInterop": true, 9 | "inlineSourceMap": false, 10 | "baseUrl": "./src", 11 | "lib": ["esnext"], 12 | "listEmittedFiles": false, 13 | "listFiles": false, 14 | "moduleResolution": "node", 15 | "allowImportingTsExtensions": true, 16 | "noFallthroughCasesInSwitch": true, 17 | "pretty": true, 18 | "isolatedModules": true, 19 | "resolveJsonModule": true, 20 | "skipLibCheck": true, 21 | "strict": true, 22 | "traceResolution": false, 23 | "target": "esnext", 24 | "module": "CommonJS", 25 | "types": ["@types/node", "jest"], 26 | "paths": { 27 | "@/*": ["./src/*"] 28 | } 29 | }, 30 | "exclude": ["node_modules", "dist", "__tests__"], 31 | "include": ["./src/**/*.ts"] 32 | } 33 | -------------------------------------------------------------------------------- /browser/Dockerfile: -------------------------------------------------------------------------------- 1 | FROM mcr.microsoft.com/playwright:v1.50.0-jammy 2 | WORKDIR /app 3 | 4 | COPY package*.json ./ 5 | 6 | RUN npm install 7 | 8 | RUN npx playwright install-deps chromium 9 | 10 | COPY . . 11 | 12 | RUN chmod +x ./entrypoint.sh 13 | 14 | ENTRYPOINT ["./entrypoint.sh"] -------------------------------------------------------------------------------- /browser/bin/docker-build.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | # seadfeng/playwright-browser-worker 4 | 5 | VERSION=$(node -p "require('./package.json').version") 6 | APP=$(node -p "require('./package.json').name") 7 | 8 | echo "Build: ${APP}.${VERSION}" 9 | 10 | docker buildx build \ 11 | --no-cache \ 12 | --build-arg VERSION=$VERSION \ 13 | -t seadfeng/playwright-${APP}:$VERSION \ 14 | -t seadfeng/playwright-${APP}:latest \ 15 | -f ./Dockerfile \ 16 | . 17 | 18 | echo "seadfeng/playwright-${APP}:${VERSION}" 19 | echo "seadfeng/playwright-${APP}:latest" -------------------------------------------------------------------------------- /browser/entrypoint.sh: -------------------------------------------------------------------------------- 1 | #!/bin/bash 2 | 3 | # xvfb-run node worker.js 4 | node worker.js 5 | -------------------------------------------------------------------------------- /browser/package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "browser-worker", 3 | "version": "1.0.3", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "browser-worker", 9 | "version": "1.0.3", 10 | "dependencies": { 11 | "ioredis": "^5.3.2", 12 | "playwright": "^1.50.0" 13 | } 14 | }, 15 | "node_modules/@ioredis/commands": { 16 | "version": "1.2.0", 17 | "resolved": "https://registry.npmjs.org/@ioredis/commands/-/commands-1.2.0.tgz", 18 | "integrity": "sha512-Sx1pU8EM64o2BrqNpEO1CNLtKQwyhuXuqyfH7oGKCk+1a33d2r5saW8zNwm3j6BTExtjrv2BxTgzzkMwts6vGg==" 19 | }, 20 | "node_modules/cluster-key-slot": { 21 | "version": "1.1.2", 22 | "resolved": "https://registry.npmjs.org/cluster-key-slot/-/cluster-key-slot-1.1.2.tgz", 23 | "integrity": "sha512-RMr0FhtfXemyinomL4hrWcYJxmX6deFdCxpJzhDttxgO1+bcCnkk+9drydLVDmAMG7NE6aN/fl4F7ucU/90gAA==", 24 | "engines": { 25 | "node": ">=0.10.0" 26 | } 27 | }, 28 | "node_modules/debug": { 29 | "version": "4.4.0", 30 | "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.0.tgz", 31 | "integrity": "sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA==", 32 | "dependencies": { 33 | "ms": "^2.1.3" 34 | }, 35 | "engines": { 36 | "node": ">=6.0" 37 | }, 38 | "peerDependenciesMeta": { 39 | "supports-color": { 40 | "optional": true 41 | } 42 | } 43 | }, 44 | "node_modules/denque": { 45 | "version": "2.1.0", 46 | "resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz", 47 | "integrity": "sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==", 48 | "engines": { 49 | "node": ">=0.10" 50 | } 51 | }, 52 | "node_modules/fsevents": { 53 | "version": "2.3.2", 54 | "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", 55 | "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", 56 | "hasInstallScript": true, 57 | "optional": true, 58 | "os": [ 59 | "darwin" 60 | ], 61 | "engines": { 62 | "node": "^8.16.0 || ^10.6.0 || >=11.0.0" 63 | } 64 | }, 65 | "node_modules/ioredis": { 66 | "version": "5.4.2", 67 | "resolved": "https://registry.npmjs.org/ioredis/-/ioredis-5.4.2.tgz", 68 | "integrity": "sha512-0SZXGNGZ+WzISQ67QDyZ2x0+wVxjjUndtD8oSeik/4ajifeiRufed8fCb8QW8VMyi4MXcS+UO1k/0NGhvq1PAg==", 69 | "dependencies": { 70 | "@ioredis/commands": "^1.1.1", 71 | "cluster-key-slot": "^1.1.0", 72 | "debug": "^4.3.4", 73 | "denque": "^2.1.0", 74 | "lodash.defaults": "^4.2.0", 75 | "lodash.isarguments": "^3.1.0", 76 | "redis-errors": "^1.2.0", 77 | "redis-parser": "^3.0.0", 78 | "standard-as-callback": "^2.1.0" 79 | }, 80 | "engines": { 81 | "node": ">=12.22.0" 82 | }, 83 | "funding": { 84 | "type": "opencollective", 85 | "url": "https://opencollective.com/ioredis" 86 | } 87 | }, 88 | "node_modules/lodash.defaults": { 89 | "version": "4.2.0", 90 | "resolved": "https://registry.npmjs.org/lodash.defaults/-/lodash.defaults-4.2.0.tgz", 91 | "integrity": "sha512-qjxPLHd3r5DnsdGacqOMU6pb/avJzdh9tFX2ymgoZE27BmjXrNy/y4LoaiTeAb+O3gL8AfpJGtqfX/ae2leYYQ==" 92 | }, 93 | "node_modules/lodash.isarguments": { 94 | "version": "3.1.0", 95 | "resolved": "https://registry.npmjs.org/lodash.isarguments/-/lodash.isarguments-3.1.0.tgz", 96 | "integrity": "sha512-chi4NHZlZqZD18a0imDHnZPrDeBbTtVN7GXMwuGdRH9qotxAjYs3aVLKc7zNOG9eddR5Ksd8rvFEBc9SsggPpg==" 97 | }, 98 | "node_modules/ms": { 99 | "version": "2.1.3", 100 | "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz", 101 | "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==" 102 | }, 103 | "node_modules/playwright": { 104 | "version": "1.50.1", 105 | "resolved": "https://registry.npmjs.org/playwright/-/playwright-1.50.1.tgz", 106 | "integrity": "sha512-G8rwsOQJ63XG6BbKj2w5rHeavFjy5zynBA9zsJMMtBoe/Uf757oG12NXz6e6OirF7RCrTVAKFXbLmn1RbL7Qaw==", 107 | "dependencies": { 108 | "playwright-core": "1.50.1" 109 | }, 110 | "bin": { 111 | "playwright": "cli.js" 112 | }, 113 | "engines": { 114 | "node": ">=18" 115 | }, 116 | "optionalDependencies": { 117 | "fsevents": "2.3.2" 118 | } 119 | }, 120 | "node_modules/playwright-core": { 121 | "version": "1.50.1", 122 | "resolved": "https://registry.npmjs.org/playwright-core/-/playwright-core-1.50.1.tgz", 123 | "integrity": "sha512-ra9fsNWayuYumt+NiM069M6OkcRb1FZSK8bgi66AtpFoWkg2+y0bJSNmkFrWhMbEBbVKC/EruAHH3g0zmtwGmQ==", 124 | "bin": { 125 | "playwright-core": "cli.js" 126 | }, 127 | "engines": { 128 | "node": ">=18" 129 | } 130 | }, 131 | "node_modules/redis-errors": { 132 | "version": "1.2.0", 133 | "resolved": "https://registry.npmjs.org/redis-errors/-/redis-errors-1.2.0.tgz", 134 | "integrity": "sha512-1qny3OExCf0UvUV/5wpYKf2YwPcOqXzkwKKSmKHiE6ZMQs5heeE/c8eXK+PNllPvmjgAbfnsbpkGZWy8cBpn9w==", 135 | "engines": { 136 | "node": ">=4" 137 | } 138 | }, 139 | "node_modules/redis-parser": { 140 | "version": "3.0.0", 141 | "resolved": "https://registry.npmjs.org/redis-parser/-/redis-parser-3.0.0.tgz", 142 | "integrity": "sha512-DJnGAeenTdpMEH6uAJRK/uiyEIH9WVsUmoLwzudwGJUwZPp80PDBWPHXSAGNPwNvIXAbe7MSUB1zQFugFml66A==", 143 | "dependencies": { 144 | "redis-errors": "^1.0.0" 145 | }, 146 | "engines": { 147 | "node": ">=4" 148 | } 149 | }, 150 | "node_modules/standard-as-callback": { 151 | "version": "2.1.0", 152 | "resolved": "https://registry.npmjs.org/standard-as-callback/-/standard-as-callback-2.1.0.tgz", 153 | "integrity": "sha512-qoRRSyROncaz1z0mvYqIE4lCd9p2R90i6GxW3uZv5ucSu8tU7B5HXUP1gG8pVZsYNVaXjk8ClXHPttLyxAL48A==" 154 | } 155 | } 156 | } 157 | -------------------------------------------------------------------------------- /browser/package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "browser-worker", 3 | "version": "1.0.3", 4 | "scripts": { 5 | "docker:build": "sh bin/docker-build.sh" 6 | }, 7 | "dependencies": { 8 | "ioredis": "^5.3.2", 9 | "playwright": "^1.50.0" 10 | } 11 | } -------------------------------------------------------------------------------- /browser/worker.js: -------------------------------------------------------------------------------- 1 | const { chromium } = require('playwright'); 2 | const Redis = require('ioredis'); 3 | const os = require('os'); 4 | 5 | // Initialize Redis connection 6 | const redis = new Redis(process.env.REDIS_URL); 7 | 8 | // Redis key constants 9 | const BROWSER_QUEUE = 'browser_queue'; 10 | const BROWSER_STATUS = 'browser_status'; 11 | 12 | // Get hostname, use environment variable or system hostname 13 | const hostname = process.env.HOST_NAME ?? os.hostname(); 14 | 15 | /** 16 | * Clean up browser-related resources in Redis 17 | * @param {string} browserId - Unique identifier for the browser 18 | */ 19 | async function cleanupBrowser(browserId) { 20 | console.log(`Cleaning up browser ${browserId}...`); 21 | try { 22 | // Remove all browser-related keys and entries from Redis 23 | await Promise.all([ 24 | redis.del(`ws:${browserId}`), // Remove WebSocket endpoint 25 | redis.lrem(BROWSER_QUEUE, 0, browserId), // Remove from browser queue 26 | redis.hdel(BROWSER_STATUS, browserId), // Remove status entry 27 | redis.del(`heartbeat:${browserId}`) // Remove heartbeat key 28 | ]); 29 | console.log(`Browser ${browserId} cleaned up successfully`); 30 | } catch (error) { 31 | console.error(`Cleanup failed for ${browserId}:`, error); 32 | } 33 | } 34 | 35 | /** 36 | * Start a Playwright browser worker 37 | * @returns {Object} Browser instance and related information 38 | */ 39 | async function startWorker() { 40 | // Launch Chromium browser server with specific configuration 41 | const browser = await chromium.launchServer({ 42 | args: [ 43 | '--no-sandbox', // Disable sandbox for container environments 44 | '--disable-setuid-sandbox', // Disable setuid sandbox 45 | '--disable-dev-shm-usage', // Overcome limited shared memory in containers 46 | '--disable-gpu', // Disable GPU hardware acceleration 47 | 48 | // Anti-detection settings 49 | '--disable-blink-features=AutomationControlled', // Hide automation flags 50 | // '--disable-features=IsolateOrigins,site-per-process', // Disable site isolation 51 | 52 | // Additional performance tweaks 53 | '--disable-web-security', // Disable web security for testing 54 | '--disable-notifications', // Disable browser notifications 55 | '--ignore-certificate-errors', // Ignore SSL/TLS errors 56 | '--window-size=1920,1080', // Set standard window size 57 | ], 58 | headless: true, // Run in headless mode 59 | handleSIGINT: false, // Manually handle process signals 60 | handleSIGTERM: false, 61 | handleSIGHUP: false, 62 | host: "0.0.0.0", // Bind to all network interfaces 63 | port: 8080 // Specific port for WebSocket 64 | }); 65 | 66 | // Get WebSocket endpoint 67 | const wsEndpoint = browser.wsEndpoint(); 68 | 69 | // Replace localhost with host name for external access 70 | const containerWsEndpoint = wsEndpoint.replace('0.0.0.0', process.env.HOST_NAME); 71 | 72 | // Generate unique browser ID 73 | const browserId = `browser_${hostname}`; 74 | console.log(`Browser ${browserId} started with endpoint ${containerWsEndpoint}`); 75 | 76 | // Initialize browser metadata in Redis 77 | await Promise.all([ 78 | redis.set(`ws:${browserId}`, containerWsEndpoint), // Store WebSocket endpoint 79 | redis.rpush(BROWSER_QUEUE, browserId), // Add to browser queue 80 | redis.hset(BROWSER_STATUS, browserId, 'idle'), // Set initial status 81 | redis.set(`heartbeat:${browserId}`, Date.now()) // Set initial heartbeat 82 | ]); 83 | 84 | // Periodic heartbeat to indicate browser is alive 85 | const heartbeatInterval = setInterval(async () => { 86 | await redis.set(`heartbeat:${browserId}`, Date.now()); 87 | }, 5000); 88 | 89 | // Cleanup function for graceful shutdown 90 | const cleanup = async () => { 91 | clearInterval(heartbeatInterval); // Stop heartbeat interval 92 | await cleanupBrowser(browserId); // Clean up Redis resources 93 | process.exit(0); // Exit process 94 | }; 95 | 96 | // Register signal handlers for graceful shutdown 97 | process.on('SIGTERM', cleanup); // Termination signal 98 | process.on('SIGINT', cleanup); // Interrupt signal (Ctrl+C) 99 | process.on('SIGQUIT', cleanup); // Quit signal 100 | 101 | return { browser, browserId, wsEndpoint }; 102 | } 103 | 104 | // Start the worker with error handling 105 | startWorker().catch(async error => { 106 | console.error('Worker failed:', error); 107 | 108 | // Attempt to clean up even if worker start fails 109 | const browserId = `browser_${hostname}`; 110 | await cleanupBrowser(browserId); 111 | 112 | process.exit(1); // Exit with error status 113 | }); 114 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "headless-browser-clusters", 3 | "version": "1.0.17", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "headless-browser-clusters", 9 | "version": "1.0.17", 10 | "devDependencies": { 11 | "standard-version": "^9.5.0" 12 | } 13 | }, 14 | "node_modules/@babel/code-frame": { 15 | "version": "7.26.2", 16 | "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.26.2.tgz", 17 | "integrity": "sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ==", 18 | "dev": true, 19 | "dependencies": { 20 | "@babel/helper-validator-identifier": "^7.25.9", 21 | "js-tokens": "^4.0.0", 22 | "picocolors": "^1.0.0" 23 | }, 24 | "engines": { 25 | "node": ">=6.9.0" 26 | } 27 | }, 28 | "node_modules/@babel/helper-validator-identifier": { 29 | "version": "7.25.9", 30 | "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz", 31 | "integrity": "sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ==", 32 | "dev": true, 33 | "engines": { 34 | "node": ">=6.9.0" 35 | } 36 | }, 37 | "node_modules/@hutson/parse-repository-url": { 38 | "version": "3.0.2", 39 | "resolved": "https://registry.npmjs.org/@hutson/parse-repository-url/-/parse-repository-url-3.0.2.tgz", 40 | "integrity": "sha512-H9XAx3hc0BQHY6l+IFSWHDySypcXsvsuLhgYLUGywmJ5pswRVQJUHpOsobnLYp2ZUaUlKiKDrgWWhosOwAEM8Q==", 41 | "dev": true, 42 | "engines": { 43 | "node": ">=6.9.0" 44 | } 45 | }, 46 | "node_modules/@types/minimist": { 47 | "version": "1.2.5", 48 | "resolved": "https://registry.npmjs.org/@types/minimist/-/minimist-1.2.5.tgz", 49 | "integrity": "sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag==", 50 | "dev": true 51 | }, 52 | "node_modules/@types/normalize-package-data": { 53 | "version": "2.4.4", 54 | "resolved": "https://registry.npmjs.org/@types/normalize-package-data/-/normalize-package-data-2.4.4.tgz", 55 | "integrity": "sha512-37i+OaWTh9qeK4LSHPsyRC7NahnGotNuZvjLSgcPzblpHB3rrCJxAOgI5gCdKm7coonsaX1Of0ILiTcnZjbfxA==", 56 | "dev": true 57 | }, 58 | "node_modules/add-stream": { 59 | "version": "1.0.0", 60 | "resolved": "https://registry.npmjs.org/add-stream/-/add-stream-1.0.0.tgz", 61 | "integrity": "sha512-qQLMr+8o0WC4FZGQTcJiKBVC59JylcPSrTtk6usvmIDFUOCKegapy1VHQwRbFMOFyb/inzUVqHs+eMYKDM1YeQ==", 62 | "dev": true 63 | }, 64 | "node_modules/ansi-regex": { 65 | "version": "5.0.1", 66 | "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.1.tgz", 67 | "integrity": "sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==", 68 | "dev": true, 69 | "engines": { 70 | "node": ">=8" 71 | } 72 | }, 73 | "node_modules/ansi-styles": { 74 | "version": "3.2.1", 75 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", 76 | "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", 77 | "dev": true, 78 | "dependencies": { 79 | "color-convert": "^1.9.0" 80 | }, 81 | "engines": { 82 | "node": ">=4" 83 | } 84 | }, 85 | "node_modules/array-ify": { 86 | "version": "1.0.0", 87 | "resolved": "https://registry.npmjs.org/array-ify/-/array-ify-1.0.0.tgz", 88 | "integrity": "sha512-c5AMf34bKdvPhQ7tBGhqkgKNUzMr4WUs+WDtC2ZUGOUncbxKMTvqxYctiseW3+L4bA8ec+GcZ6/A/FW4m8ukng==", 89 | "dev": true 90 | }, 91 | "node_modules/arrify": { 92 | "version": "1.0.1", 93 | "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", 94 | "integrity": "sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA==", 95 | "dev": true, 96 | "engines": { 97 | "node": ">=0.10.0" 98 | } 99 | }, 100 | "node_modules/balanced-match": { 101 | "version": "1.0.2", 102 | "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz", 103 | "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==", 104 | "dev": true 105 | }, 106 | "node_modules/brace-expansion": { 107 | "version": "1.1.11", 108 | "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", 109 | "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", 110 | "dev": true, 111 | "dependencies": { 112 | "balanced-match": "^1.0.0", 113 | "concat-map": "0.0.1" 114 | } 115 | }, 116 | "node_modules/buffer-from": { 117 | "version": "1.1.2", 118 | "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.2.tgz", 119 | "integrity": "sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==", 120 | "dev": true 121 | }, 122 | "node_modules/camelcase": { 123 | "version": "5.3.1", 124 | "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", 125 | "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", 126 | "dev": true, 127 | "engines": { 128 | "node": ">=6" 129 | } 130 | }, 131 | "node_modules/camelcase-keys": { 132 | "version": "6.2.2", 133 | "resolved": "https://registry.npmjs.org/camelcase-keys/-/camelcase-keys-6.2.2.tgz", 134 | "integrity": "sha512-YrwaA0vEKazPBkn0ipTiMpSajYDSe+KjQfrjhcBMxJt/znbvlHd8Pw/Vamaz5EB4Wfhs3SUR3Z9mwRu/P3s3Yg==", 135 | "dev": true, 136 | "dependencies": { 137 | "camelcase": "^5.3.1", 138 | "map-obj": "^4.0.0", 139 | "quick-lru": "^4.0.1" 140 | }, 141 | "engines": { 142 | "node": ">=8" 143 | }, 144 | "funding": { 145 | "url": "https://github.com/sponsors/sindresorhus" 146 | } 147 | }, 148 | "node_modules/chalk": { 149 | "version": "2.4.2", 150 | "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", 151 | "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", 152 | "dev": true, 153 | "dependencies": { 154 | "ansi-styles": "^3.2.1", 155 | "escape-string-regexp": "^1.0.5", 156 | "supports-color": "^5.3.0" 157 | }, 158 | "engines": { 159 | "node": ">=4" 160 | } 161 | }, 162 | "node_modules/cliui": { 163 | "version": "7.0.4", 164 | "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", 165 | "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", 166 | "dev": true, 167 | "dependencies": { 168 | "string-width": "^4.2.0", 169 | "strip-ansi": "^6.0.0", 170 | "wrap-ansi": "^7.0.0" 171 | } 172 | }, 173 | "node_modules/color-convert": { 174 | "version": "1.9.3", 175 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", 176 | "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", 177 | "dev": true, 178 | "dependencies": { 179 | "color-name": "1.1.3" 180 | } 181 | }, 182 | "node_modules/color-name": { 183 | "version": "1.1.3", 184 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", 185 | "integrity": "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==", 186 | "dev": true 187 | }, 188 | "node_modules/compare-func": { 189 | "version": "2.0.0", 190 | "resolved": "https://registry.npmjs.org/compare-func/-/compare-func-2.0.0.tgz", 191 | "integrity": "sha512-zHig5N+tPWARooBnb0Zx1MFcdfpyJrfTJ3Y5L+IFvUm8rM74hHz66z0gw0x4tijh5CorKkKUCnW82R2vmpeCRA==", 192 | "dev": true, 193 | "dependencies": { 194 | "array-ify": "^1.0.0", 195 | "dot-prop": "^5.1.0" 196 | } 197 | }, 198 | "node_modules/concat-map": { 199 | "version": "0.0.1", 200 | "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", 201 | "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==", 202 | "dev": true 203 | }, 204 | "node_modules/concat-stream": { 205 | "version": "2.0.0", 206 | "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-2.0.0.tgz", 207 | "integrity": "sha512-MWufYdFw53ccGjCA+Ol7XJYpAlW6/prSMzuPOTRnJGcGzuhLn4Scrz7qf6o8bROZ514ltazcIFJZevcfbo0x7A==", 208 | "dev": true, 209 | "engines": [ 210 | "node >= 6.0" 211 | ], 212 | "dependencies": { 213 | "buffer-from": "^1.0.0", 214 | "inherits": "^2.0.3", 215 | "readable-stream": "^3.0.2", 216 | "typedarray": "^0.0.6" 217 | } 218 | }, 219 | "node_modules/conventional-changelog": { 220 | "version": "3.1.25", 221 | "resolved": "https://registry.npmjs.org/conventional-changelog/-/conventional-changelog-3.1.25.tgz", 222 | "integrity": "sha512-ryhi3fd1mKf3fSjbLXOfK2D06YwKNic1nC9mWqybBHdObPd8KJ2vjaXZfYj1U23t+V8T8n0d7gwnc9XbIdFbyQ==", 223 | "dev": true, 224 | "dependencies": { 225 | "conventional-changelog-angular": "^5.0.12", 226 | "conventional-changelog-atom": "^2.0.8", 227 | "conventional-changelog-codemirror": "^2.0.8", 228 | "conventional-changelog-conventionalcommits": "^4.5.0", 229 | "conventional-changelog-core": "^4.2.1", 230 | "conventional-changelog-ember": "^2.0.9", 231 | "conventional-changelog-eslint": "^3.0.9", 232 | "conventional-changelog-express": "^2.0.6", 233 | "conventional-changelog-jquery": "^3.0.11", 234 | "conventional-changelog-jshint": "^2.0.9", 235 | "conventional-changelog-preset-loader": "^2.3.4" 236 | }, 237 | "engines": { 238 | "node": ">=10" 239 | } 240 | }, 241 | "node_modules/conventional-changelog-angular": { 242 | "version": "5.0.13", 243 | "resolved": "https://registry.npmjs.org/conventional-changelog-angular/-/conventional-changelog-angular-5.0.13.tgz", 244 | "integrity": "sha512-i/gipMxs7s8L/QeuavPF2hLnJgH6pEZAttySB6aiQLWcX3puWDL3ACVmvBhJGxnAy52Qc15ua26BufY6KpmrVA==", 245 | "dev": true, 246 | "dependencies": { 247 | "compare-func": "^2.0.0", 248 | "q": "^1.5.1" 249 | }, 250 | "engines": { 251 | "node": ">=10" 252 | } 253 | }, 254 | "node_modules/conventional-changelog-atom": { 255 | "version": "2.0.8", 256 | "resolved": "https://registry.npmjs.org/conventional-changelog-atom/-/conventional-changelog-atom-2.0.8.tgz", 257 | "integrity": "sha512-xo6v46icsFTK3bb7dY/8m2qvc8sZemRgdqLb/bjpBsH2UyOS8rKNTgcb5025Hri6IpANPApbXMg15QLb1LJpBw==", 258 | "dev": true, 259 | "dependencies": { 260 | "q": "^1.5.1" 261 | }, 262 | "engines": { 263 | "node": ">=10" 264 | } 265 | }, 266 | "node_modules/conventional-changelog-codemirror": { 267 | "version": "2.0.8", 268 | "resolved": "https://registry.npmjs.org/conventional-changelog-codemirror/-/conventional-changelog-codemirror-2.0.8.tgz", 269 | "integrity": "sha512-z5DAsn3uj1Vfp7po3gpt2Boc+Bdwmw2++ZHa5Ak9k0UKsYAO5mH1UBTN0qSCuJZREIhX6WU4E1p3IW2oRCNzQw==", 270 | "dev": true, 271 | "dependencies": { 272 | "q": "^1.5.1" 273 | }, 274 | "engines": { 275 | "node": ">=10" 276 | } 277 | }, 278 | "node_modules/conventional-changelog-config-spec": { 279 | "version": "2.1.0", 280 | "resolved": "https://registry.npmjs.org/conventional-changelog-config-spec/-/conventional-changelog-config-spec-2.1.0.tgz", 281 | "integrity": "sha512-IpVePh16EbbB02V+UA+HQnnPIohgXvJRxHcS5+Uwk4AT5LjzCZJm5sp/yqs5C6KZJ1jMsV4paEV13BN1pvDuxQ==", 282 | "dev": true 283 | }, 284 | "node_modules/conventional-changelog-conventionalcommits": { 285 | "version": "4.6.3", 286 | "resolved": "https://registry.npmjs.org/conventional-changelog-conventionalcommits/-/conventional-changelog-conventionalcommits-4.6.3.tgz", 287 | "integrity": "sha512-LTTQV4fwOM4oLPad317V/QNQ1FY4Hju5qeBIM1uTHbrnCE+Eg4CdRZ3gO2pUeR+tzWdp80M2j3qFFEDWVqOV4g==", 288 | "dev": true, 289 | "dependencies": { 290 | "compare-func": "^2.0.0", 291 | "lodash": "^4.17.15", 292 | "q": "^1.5.1" 293 | }, 294 | "engines": { 295 | "node": ">=10" 296 | } 297 | }, 298 | "node_modules/conventional-changelog-core": { 299 | "version": "4.2.4", 300 | "resolved": "https://registry.npmjs.org/conventional-changelog-core/-/conventional-changelog-core-4.2.4.tgz", 301 | "integrity": "sha512-gDVS+zVJHE2v4SLc6B0sLsPiloR0ygU7HaDW14aNJE1v4SlqJPILPl/aJC7YdtRE4CybBf8gDwObBvKha8Xlyg==", 302 | "dev": true, 303 | "dependencies": { 304 | "add-stream": "^1.0.0", 305 | "conventional-changelog-writer": "^5.0.0", 306 | "conventional-commits-parser": "^3.2.0", 307 | "dateformat": "^3.0.0", 308 | "get-pkg-repo": "^4.0.0", 309 | "git-raw-commits": "^2.0.8", 310 | "git-remote-origin-url": "^2.0.0", 311 | "git-semver-tags": "^4.1.1", 312 | "lodash": "^4.17.15", 313 | "normalize-package-data": "^3.0.0", 314 | "q": "^1.5.1", 315 | "read-pkg": "^3.0.0", 316 | "read-pkg-up": "^3.0.0", 317 | "through2": "^4.0.0" 318 | }, 319 | "engines": { 320 | "node": ">=10" 321 | } 322 | }, 323 | "node_modules/conventional-changelog-ember": { 324 | "version": "2.0.9", 325 | "resolved": "https://registry.npmjs.org/conventional-changelog-ember/-/conventional-changelog-ember-2.0.9.tgz", 326 | "integrity": "sha512-ulzIReoZEvZCBDhcNYfDIsLTHzYHc7awh+eI44ZtV5cx6LVxLlVtEmcO+2/kGIHGtw+qVabJYjdI5cJOQgXh1A==", 327 | "dev": true, 328 | "dependencies": { 329 | "q": "^1.5.1" 330 | }, 331 | "engines": { 332 | "node": ">=10" 333 | } 334 | }, 335 | "node_modules/conventional-changelog-eslint": { 336 | "version": "3.0.9", 337 | "resolved": "https://registry.npmjs.org/conventional-changelog-eslint/-/conventional-changelog-eslint-3.0.9.tgz", 338 | "integrity": "sha512-6NpUCMgU8qmWmyAMSZO5NrRd7rTgErjrm4VASam2u5jrZS0n38V7Y9CzTtLT2qwz5xEChDR4BduoWIr8TfwvXA==", 339 | "dev": true, 340 | "dependencies": { 341 | "q": "^1.5.1" 342 | }, 343 | "engines": { 344 | "node": ">=10" 345 | } 346 | }, 347 | "node_modules/conventional-changelog-express": { 348 | "version": "2.0.6", 349 | "resolved": "https://registry.npmjs.org/conventional-changelog-express/-/conventional-changelog-express-2.0.6.tgz", 350 | "integrity": "sha512-SDez2f3iVJw6V563O3pRtNwXtQaSmEfTCaTBPCqn0oG0mfkq0rX4hHBq5P7De2MncoRixrALj3u3oQsNK+Q0pQ==", 351 | "dev": true, 352 | "dependencies": { 353 | "q": "^1.5.1" 354 | }, 355 | "engines": { 356 | "node": ">=10" 357 | } 358 | }, 359 | "node_modules/conventional-changelog-jquery": { 360 | "version": "3.0.11", 361 | "resolved": "https://registry.npmjs.org/conventional-changelog-jquery/-/conventional-changelog-jquery-3.0.11.tgz", 362 | "integrity": "sha512-x8AWz5/Td55F7+o/9LQ6cQIPwrCjfJQ5Zmfqi8thwUEKHstEn4kTIofXub7plf1xvFA2TqhZlq7fy5OmV6BOMw==", 363 | "dev": true, 364 | "dependencies": { 365 | "q": "^1.5.1" 366 | }, 367 | "engines": { 368 | "node": ">=10" 369 | } 370 | }, 371 | "node_modules/conventional-changelog-jshint": { 372 | "version": "2.0.9", 373 | "resolved": "https://registry.npmjs.org/conventional-changelog-jshint/-/conventional-changelog-jshint-2.0.9.tgz", 374 | "integrity": "sha512-wMLdaIzq6TNnMHMy31hql02OEQ8nCQfExw1SE0hYL5KvU+JCTuPaDO+7JiogGT2gJAxiUGATdtYYfh+nT+6riA==", 375 | "dev": true, 376 | "dependencies": { 377 | "compare-func": "^2.0.0", 378 | "q": "^1.5.1" 379 | }, 380 | "engines": { 381 | "node": ">=10" 382 | } 383 | }, 384 | "node_modules/conventional-changelog-preset-loader": { 385 | "version": "2.3.4", 386 | "resolved": "https://registry.npmjs.org/conventional-changelog-preset-loader/-/conventional-changelog-preset-loader-2.3.4.tgz", 387 | "integrity": "sha512-GEKRWkrSAZeTq5+YjUZOYxdHq+ci4dNwHvpaBC3+ENalzFWuCWa9EZXSuZBpkr72sMdKB+1fyDV4takK1Lf58g==", 388 | "dev": true, 389 | "engines": { 390 | "node": ">=10" 391 | } 392 | }, 393 | "node_modules/conventional-changelog-writer": { 394 | "version": "5.0.1", 395 | "resolved": "https://registry.npmjs.org/conventional-changelog-writer/-/conventional-changelog-writer-5.0.1.tgz", 396 | "integrity": "sha512-5WsuKUfxW7suLblAbFnxAcrvf6r+0b7GvNaWUwUIk0bXMnENP/PEieGKVUQrjPqwPT4o3EPAASBXiY6iHooLOQ==", 397 | "dev": true, 398 | "dependencies": { 399 | "conventional-commits-filter": "^2.0.7", 400 | "dateformat": "^3.0.0", 401 | "handlebars": "^4.7.7", 402 | "json-stringify-safe": "^5.0.1", 403 | "lodash": "^4.17.15", 404 | "meow": "^8.0.0", 405 | "semver": "^6.0.0", 406 | "split": "^1.0.0", 407 | "through2": "^4.0.0" 408 | }, 409 | "bin": { 410 | "conventional-changelog-writer": "cli.js" 411 | }, 412 | "engines": { 413 | "node": ">=10" 414 | } 415 | }, 416 | "node_modules/conventional-changelog-writer/node_modules/semver": { 417 | "version": "6.3.1", 418 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", 419 | "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", 420 | "dev": true, 421 | "bin": { 422 | "semver": "bin/semver.js" 423 | } 424 | }, 425 | "node_modules/conventional-commits-filter": { 426 | "version": "2.0.7", 427 | "resolved": "https://registry.npmjs.org/conventional-commits-filter/-/conventional-commits-filter-2.0.7.tgz", 428 | "integrity": "sha512-ASS9SamOP4TbCClsRHxIHXRfcGCnIoQqkvAzCSbZzTFLfcTqJVugB0agRgsEELsqaeWgsXv513eS116wnlSSPA==", 429 | "dev": true, 430 | "dependencies": { 431 | "lodash.ismatch": "^4.4.0", 432 | "modify-values": "^1.0.0" 433 | }, 434 | "engines": { 435 | "node": ">=10" 436 | } 437 | }, 438 | "node_modules/conventional-commits-parser": { 439 | "version": "3.2.4", 440 | "resolved": "https://registry.npmjs.org/conventional-commits-parser/-/conventional-commits-parser-3.2.4.tgz", 441 | "integrity": "sha512-nK7sAtfi+QXbxHCYfhpZsfRtaitZLIA6889kFIouLvz6repszQDgxBu7wf2WbU+Dco7sAnNCJYERCwt54WPC2Q==", 442 | "dev": true, 443 | "dependencies": { 444 | "is-text-path": "^1.0.1", 445 | "JSONStream": "^1.0.4", 446 | "lodash": "^4.17.15", 447 | "meow": "^8.0.0", 448 | "split2": "^3.0.0", 449 | "through2": "^4.0.0" 450 | }, 451 | "bin": { 452 | "conventional-commits-parser": "cli.js" 453 | }, 454 | "engines": { 455 | "node": ">=10" 456 | } 457 | }, 458 | "node_modules/conventional-recommended-bump": { 459 | "version": "6.1.0", 460 | "resolved": "https://registry.npmjs.org/conventional-recommended-bump/-/conventional-recommended-bump-6.1.0.tgz", 461 | "integrity": "sha512-uiApbSiNGM/kkdL9GTOLAqC4hbptObFo4wW2QRyHsKciGAfQuLU1ShZ1BIVI/+K2BE/W1AWYQMCXAsv4dyKPaw==", 462 | "dev": true, 463 | "dependencies": { 464 | "concat-stream": "^2.0.0", 465 | "conventional-changelog-preset-loader": "^2.3.4", 466 | "conventional-commits-filter": "^2.0.7", 467 | "conventional-commits-parser": "^3.2.0", 468 | "git-raw-commits": "^2.0.8", 469 | "git-semver-tags": "^4.1.1", 470 | "meow": "^8.0.0", 471 | "q": "^1.5.1" 472 | }, 473 | "bin": { 474 | "conventional-recommended-bump": "cli.js" 475 | }, 476 | "engines": { 477 | "node": ">=10" 478 | } 479 | }, 480 | "node_modules/core-util-is": { 481 | "version": "1.0.3", 482 | "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.3.tgz", 483 | "integrity": "sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==", 484 | "dev": true 485 | }, 486 | "node_modules/dargs": { 487 | "version": "7.0.0", 488 | "resolved": "https://registry.npmjs.org/dargs/-/dargs-7.0.0.tgz", 489 | "integrity": "sha512-2iy1EkLdlBzQGvbweYRFxmFath8+K7+AKB0TlhHWkNuH+TmovaMH/Wp7V7R4u7f4SnX3OgLsU9t1NI9ioDnUpg==", 490 | "dev": true, 491 | "engines": { 492 | "node": ">=8" 493 | } 494 | }, 495 | "node_modules/dateformat": { 496 | "version": "3.0.3", 497 | "resolved": "https://registry.npmjs.org/dateformat/-/dateformat-3.0.3.tgz", 498 | "integrity": "sha512-jyCETtSl3VMZMWeRo7iY1FL19ges1t55hMo5yaam4Jrsm5EPL89UQkoQRyiI+Yf4k8r2ZpdngkV8hr1lIdjb3Q==", 499 | "dev": true, 500 | "engines": { 501 | "node": "*" 502 | } 503 | }, 504 | "node_modules/decamelize": { 505 | "version": "1.2.0", 506 | "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", 507 | "integrity": "sha512-z2S+W9X73hAUUki+N+9Za2lBlun89zigOyGrsax+KUQ6wKW4ZoWpEYBkGhQjwAjjDCkWxhY0VKEhk8wzY7F5cA==", 508 | "dev": true, 509 | "engines": { 510 | "node": ">=0.10.0" 511 | } 512 | }, 513 | "node_modules/decamelize-keys": { 514 | "version": "1.1.1", 515 | "resolved": "https://registry.npmjs.org/decamelize-keys/-/decamelize-keys-1.1.1.tgz", 516 | "integrity": "sha512-WiPxgEirIV0/eIOMcnFBA3/IJZAZqKnwAwWyvvdi4lsr1WCN22nhdf/3db3DoZcUjTV2SqfzIwNyp6y2xs3nmg==", 517 | "dev": true, 518 | "dependencies": { 519 | "decamelize": "^1.1.0", 520 | "map-obj": "^1.0.0" 521 | }, 522 | "engines": { 523 | "node": ">=0.10.0" 524 | }, 525 | "funding": { 526 | "url": "https://github.com/sponsors/sindresorhus" 527 | } 528 | }, 529 | "node_modules/decamelize-keys/node_modules/map-obj": { 530 | "version": "1.0.1", 531 | "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-1.0.1.tgz", 532 | "integrity": "sha512-7N/q3lyZ+LVCp7PzuxrJr4KMbBE2hW7BT7YNia330OFxIf4d3r5zVpicP2650l7CPN6RM9zOJRl3NGpqSiw3Eg==", 533 | "dev": true, 534 | "engines": { 535 | "node": ">=0.10.0" 536 | } 537 | }, 538 | "node_modules/detect-indent": { 539 | "version": "6.1.0", 540 | "resolved": "https://registry.npmjs.org/detect-indent/-/detect-indent-6.1.0.tgz", 541 | "integrity": "sha512-reYkTUJAZb9gUuZ2RvVCNhVHdg62RHnJ7WJl8ftMi4diZ6NWlciOzQN88pUhSELEwflJht4oQDv0F0BMlwaYtA==", 542 | "dev": true, 543 | "engines": { 544 | "node": ">=8" 545 | } 546 | }, 547 | "node_modules/detect-newline": { 548 | "version": "3.1.0", 549 | "resolved": "https://registry.npmjs.org/detect-newline/-/detect-newline-3.1.0.tgz", 550 | "integrity": "sha512-TLz+x/vEXm/Y7P7wn1EJFNLxYpUD4TgMosxY6fAVJUnJMbupHBOncxyWUG9OpTaH9EBD7uFI5LfEgmMOc54DsA==", 551 | "dev": true, 552 | "engines": { 553 | "node": ">=8" 554 | } 555 | }, 556 | "node_modules/dot-prop": { 557 | "version": "5.3.0", 558 | "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", 559 | "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", 560 | "dev": true, 561 | "dependencies": { 562 | "is-obj": "^2.0.0" 563 | }, 564 | "engines": { 565 | "node": ">=8" 566 | } 567 | }, 568 | "node_modules/dotgitignore": { 569 | "version": "2.1.0", 570 | "resolved": "https://registry.npmjs.org/dotgitignore/-/dotgitignore-2.1.0.tgz", 571 | "integrity": "sha512-sCm11ak2oY6DglEPpCB8TixLjWAxd3kJTs6UIcSasNYxXdFPV+YKlye92c8H4kKFqV5qYMIh7d+cYecEg0dIkA==", 572 | "dev": true, 573 | "dependencies": { 574 | "find-up": "^3.0.0", 575 | "minimatch": "^3.0.4" 576 | }, 577 | "engines": { 578 | "node": ">=6" 579 | } 580 | }, 581 | "node_modules/dotgitignore/node_modules/find-up": { 582 | "version": "3.0.0", 583 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", 584 | "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", 585 | "dev": true, 586 | "dependencies": { 587 | "locate-path": "^3.0.0" 588 | }, 589 | "engines": { 590 | "node": ">=6" 591 | } 592 | }, 593 | "node_modules/dotgitignore/node_modules/locate-path": { 594 | "version": "3.0.0", 595 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", 596 | "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", 597 | "dev": true, 598 | "dependencies": { 599 | "p-locate": "^3.0.0", 600 | "path-exists": "^3.0.0" 601 | }, 602 | "engines": { 603 | "node": ">=6" 604 | } 605 | }, 606 | "node_modules/dotgitignore/node_modules/p-limit": { 607 | "version": "2.3.0", 608 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 609 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 610 | "dev": true, 611 | "dependencies": { 612 | "p-try": "^2.0.0" 613 | }, 614 | "engines": { 615 | "node": ">=6" 616 | }, 617 | "funding": { 618 | "url": "https://github.com/sponsors/sindresorhus" 619 | } 620 | }, 621 | "node_modules/dotgitignore/node_modules/p-locate": { 622 | "version": "3.0.0", 623 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", 624 | "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", 625 | "dev": true, 626 | "dependencies": { 627 | "p-limit": "^2.0.0" 628 | }, 629 | "engines": { 630 | "node": ">=6" 631 | } 632 | }, 633 | "node_modules/dotgitignore/node_modules/path-exists": { 634 | "version": "3.0.0", 635 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 636 | "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", 637 | "dev": true, 638 | "engines": { 639 | "node": ">=4" 640 | } 641 | }, 642 | "node_modules/emoji-regex": { 643 | "version": "8.0.0", 644 | "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", 645 | "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", 646 | "dev": true 647 | }, 648 | "node_modules/error-ex": { 649 | "version": "1.3.2", 650 | "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", 651 | "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", 652 | "dev": true, 653 | "dependencies": { 654 | "is-arrayish": "^0.2.1" 655 | } 656 | }, 657 | "node_modules/escalade": { 658 | "version": "3.2.0", 659 | "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz", 660 | "integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==", 661 | "dev": true, 662 | "engines": { 663 | "node": ">=6" 664 | } 665 | }, 666 | "node_modules/escape-string-regexp": { 667 | "version": "1.0.5", 668 | "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", 669 | "integrity": "sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==", 670 | "dev": true, 671 | "engines": { 672 | "node": ">=0.8.0" 673 | } 674 | }, 675 | "node_modules/figures": { 676 | "version": "3.2.0", 677 | "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", 678 | "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", 679 | "dev": true, 680 | "dependencies": { 681 | "escape-string-regexp": "^1.0.5" 682 | }, 683 | "engines": { 684 | "node": ">=8" 685 | }, 686 | "funding": { 687 | "url": "https://github.com/sponsors/sindresorhus" 688 | } 689 | }, 690 | "node_modules/find-up": { 691 | "version": "5.0.0", 692 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-5.0.0.tgz", 693 | "integrity": "sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==", 694 | "dev": true, 695 | "dependencies": { 696 | "locate-path": "^6.0.0", 697 | "path-exists": "^4.0.0" 698 | }, 699 | "engines": { 700 | "node": ">=10" 701 | }, 702 | "funding": { 703 | "url": "https://github.com/sponsors/sindresorhus" 704 | } 705 | }, 706 | "node_modules/function-bind": { 707 | "version": "1.1.2", 708 | "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", 709 | "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", 710 | "dev": true, 711 | "funding": { 712 | "url": "https://github.com/sponsors/ljharb" 713 | } 714 | }, 715 | "node_modules/get-caller-file": { 716 | "version": "2.0.5", 717 | "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", 718 | "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", 719 | "dev": true, 720 | "engines": { 721 | "node": "6.* || 8.* || >= 10.*" 722 | } 723 | }, 724 | "node_modules/get-pkg-repo": { 725 | "version": "4.2.1", 726 | "resolved": "https://registry.npmjs.org/get-pkg-repo/-/get-pkg-repo-4.2.1.tgz", 727 | "integrity": "sha512-2+QbHjFRfGB74v/pYWjd5OhU3TDIC2Gv/YKUTk/tCvAz0pkn/Mz6P3uByuBimLOcPvN2jYdScl3xGFSrx0jEcA==", 728 | "dev": true, 729 | "dependencies": { 730 | "@hutson/parse-repository-url": "^3.0.0", 731 | "hosted-git-info": "^4.0.0", 732 | "through2": "^2.0.0", 733 | "yargs": "^16.2.0" 734 | }, 735 | "bin": { 736 | "get-pkg-repo": "src/cli.js" 737 | }, 738 | "engines": { 739 | "node": ">=6.9.0" 740 | } 741 | }, 742 | "node_modules/get-pkg-repo/node_modules/readable-stream": { 743 | "version": "2.3.8", 744 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.8.tgz", 745 | "integrity": "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==", 746 | "dev": true, 747 | "dependencies": { 748 | "core-util-is": "~1.0.0", 749 | "inherits": "~2.0.3", 750 | "isarray": "~1.0.0", 751 | "process-nextick-args": "~2.0.0", 752 | "safe-buffer": "~5.1.1", 753 | "string_decoder": "~1.1.1", 754 | "util-deprecate": "~1.0.1" 755 | } 756 | }, 757 | "node_modules/get-pkg-repo/node_modules/safe-buffer": { 758 | "version": "5.1.2", 759 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", 760 | "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", 761 | "dev": true 762 | }, 763 | "node_modules/get-pkg-repo/node_modules/string_decoder": { 764 | "version": "1.1.1", 765 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", 766 | "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", 767 | "dev": true, 768 | "dependencies": { 769 | "safe-buffer": "~5.1.0" 770 | } 771 | }, 772 | "node_modules/get-pkg-repo/node_modules/through2": { 773 | "version": "2.0.5", 774 | "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", 775 | "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", 776 | "dev": true, 777 | "dependencies": { 778 | "readable-stream": "~2.3.6", 779 | "xtend": "~4.0.1" 780 | } 781 | }, 782 | "node_modules/git-raw-commits": { 783 | "version": "2.0.11", 784 | "resolved": "https://registry.npmjs.org/git-raw-commits/-/git-raw-commits-2.0.11.tgz", 785 | "integrity": "sha512-VnctFhw+xfj8Va1xtfEqCUD2XDrbAPSJx+hSrE5K7fGdjZruW7XV+QOrN7LF/RJyvspRiD2I0asWsxFp0ya26A==", 786 | "dev": true, 787 | "dependencies": { 788 | "dargs": "^7.0.0", 789 | "lodash": "^4.17.15", 790 | "meow": "^8.0.0", 791 | "split2": "^3.0.0", 792 | "through2": "^4.0.0" 793 | }, 794 | "bin": { 795 | "git-raw-commits": "cli.js" 796 | }, 797 | "engines": { 798 | "node": ">=10" 799 | } 800 | }, 801 | "node_modules/git-remote-origin-url": { 802 | "version": "2.0.0", 803 | "resolved": "https://registry.npmjs.org/git-remote-origin-url/-/git-remote-origin-url-2.0.0.tgz", 804 | "integrity": "sha512-eU+GGrZgccNJcsDH5LkXR3PB9M958hxc7sbA8DFJjrv9j4L2P/eZfKhM+QD6wyzpiv+b1BpK0XrYCxkovtjSLw==", 805 | "dev": true, 806 | "dependencies": { 807 | "gitconfiglocal": "^1.0.0", 808 | "pify": "^2.3.0" 809 | }, 810 | "engines": { 811 | "node": ">=4" 812 | } 813 | }, 814 | "node_modules/git-semver-tags": { 815 | "version": "4.1.1", 816 | "resolved": "https://registry.npmjs.org/git-semver-tags/-/git-semver-tags-4.1.1.tgz", 817 | "integrity": "sha512-OWyMt5zBe7xFs8vglMmhM9lRQzCWL3WjHtxNNfJTMngGym7pC1kh8sP6jevfydJ6LP3ZvGxfb6ABYgPUM0mtsA==", 818 | "dev": true, 819 | "dependencies": { 820 | "meow": "^8.0.0", 821 | "semver": "^6.0.0" 822 | }, 823 | "bin": { 824 | "git-semver-tags": "cli.js" 825 | }, 826 | "engines": { 827 | "node": ">=10" 828 | } 829 | }, 830 | "node_modules/git-semver-tags/node_modules/semver": { 831 | "version": "6.3.1", 832 | "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz", 833 | "integrity": "sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA==", 834 | "dev": true, 835 | "bin": { 836 | "semver": "bin/semver.js" 837 | } 838 | }, 839 | "node_modules/gitconfiglocal": { 840 | "version": "1.0.0", 841 | "resolved": "https://registry.npmjs.org/gitconfiglocal/-/gitconfiglocal-1.0.0.tgz", 842 | "integrity": "sha512-spLUXeTAVHxDtKsJc8FkFVgFtMdEN9qPGpL23VfSHx4fP4+Ds097IXLvymbnDH8FnmxX5Nr9bPw3A+AQ6mWEaQ==", 843 | "dev": true, 844 | "dependencies": { 845 | "ini": "^1.3.2" 846 | } 847 | }, 848 | "node_modules/graceful-fs": { 849 | "version": "4.2.11", 850 | "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", 851 | "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", 852 | "dev": true 853 | }, 854 | "node_modules/handlebars": { 855 | "version": "4.7.8", 856 | "resolved": "https://registry.npmjs.org/handlebars/-/handlebars-4.7.8.tgz", 857 | "integrity": "sha512-vafaFqs8MZkRrSX7sFVUdo3ap/eNiLnb4IakshzvP56X5Nr1iGKAIqdX6tMlm6HcNRIkr6AxO5jFEoJzzpT8aQ==", 858 | "dev": true, 859 | "dependencies": { 860 | "minimist": "^1.2.5", 861 | "neo-async": "^2.6.2", 862 | "source-map": "^0.6.1", 863 | "wordwrap": "^1.0.0" 864 | }, 865 | "bin": { 866 | "handlebars": "bin/handlebars" 867 | }, 868 | "engines": { 869 | "node": ">=0.4.7" 870 | }, 871 | "optionalDependencies": { 872 | "uglify-js": "^3.1.4" 873 | } 874 | }, 875 | "node_modules/hard-rejection": { 876 | "version": "2.1.0", 877 | "resolved": "https://registry.npmjs.org/hard-rejection/-/hard-rejection-2.1.0.tgz", 878 | "integrity": "sha512-VIZB+ibDhx7ObhAe7OVtoEbuP4h/MuOTHJ+J8h/eBXotJYl0fBgR72xDFCKgIh22OJZIOVNxBMWuhAr10r8HdA==", 879 | "dev": true, 880 | "engines": { 881 | "node": ">=6" 882 | } 883 | }, 884 | "node_modules/has-flag": { 885 | "version": "3.0.0", 886 | "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", 887 | "integrity": "sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==", 888 | "dev": true, 889 | "engines": { 890 | "node": ">=4" 891 | } 892 | }, 893 | "node_modules/hasown": { 894 | "version": "2.0.2", 895 | "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz", 896 | "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==", 897 | "dev": true, 898 | "dependencies": { 899 | "function-bind": "^1.1.2" 900 | }, 901 | "engines": { 902 | "node": ">= 0.4" 903 | } 904 | }, 905 | "node_modules/hosted-git-info": { 906 | "version": "4.1.0", 907 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.1.0.tgz", 908 | "integrity": "sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA==", 909 | "dev": true, 910 | "dependencies": { 911 | "lru-cache": "^6.0.0" 912 | }, 913 | "engines": { 914 | "node": ">=10" 915 | } 916 | }, 917 | "node_modules/indent-string": { 918 | "version": "4.0.0", 919 | "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", 920 | "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", 921 | "dev": true, 922 | "engines": { 923 | "node": ">=8" 924 | } 925 | }, 926 | "node_modules/inherits": { 927 | "version": "2.0.4", 928 | "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", 929 | "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", 930 | "dev": true 931 | }, 932 | "node_modules/ini": { 933 | "version": "1.3.8", 934 | "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", 935 | "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", 936 | "dev": true 937 | }, 938 | "node_modules/is-arrayish": { 939 | "version": "0.2.1", 940 | "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", 941 | "integrity": "sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg==", 942 | "dev": true 943 | }, 944 | "node_modules/is-core-module": { 945 | "version": "2.16.1", 946 | "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz", 947 | "integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==", 948 | "dev": true, 949 | "dependencies": { 950 | "hasown": "^2.0.2" 951 | }, 952 | "engines": { 953 | "node": ">= 0.4" 954 | }, 955 | "funding": { 956 | "url": "https://github.com/sponsors/ljharb" 957 | } 958 | }, 959 | "node_modules/is-fullwidth-code-point": { 960 | "version": "3.0.0", 961 | "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", 962 | "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", 963 | "dev": true, 964 | "engines": { 965 | "node": ">=8" 966 | } 967 | }, 968 | "node_modules/is-obj": { 969 | "version": "2.0.0", 970 | "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", 971 | "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", 972 | "dev": true, 973 | "engines": { 974 | "node": ">=8" 975 | } 976 | }, 977 | "node_modules/is-plain-obj": { 978 | "version": "1.1.0", 979 | "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", 980 | "integrity": "sha512-yvkRyxmFKEOQ4pNXCmJG5AEQNlXJS5LaONXo5/cLdTZdWvsZ1ioJEonLGAosKlMWE8lwUy/bJzMjcw8az73+Fg==", 981 | "dev": true, 982 | "engines": { 983 | "node": ">=0.10.0" 984 | } 985 | }, 986 | "node_modules/is-text-path": { 987 | "version": "1.0.1", 988 | "resolved": "https://registry.npmjs.org/is-text-path/-/is-text-path-1.0.1.tgz", 989 | "integrity": "sha512-xFuJpne9oFz5qDaodwmmG08e3CawH/2ZV8Qqza1Ko7Sk8POWbkRdwIoAWVhqvq0XeUzANEhKo2n0IXUGBm7A/w==", 990 | "dev": true, 991 | "dependencies": { 992 | "text-extensions": "^1.0.0" 993 | }, 994 | "engines": { 995 | "node": ">=0.10.0" 996 | } 997 | }, 998 | "node_modules/isarray": { 999 | "version": "1.0.0", 1000 | "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", 1001 | "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", 1002 | "dev": true 1003 | }, 1004 | "node_modules/js-tokens": { 1005 | "version": "4.0.0", 1006 | "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", 1007 | "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", 1008 | "dev": true 1009 | }, 1010 | "node_modules/json-parse-better-errors": { 1011 | "version": "1.0.2", 1012 | "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", 1013 | "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", 1014 | "dev": true 1015 | }, 1016 | "node_modules/json-parse-even-better-errors": { 1017 | "version": "2.3.1", 1018 | "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", 1019 | "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", 1020 | "dev": true 1021 | }, 1022 | "node_modules/json-stringify-safe": { 1023 | "version": "5.0.1", 1024 | "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", 1025 | "integrity": "sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==", 1026 | "dev": true 1027 | }, 1028 | "node_modules/jsonparse": { 1029 | "version": "1.3.1", 1030 | "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", 1031 | "integrity": "sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==", 1032 | "dev": true, 1033 | "engines": [ 1034 | "node >= 0.2.0" 1035 | ] 1036 | }, 1037 | "node_modules/JSONStream": { 1038 | "version": "1.3.5", 1039 | "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", 1040 | "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", 1041 | "dev": true, 1042 | "dependencies": { 1043 | "jsonparse": "^1.2.0", 1044 | "through": ">=2.2.7 <3" 1045 | }, 1046 | "bin": { 1047 | "JSONStream": "bin.js" 1048 | }, 1049 | "engines": { 1050 | "node": "*" 1051 | } 1052 | }, 1053 | "node_modules/kind-of": { 1054 | "version": "6.0.3", 1055 | "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", 1056 | "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", 1057 | "dev": true, 1058 | "engines": { 1059 | "node": ">=0.10.0" 1060 | } 1061 | }, 1062 | "node_modules/lines-and-columns": { 1063 | "version": "1.2.4", 1064 | "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz", 1065 | "integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==", 1066 | "dev": true 1067 | }, 1068 | "node_modules/load-json-file": { 1069 | "version": "4.0.0", 1070 | "resolved": "https://registry.npmjs.org/load-json-file/-/load-json-file-4.0.0.tgz", 1071 | "integrity": "sha512-Kx8hMakjX03tiGTLAIdJ+lL0htKnXjEZN6hk/tozf/WOuYGdZBJrZ+rCJRbVCugsjB3jMLn9746NsQIf5VjBMw==", 1072 | "dev": true, 1073 | "dependencies": { 1074 | "graceful-fs": "^4.1.2", 1075 | "parse-json": "^4.0.0", 1076 | "pify": "^3.0.0", 1077 | "strip-bom": "^3.0.0" 1078 | }, 1079 | "engines": { 1080 | "node": ">=4" 1081 | } 1082 | }, 1083 | "node_modules/load-json-file/node_modules/pify": { 1084 | "version": "3.0.0", 1085 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 1086 | "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", 1087 | "dev": true, 1088 | "engines": { 1089 | "node": ">=4" 1090 | } 1091 | }, 1092 | "node_modules/locate-path": { 1093 | "version": "6.0.0", 1094 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-6.0.0.tgz", 1095 | "integrity": "sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==", 1096 | "dev": true, 1097 | "dependencies": { 1098 | "p-locate": "^5.0.0" 1099 | }, 1100 | "engines": { 1101 | "node": ">=10" 1102 | }, 1103 | "funding": { 1104 | "url": "https://github.com/sponsors/sindresorhus" 1105 | } 1106 | }, 1107 | "node_modules/lodash": { 1108 | "version": "4.17.21", 1109 | "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", 1110 | "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==", 1111 | "dev": true 1112 | }, 1113 | "node_modules/lodash.ismatch": { 1114 | "version": "4.4.0", 1115 | "resolved": "https://registry.npmjs.org/lodash.ismatch/-/lodash.ismatch-4.4.0.tgz", 1116 | "integrity": "sha512-fPMfXjGQEV9Xsq/8MTSgUf255gawYRbjwMyDbcvDhXgV7enSZA0hynz6vMPnpAb5iONEzBHBPsT+0zes5Z301g==", 1117 | "dev": true 1118 | }, 1119 | "node_modules/lru-cache": { 1120 | "version": "6.0.0", 1121 | "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", 1122 | "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", 1123 | "dev": true, 1124 | "dependencies": { 1125 | "yallist": "^4.0.0" 1126 | }, 1127 | "engines": { 1128 | "node": ">=10" 1129 | } 1130 | }, 1131 | "node_modules/map-obj": { 1132 | "version": "4.3.0", 1133 | "resolved": "https://registry.npmjs.org/map-obj/-/map-obj-4.3.0.tgz", 1134 | "integrity": "sha512-hdN1wVrZbb29eBGiGjJbeP8JbKjq1urkHJ/LIP/NY48MZ1QVXUsQBV1G1zvYFHn1XE06cwjBsOI2K3Ulnj1YXQ==", 1135 | "dev": true, 1136 | "engines": { 1137 | "node": ">=8" 1138 | }, 1139 | "funding": { 1140 | "url": "https://github.com/sponsors/sindresorhus" 1141 | } 1142 | }, 1143 | "node_modules/meow": { 1144 | "version": "8.1.2", 1145 | "resolved": "https://registry.npmjs.org/meow/-/meow-8.1.2.tgz", 1146 | "integrity": "sha512-r85E3NdZ+mpYk1C6RjPFEMSE+s1iZMuHtsHAqY0DT3jZczl0diWUZ8g6oU7h0M9cD2EL+PzaYghhCLzR0ZNn5Q==", 1147 | "dev": true, 1148 | "dependencies": { 1149 | "@types/minimist": "^1.2.0", 1150 | "camelcase-keys": "^6.2.2", 1151 | "decamelize-keys": "^1.1.0", 1152 | "hard-rejection": "^2.1.0", 1153 | "minimist-options": "4.1.0", 1154 | "normalize-package-data": "^3.0.0", 1155 | "read-pkg-up": "^7.0.1", 1156 | "redent": "^3.0.0", 1157 | "trim-newlines": "^3.0.0", 1158 | "type-fest": "^0.18.0", 1159 | "yargs-parser": "^20.2.3" 1160 | }, 1161 | "engines": { 1162 | "node": ">=10" 1163 | }, 1164 | "funding": { 1165 | "url": "https://github.com/sponsors/sindresorhus" 1166 | } 1167 | }, 1168 | "node_modules/meow/node_modules/find-up": { 1169 | "version": "4.1.0", 1170 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", 1171 | "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", 1172 | "dev": true, 1173 | "dependencies": { 1174 | "locate-path": "^5.0.0", 1175 | "path-exists": "^4.0.0" 1176 | }, 1177 | "engines": { 1178 | "node": ">=8" 1179 | } 1180 | }, 1181 | "node_modules/meow/node_modules/hosted-git-info": { 1182 | "version": "2.8.9", 1183 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", 1184 | "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", 1185 | "dev": true 1186 | }, 1187 | "node_modules/meow/node_modules/locate-path": { 1188 | "version": "5.0.0", 1189 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", 1190 | "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", 1191 | "dev": true, 1192 | "dependencies": { 1193 | "p-locate": "^4.1.0" 1194 | }, 1195 | "engines": { 1196 | "node": ">=8" 1197 | } 1198 | }, 1199 | "node_modules/meow/node_modules/p-limit": { 1200 | "version": "2.3.0", 1201 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", 1202 | "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", 1203 | "dev": true, 1204 | "dependencies": { 1205 | "p-try": "^2.0.0" 1206 | }, 1207 | "engines": { 1208 | "node": ">=6" 1209 | }, 1210 | "funding": { 1211 | "url": "https://github.com/sponsors/sindresorhus" 1212 | } 1213 | }, 1214 | "node_modules/meow/node_modules/p-locate": { 1215 | "version": "4.1.0", 1216 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", 1217 | "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", 1218 | "dev": true, 1219 | "dependencies": { 1220 | "p-limit": "^2.2.0" 1221 | }, 1222 | "engines": { 1223 | "node": ">=8" 1224 | } 1225 | }, 1226 | "node_modules/meow/node_modules/parse-json": { 1227 | "version": "5.2.0", 1228 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", 1229 | "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", 1230 | "dev": true, 1231 | "dependencies": { 1232 | "@babel/code-frame": "^7.0.0", 1233 | "error-ex": "^1.3.1", 1234 | "json-parse-even-better-errors": "^2.3.0", 1235 | "lines-and-columns": "^1.1.6" 1236 | }, 1237 | "engines": { 1238 | "node": ">=8" 1239 | }, 1240 | "funding": { 1241 | "url": "https://github.com/sponsors/sindresorhus" 1242 | } 1243 | }, 1244 | "node_modules/meow/node_modules/read-pkg": { 1245 | "version": "5.2.0", 1246 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-5.2.0.tgz", 1247 | "integrity": "sha512-Ug69mNOpfvKDAc2Q8DRpMjjzdtrnv9HcSMX+4VsZxD1aZ6ZzrIE7rlzXBtWTyhULSMKg076AW6WR5iZpD0JiOg==", 1248 | "dev": true, 1249 | "dependencies": { 1250 | "@types/normalize-package-data": "^2.4.0", 1251 | "normalize-package-data": "^2.5.0", 1252 | "parse-json": "^5.0.0", 1253 | "type-fest": "^0.6.0" 1254 | }, 1255 | "engines": { 1256 | "node": ">=8" 1257 | } 1258 | }, 1259 | "node_modules/meow/node_modules/read-pkg-up": { 1260 | "version": "7.0.1", 1261 | "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-7.0.1.tgz", 1262 | "integrity": "sha512-zK0TB7Xd6JpCLmlLmufqykGE+/TlOePD6qKClNW7hHDKFh/J7/7gCWGR7joEQEW1bKq3a3yUZSObOoWLFQ4ohg==", 1263 | "dev": true, 1264 | "dependencies": { 1265 | "find-up": "^4.1.0", 1266 | "read-pkg": "^5.2.0", 1267 | "type-fest": "^0.8.1" 1268 | }, 1269 | "engines": { 1270 | "node": ">=8" 1271 | }, 1272 | "funding": { 1273 | "url": "https://github.com/sponsors/sindresorhus" 1274 | } 1275 | }, 1276 | "node_modules/meow/node_modules/read-pkg-up/node_modules/type-fest": { 1277 | "version": "0.8.1", 1278 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.8.1.tgz", 1279 | "integrity": "sha512-4dbzIzqvjtgiM5rw1k5rEHtBANKmdudhGyBEajN01fEyhaAIhsoKNy6y7+IN93IfpFtwY9iqi7kD+xwKhQsNJA==", 1280 | "dev": true, 1281 | "engines": { 1282 | "node": ">=8" 1283 | } 1284 | }, 1285 | "node_modules/meow/node_modules/read-pkg/node_modules/normalize-package-data": { 1286 | "version": "2.5.0", 1287 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", 1288 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", 1289 | "dev": true, 1290 | "dependencies": { 1291 | "hosted-git-info": "^2.1.4", 1292 | "resolve": "^1.10.0", 1293 | "semver": "2 || 3 || 4 || 5", 1294 | "validate-npm-package-license": "^3.0.1" 1295 | } 1296 | }, 1297 | "node_modules/meow/node_modules/read-pkg/node_modules/type-fest": { 1298 | "version": "0.6.0", 1299 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.6.0.tgz", 1300 | "integrity": "sha512-q+MB8nYR1KDLrgr4G5yemftpMC7/QLqVndBmEEdqzmNj5dcFOO4Oo8qlwZE3ULT3+Zim1F8Kq4cBnikNhlCMlg==", 1301 | "dev": true, 1302 | "engines": { 1303 | "node": ">=8" 1304 | } 1305 | }, 1306 | "node_modules/meow/node_modules/semver": { 1307 | "version": "5.7.2", 1308 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", 1309 | "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", 1310 | "dev": true, 1311 | "bin": { 1312 | "semver": "bin/semver" 1313 | } 1314 | }, 1315 | "node_modules/min-indent": { 1316 | "version": "1.0.1", 1317 | "resolved": "https://registry.npmjs.org/min-indent/-/min-indent-1.0.1.tgz", 1318 | "integrity": "sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg==", 1319 | "dev": true, 1320 | "engines": { 1321 | "node": ">=4" 1322 | } 1323 | }, 1324 | "node_modules/minimatch": { 1325 | "version": "3.1.2", 1326 | "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz", 1327 | "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==", 1328 | "dev": true, 1329 | "dependencies": { 1330 | "brace-expansion": "^1.1.7" 1331 | }, 1332 | "engines": { 1333 | "node": "*" 1334 | } 1335 | }, 1336 | "node_modules/minimist": { 1337 | "version": "1.2.8", 1338 | "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz", 1339 | "integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==", 1340 | "dev": true, 1341 | "funding": { 1342 | "url": "https://github.com/sponsors/ljharb" 1343 | } 1344 | }, 1345 | "node_modules/minimist-options": { 1346 | "version": "4.1.0", 1347 | "resolved": "https://registry.npmjs.org/minimist-options/-/minimist-options-4.1.0.tgz", 1348 | "integrity": "sha512-Q4r8ghd80yhO/0j1O3B2BjweX3fiHg9cdOwjJd2J76Q135c+NDxGCqdYKQ1SKBuFfgWbAUzBfvYjPUEeNgqN1A==", 1349 | "dev": true, 1350 | "dependencies": { 1351 | "arrify": "^1.0.1", 1352 | "is-plain-obj": "^1.1.0", 1353 | "kind-of": "^6.0.3" 1354 | }, 1355 | "engines": { 1356 | "node": ">= 6" 1357 | } 1358 | }, 1359 | "node_modules/modify-values": { 1360 | "version": "1.0.1", 1361 | "resolved": "https://registry.npmjs.org/modify-values/-/modify-values-1.0.1.tgz", 1362 | "integrity": "sha512-xV2bxeN6F7oYjZWTe/YPAy6MN2M+sL4u/Rlm2AHCIVGfo2p1yGmBHQ6vHehl4bRTZBdHu3TSkWdYgkwpYzAGSw==", 1363 | "dev": true, 1364 | "engines": { 1365 | "node": ">=0.10.0" 1366 | } 1367 | }, 1368 | "node_modules/neo-async": { 1369 | "version": "2.6.2", 1370 | "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", 1371 | "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", 1372 | "dev": true 1373 | }, 1374 | "node_modules/normalize-package-data": { 1375 | "version": "3.0.3", 1376 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-3.0.3.tgz", 1377 | "integrity": "sha512-p2W1sgqij3zMMyRC067Dg16bfzVH+w7hyegmpIvZ4JNjqtGOVAIvLmjBx3yP7YTe9vKJgkoNOPjwQGogDoMXFA==", 1378 | "dev": true, 1379 | "dependencies": { 1380 | "hosted-git-info": "^4.0.1", 1381 | "is-core-module": "^2.5.0", 1382 | "semver": "^7.3.4", 1383 | "validate-npm-package-license": "^3.0.1" 1384 | }, 1385 | "engines": { 1386 | "node": ">=10" 1387 | } 1388 | }, 1389 | "node_modules/p-limit": { 1390 | "version": "3.1.0", 1391 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", 1392 | "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", 1393 | "dev": true, 1394 | "dependencies": { 1395 | "yocto-queue": "^0.1.0" 1396 | }, 1397 | "engines": { 1398 | "node": ">=10" 1399 | }, 1400 | "funding": { 1401 | "url": "https://github.com/sponsors/sindresorhus" 1402 | } 1403 | }, 1404 | "node_modules/p-locate": { 1405 | "version": "5.0.0", 1406 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-5.0.0.tgz", 1407 | "integrity": "sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==", 1408 | "dev": true, 1409 | "dependencies": { 1410 | "p-limit": "^3.0.2" 1411 | }, 1412 | "engines": { 1413 | "node": ">=10" 1414 | }, 1415 | "funding": { 1416 | "url": "https://github.com/sponsors/sindresorhus" 1417 | } 1418 | }, 1419 | "node_modules/p-try": { 1420 | "version": "2.2.0", 1421 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", 1422 | "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", 1423 | "dev": true, 1424 | "engines": { 1425 | "node": ">=6" 1426 | } 1427 | }, 1428 | "node_modules/parse-json": { 1429 | "version": "4.0.0", 1430 | "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", 1431 | "integrity": "sha512-aOIos8bujGN93/8Ox/jPLh7RwVnPEysynVFE+fQZyg6jKELEHwzgKdLRFHUgXJL6kylijVSBC4BvN9OmsB48Rw==", 1432 | "dev": true, 1433 | "dependencies": { 1434 | "error-ex": "^1.3.1", 1435 | "json-parse-better-errors": "^1.0.1" 1436 | }, 1437 | "engines": { 1438 | "node": ">=4" 1439 | } 1440 | }, 1441 | "node_modules/path-exists": { 1442 | "version": "4.0.0", 1443 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", 1444 | "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", 1445 | "dev": true, 1446 | "engines": { 1447 | "node": ">=8" 1448 | } 1449 | }, 1450 | "node_modules/path-parse": { 1451 | "version": "1.0.7", 1452 | "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz", 1453 | "integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==", 1454 | "dev": true 1455 | }, 1456 | "node_modules/path-type": { 1457 | "version": "3.0.0", 1458 | "resolved": "https://registry.npmjs.org/path-type/-/path-type-3.0.0.tgz", 1459 | "integrity": "sha512-T2ZUsdZFHgA3u4e5PfPbjd7HDDpxPnQb5jN0SrDsjNSuVXHJqtwTnWqG0B1jZrgmJ/7lj1EmVIByWt1gxGkWvg==", 1460 | "dev": true, 1461 | "dependencies": { 1462 | "pify": "^3.0.0" 1463 | }, 1464 | "engines": { 1465 | "node": ">=4" 1466 | } 1467 | }, 1468 | "node_modules/path-type/node_modules/pify": { 1469 | "version": "3.0.0", 1470 | "resolved": "https://registry.npmjs.org/pify/-/pify-3.0.0.tgz", 1471 | "integrity": "sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg==", 1472 | "dev": true, 1473 | "engines": { 1474 | "node": ">=4" 1475 | } 1476 | }, 1477 | "node_modules/picocolors": { 1478 | "version": "1.1.1", 1479 | "resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz", 1480 | "integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==", 1481 | "dev": true 1482 | }, 1483 | "node_modules/pify": { 1484 | "version": "2.3.0", 1485 | "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", 1486 | "integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==", 1487 | "dev": true, 1488 | "engines": { 1489 | "node": ">=0.10.0" 1490 | } 1491 | }, 1492 | "node_modules/process-nextick-args": { 1493 | "version": "2.0.1", 1494 | "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", 1495 | "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", 1496 | "dev": true 1497 | }, 1498 | "node_modules/q": { 1499 | "version": "1.5.1", 1500 | "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", 1501 | "integrity": "sha512-kV/CThkXo6xyFEZUugw/+pIOywXcDbFYgSct5cT3gqlbkBE1SJdwy6UQoZvodiWF/ckQLZyDE/Bu1M6gVu5lVw==", 1502 | "deprecated": "You or someone you depend on is using Q, the JavaScript Promise library that gave JavaScript developers strong feelings about promises. They can almost certainly migrate to the native JavaScript promise now. Thank you literally everyone for joining me in this bet against the odds. Be excellent to each other.\n\n(For a CapTP with native promises, see @endo/eventual-send and @endo/captp)", 1503 | "dev": true, 1504 | "engines": { 1505 | "node": ">=0.6.0", 1506 | "teleport": ">=0.2.0" 1507 | } 1508 | }, 1509 | "node_modules/quick-lru": { 1510 | "version": "4.0.1", 1511 | "resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-4.0.1.tgz", 1512 | "integrity": "sha512-ARhCpm70fzdcvNQfPoy49IaanKkTlRWF2JMzqhcJbhSFRZv7nPTvZJdcY7301IPmvW+/p0RgIWnQDLJxifsQ7g==", 1513 | "dev": true, 1514 | "engines": { 1515 | "node": ">=8" 1516 | } 1517 | }, 1518 | "node_modules/read-pkg": { 1519 | "version": "3.0.0", 1520 | "resolved": "https://registry.npmjs.org/read-pkg/-/read-pkg-3.0.0.tgz", 1521 | "integrity": "sha512-BLq/cCO9two+lBgiTYNqD6GdtK8s4NpaWrl6/rCO9w0TUS8oJl7cmToOZfRYllKTISY6nt1U7jQ53brmKqY6BA==", 1522 | "dev": true, 1523 | "dependencies": { 1524 | "load-json-file": "^4.0.0", 1525 | "normalize-package-data": "^2.3.2", 1526 | "path-type": "^3.0.0" 1527 | }, 1528 | "engines": { 1529 | "node": ">=4" 1530 | } 1531 | }, 1532 | "node_modules/read-pkg-up": { 1533 | "version": "3.0.0", 1534 | "resolved": "https://registry.npmjs.org/read-pkg-up/-/read-pkg-up-3.0.0.tgz", 1535 | "integrity": "sha512-YFzFrVvpC6frF1sz8psoHDBGF7fLPc+llq/8NB43oagqWkx8ar5zYtsTORtOjw9W2RHLpWP+zTWwBvf1bCmcSw==", 1536 | "dev": true, 1537 | "dependencies": { 1538 | "find-up": "^2.0.0", 1539 | "read-pkg": "^3.0.0" 1540 | }, 1541 | "engines": { 1542 | "node": ">=4" 1543 | } 1544 | }, 1545 | "node_modules/read-pkg-up/node_modules/find-up": { 1546 | "version": "2.1.0", 1547 | "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", 1548 | "integrity": "sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ==", 1549 | "dev": true, 1550 | "dependencies": { 1551 | "locate-path": "^2.0.0" 1552 | }, 1553 | "engines": { 1554 | "node": ">=4" 1555 | } 1556 | }, 1557 | "node_modules/read-pkg-up/node_modules/locate-path": { 1558 | "version": "2.0.0", 1559 | "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", 1560 | "integrity": "sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA==", 1561 | "dev": true, 1562 | "dependencies": { 1563 | "p-locate": "^2.0.0", 1564 | "path-exists": "^3.0.0" 1565 | }, 1566 | "engines": { 1567 | "node": ">=4" 1568 | } 1569 | }, 1570 | "node_modules/read-pkg-up/node_modules/p-limit": { 1571 | "version": "1.3.0", 1572 | "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", 1573 | "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", 1574 | "dev": true, 1575 | "dependencies": { 1576 | "p-try": "^1.0.0" 1577 | }, 1578 | "engines": { 1579 | "node": ">=4" 1580 | } 1581 | }, 1582 | "node_modules/read-pkg-up/node_modules/p-locate": { 1583 | "version": "2.0.0", 1584 | "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", 1585 | "integrity": "sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg==", 1586 | "dev": true, 1587 | "dependencies": { 1588 | "p-limit": "^1.1.0" 1589 | }, 1590 | "engines": { 1591 | "node": ">=4" 1592 | } 1593 | }, 1594 | "node_modules/read-pkg-up/node_modules/p-try": { 1595 | "version": "1.0.0", 1596 | "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", 1597 | "integrity": "sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww==", 1598 | "dev": true, 1599 | "engines": { 1600 | "node": ">=4" 1601 | } 1602 | }, 1603 | "node_modules/read-pkg-up/node_modules/path-exists": { 1604 | "version": "3.0.0", 1605 | "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", 1606 | "integrity": "sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ==", 1607 | "dev": true, 1608 | "engines": { 1609 | "node": ">=4" 1610 | } 1611 | }, 1612 | "node_modules/read-pkg/node_modules/hosted-git-info": { 1613 | "version": "2.8.9", 1614 | "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.9.tgz", 1615 | "integrity": "sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw==", 1616 | "dev": true 1617 | }, 1618 | "node_modules/read-pkg/node_modules/normalize-package-data": { 1619 | "version": "2.5.0", 1620 | "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", 1621 | "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", 1622 | "dev": true, 1623 | "dependencies": { 1624 | "hosted-git-info": "^2.1.4", 1625 | "resolve": "^1.10.0", 1626 | "semver": "2 || 3 || 4 || 5", 1627 | "validate-npm-package-license": "^3.0.1" 1628 | } 1629 | }, 1630 | "node_modules/read-pkg/node_modules/semver": { 1631 | "version": "5.7.2", 1632 | "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.2.tgz", 1633 | "integrity": "sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g==", 1634 | "dev": true, 1635 | "bin": { 1636 | "semver": "bin/semver" 1637 | } 1638 | }, 1639 | "node_modules/readable-stream": { 1640 | "version": "3.6.2", 1641 | "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", 1642 | "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", 1643 | "dev": true, 1644 | "dependencies": { 1645 | "inherits": "^2.0.3", 1646 | "string_decoder": "^1.1.1", 1647 | "util-deprecate": "^1.0.1" 1648 | }, 1649 | "engines": { 1650 | "node": ">= 6" 1651 | } 1652 | }, 1653 | "node_modules/redent": { 1654 | "version": "3.0.0", 1655 | "resolved": "https://registry.npmjs.org/redent/-/redent-3.0.0.tgz", 1656 | "integrity": "sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg==", 1657 | "dev": true, 1658 | "dependencies": { 1659 | "indent-string": "^4.0.0", 1660 | "strip-indent": "^3.0.0" 1661 | }, 1662 | "engines": { 1663 | "node": ">=8" 1664 | } 1665 | }, 1666 | "node_modules/require-directory": { 1667 | "version": "2.1.1", 1668 | "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", 1669 | "integrity": "sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==", 1670 | "dev": true, 1671 | "engines": { 1672 | "node": ">=0.10.0" 1673 | } 1674 | }, 1675 | "node_modules/resolve": { 1676 | "version": "1.22.10", 1677 | "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.10.tgz", 1678 | "integrity": "sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==", 1679 | "dev": true, 1680 | "dependencies": { 1681 | "is-core-module": "^2.16.0", 1682 | "path-parse": "^1.0.7", 1683 | "supports-preserve-symlinks-flag": "^1.0.0" 1684 | }, 1685 | "bin": { 1686 | "resolve": "bin/resolve" 1687 | }, 1688 | "engines": { 1689 | "node": ">= 0.4" 1690 | }, 1691 | "funding": { 1692 | "url": "https://github.com/sponsors/ljharb" 1693 | } 1694 | }, 1695 | "node_modules/safe-buffer": { 1696 | "version": "5.2.1", 1697 | "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", 1698 | "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", 1699 | "dev": true, 1700 | "funding": [ 1701 | { 1702 | "type": "github", 1703 | "url": "https://github.com/sponsors/feross" 1704 | }, 1705 | { 1706 | "type": "patreon", 1707 | "url": "https://www.patreon.com/feross" 1708 | }, 1709 | { 1710 | "type": "consulting", 1711 | "url": "https://feross.org/support" 1712 | } 1713 | ] 1714 | }, 1715 | "node_modules/semver": { 1716 | "version": "7.6.3", 1717 | "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz", 1718 | "integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==", 1719 | "dev": true, 1720 | "bin": { 1721 | "semver": "bin/semver.js" 1722 | }, 1723 | "engines": { 1724 | "node": ">=10" 1725 | } 1726 | }, 1727 | "node_modules/source-map": { 1728 | "version": "0.6.1", 1729 | "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", 1730 | "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", 1731 | "dev": true, 1732 | "engines": { 1733 | "node": ">=0.10.0" 1734 | } 1735 | }, 1736 | "node_modules/spdx-correct": { 1737 | "version": "3.2.0", 1738 | "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.2.0.tgz", 1739 | "integrity": "sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA==", 1740 | "dev": true, 1741 | "dependencies": { 1742 | "spdx-expression-parse": "^3.0.0", 1743 | "spdx-license-ids": "^3.0.0" 1744 | } 1745 | }, 1746 | "node_modules/spdx-exceptions": { 1747 | "version": "2.5.0", 1748 | "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz", 1749 | "integrity": "sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w==", 1750 | "dev": true 1751 | }, 1752 | "node_modules/spdx-expression-parse": { 1753 | "version": "3.0.1", 1754 | "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", 1755 | "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", 1756 | "dev": true, 1757 | "dependencies": { 1758 | "spdx-exceptions": "^2.1.0", 1759 | "spdx-license-ids": "^3.0.0" 1760 | } 1761 | }, 1762 | "node_modules/spdx-license-ids": { 1763 | "version": "3.0.21", 1764 | "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.21.tgz", 1765 | "integrity": "sha512-Bvg/8F5XephndSK3JffaRqdT+gyhfqIPwDHpX80tJrF8QQRYMo8sNMeaZ2Dp5+jhwKnUmIOyFFQfHRkjJm5nXg==", 1766 | "dev": true 1767 | }, 1768 | "node_modules/split": { 1769 | "version": "1.0.1", 1770 | "resolved": "https://registry.npmjs.org/split/-/split-1.0.1.tgz", 1771 | "integrity": "sha512-mTyOoPbrivtXnwnIxZRFYRrPNtEFKlpB2fvjSnCQUiAA6qAZzqwna5envK4uk6OIeP17CsdF3rSBGYVBsU0Tkg==", 1772 | "dev": true, 1773 | "dependencies": { 1774 | "through": "2" 1775 | }, 1776 | "engines": { 1777 | "node": "*" 1778 | } 1779 | }, 1780 | "node_modules/split2": { 1781 | "version": "3.2.2", 1782 | "resolved": "https://registry.npmjs.org/split2/-/split2-3.2.2.tgz", 1783 | "integrity": "sha512-9NThjpgZnifTkJpzTZ7Eue85S49QwpNhZTq6GRJwObb6jnLFNGB7Qm73V5HewTROPyxD0C29xqmaI68bQtV+hg==", 1784 | "dev": true, 1785 | "dependencies": { 1786 | "readable-stream": "^3.0.0" 1787 | } 1788 | }, 1789 | "node_modules/standard-version": { 1790 | "version": "9.5.0", 1791 | "resolved": "https://registry.npmjs.org/standard-version/-/standard-version-9.5.0.tgz", 1792 | "integrity": "sha512-3zWJ/mmZQsOaO+fOlsa0+QK90pwhNd042qEcw6hKFNoLFs7peGyvPffpEBbK/DSGPbyOvli0mUIFv5A4qTjh2Q==", 1793 | "dev": true, 1794 | "dependencies": { 1795 | "chalk": "^2.4.2", 1796 | "conventional-changelog": "3.1.25", 1797 | "conventional-changelog-config-spec": "2.1.0", 1798 | "conventional-changelog-conventionalcommits": "4.6.3", 1799 | "conventional-recommended-bump": "6.1.0", 1800 | "detect-indent": "^6.0.0", 1801 | "detect-newline": "^3.1.0", 1802 | "dotgitignore": "^2.1.0", 1803 | "figures": "^3.1.0", 1804 | "find-up": "^5.0.0", 1805 | "git-semver-tags": "^4.0.0", 1806 | "semver": "^7.1.1", 1807 | "stringify-package": "^1.0.1", 1808 | "yargs": "^16.0.0" 1809 | }, 1810 | "bin": { 1811 | "standard-version": "bin/cli.js" 1812 | }, 1813 | "engines": { 1814 | "node": ">=10" 1815 | } 1816 | }, 1817 | "node_modules/string_decoder": { 1818 | "version": "1.3.0", 1819 | "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", 1820 | "integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==", 1821 | "dev": true, 1822 | "dependencies": { 1823 | "safe-buffer": "~5.2.0" 1824 | } 1825 | }, 1826 | "node_modules/string-width": { 1827 | "version": "4.2.3", 1828 | "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.3.tgz", 1829 | "integrity": "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==", 1830 | "dev": true, 1831 | "dependencies": { 1832 | "emoji-regex": "^8.0.0", 1833 | "is-fullwidth-code-point": "^3.0.0", 1834 | "strip-ansi": "^6.0.1" 1835 | }, 1836 | "engines": { 1837 | "node": ">=8" 1838 | } 1839 | }, 1840 | "node_modules/stringify-package": { 1841 | "version": "1.0.1", 1842 | "resolved": "https://registry.npmjs.org/stringify-package/-/stringify-package-1.0.1.tgz", 1843 | "integrity": "sha512-sa4DUQsYciMP1xhKWGuFM04fB0LG/9DlluZoSVywUMRNvzid6XucHK0/90xGxRoHrAaROrcHK1aPKaijCtSrhg==", 1844 | "deprecated": "This module is not used anymore, and has been replaced by @npmcli/package-json", 1845 | "dev": true 1846 | }, 1847 | "node_modules/strip-ansi": { 1848 | "version": "6.0.1", 1849 | "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.1.tgz", 1850 | "integrity": "sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==", 1851 | "dev": true, 1852 | "dependencies": { 1853 | "ansi-regex": "^5.0.1" 1854 | }, 1855 | "engines": { 1856 | "node": ">=8" 1857 | } 1858 | }, 1859 | "node_modules/strip-bom": { 1860 | "version": "3.0.0", 1861 | "resolved": "https://registry.npmjs.org/strip-bom/-/strip-bom-3.0.0.tgz", 1862 | "integrity": "sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==", 1863 | "dev": true, 1864 | "engines": { 1865 | "node": ">=4" 1866 | } 1867 | }, 1868 | "node_modules/strip-indent": { 1869 | "version": "3.0.0", 1870 | "resolved": "https://registry.npmjs.org/strip-indent/-/strip-indent-3.0.0.tgz", 1871 | "integrity": "sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ==", 1872 | "dev": true, 1873 | "dependencies": { 1874 | "min-indent": "^1.0.0" 1875 | }, 1876 | "engines": { 1877 | "node": ">=8" 1878 | } 1879 | }, 1880 | "node_modules/supports-color": { 1881 | "version": "5.5.0", 1882 | "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", 1883 | "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", 1884 | "dev": true, 1885 | "dependencies": { 1886 | "has-flag": "^3.0.0" 1887 | }, 1888 | "engines": { 1889 | "node": ">=4" 1890 | } 1891 | }, 1892 | "node_modules/supports-preserve-symlinks-flag": { 1893 | "version": "1.0.0", 1894 | "resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz", 1895 | "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", 1896 | "dev": true, 1897 | "engines": { 1898 | "node": ">= 0.4" 1899 | }, 1900 | "funding": { 1901 | "url": "https://github.com/sponsors/ljharb" 1902 | } 1903 | }, 1904 | "node_modules/text-extensions": { 1905 | "version": "1.9.0", 1906 | "resolved": "https://registry.npmjs.org/text-extensions/-/text-extensions-1.9.0.tgz", 1907 | "integrity": "sha512-wiBrwC1EhBelW12Zy26JeOUkQ5mRu+5o8rpsJk5+2t+Y5vE7e842qtZDQ2g1NpX/29HdyFeJ4nSIhI47ENSxlQ==", 1908 | "dev": true, 1909 | "engines": { 1910 | "node": ">=0.10" 1911 | } 1912 | }, 1913 | "node_modules/through": { 1914 | "version": "2.3.8", 1915 | "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", 1916 | "integrity": "sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==", 1917 | "dev": true 1918 | }, 1919 | "node_modules/through2": { 1920 | "version": "4.0.2", 1921 | "resolved": "https://registry.npmjs.org/through2/-/through2-4.0.2.tgz", 1922 | "integrity": "sha512-iOqSav00cVxEEICeD7TjLB1sueEL+81Wpzp2bY17uZjZN0pWZPuo4suZ/61VujxmqSGFfgOcNuTZ85QJwNZQpw==", 1923 | "dev": true, 1924 | "dependencies": { 1925 | "readable-stream": "3" 1926 | } 1927 | }, 1928 | "node_modules/trim-newlines": { 1929 | "version": "3.0.1", 1930 | "resolved": "https://registry.npmjs.org/trim-newlines/-/trim-newlines-3.0.1.tgz", 1931 | "integrity": "sha512-c1PTsA3tYrIsLGkJkzHF+w9F2EyxfXGo4UyJc4pFL++FMjnq0HJS69T3M7d//gKrFKwy429bouPescbjecU+Zw==", 1932 | "dev": true, 1933 | "engines": { 1934 | "node": ">=8" 1935 | } 1936 | }, 1937 | "node_modules/type-fest": { 1938 | "version": "0.18.1", 1939 | "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.18.1.tgz", 1940 | "integrity": "sha512-OIAYXk8+ISY+qTOwkHtKqzAuxchoMiD9Udx+FSGQDuiRR+PJKJHc2NJAXlbhkGwTt/4/nKZxELY1w3ReWOL8mw==", 1941 | "dev": true, 1942 | "engines": { 1943 | "node": ">=10" 1944 | }, 1945 | "funding": { 1946 | "url": "https://github.com/sponsors/sindresorhus" 1947 | } 1948 | }, 1949 | "node_modules/typedarray": { 1950 | "version": "0.0.6", 1951 | "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", 1952 | "integrity": "sha512-/aCDEGatGvZ2BIk+HmLf4ifCJFwvKFNb9/JeZPMulfgFracn9QFcAf5GO8B/mweUjSoblS5In0cWhqpfs/5PQA==", 1953 | "dev": true 1954 | }, 1955 | "node_modules/uglify-js": { 1956 | "version": "3.19.3", 1957 | "resolved": "https://registry.npmjs.org/uglify-js/-/uglify-js-3.19.3.tgz", 1958 | "integrity": "sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==", 1959 | "dev": true, 1960 | "optional": true, 1961 | "bin": { 1962 | "uglifyjs": "bin/uglifyjs" 1963 | }, 1964 | "engines": { 1965 | "node": ">=0.8.0" 1966 | } 1967 | }, 1968 | "node_modules/util-deprecate": { 1969 | "version": "1.0.2", 1970 | "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", 1971 | "integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==", 1972 | "dev": true 1973 | }, 1974 | "node_modules/validate-npm-package-license": { 1975 | "version": "3.0.4", 1976 | "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", 1977 | "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", 1978 | "dev": true, 1979 | "dependencies": { 1980 | "spdx-correct": "^3.0.0", 1981 | "spdx-expression-parse": "^3.0.0" 1982 | } 1983 | }, 1984 | "node_modules/wordwrap": { 1985 | "version": "1.0.0", 1986 | "resolved": "https://registry.npmjs.org/wordwrap/-/wordwrap-1.0.0.tgz", 1987 | "integrity": "sha512-gvVzJFlPycKc5dZN4yPkP8w7Dc37BtP1yczEneOb4uq34pXZcvrtRTmWV8W+Ume+XCxKgbjM+nevkyFPMybd4Q==", 1988 | "dev": true 1989 | }, 1990 | "node_modules/wrap-ansi": { 1991 | "version": "7.0.0", 1992 | "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", 1993 | "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", 1994 | "dev": true, 1995 | "dependencies": { 1996 | "ansi-styles": "^4.0.0", 1997 | "string-width": "^4.1.0", 1998 | "strip-ansi": "^6.0.0" 1999 | }, 2000 | "engines": { 2001 | "node": ">=10" 2002 | }, 2003 | "funding": { 2004 | "url": "https://github.com/chalk/wrap-ansi?sponsor=1" 2005 | } 2006 | }, 2007 | "node_modules/wrap-ansi/node_modules/ansi-styles": { 2008 | "version": "4.3.0", 2009 | "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", 2010 | "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", 2011 | "dev": true, 2012 | "dependencies": { 2013 | "color-convert": "^2.0.1" 2014 | }, 2015 | "engines": { 2016 | "node": ">=8" 2017 | }, 2018 | "funding": { 2019 | "url": "https://github.com/chalk/ansi-styles?sponsor=1" 2020 | } 2021 | }, 2022 | "node_modules/wrap-ansi/node_modules/color-convert": { 2023 | "version": "2.0.1", 2024 | "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", 2025 | "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", 2026 | "dev": true, 2027 | "dependencies": { 2028 | "color-name": "~1.1.4" 2029 | }, 2030 | "engines": { 2031 | "node": ">=7.0.0" 2032 | } 2033 | }, 2034 | "node_modules/wrap-ansi/node_modules/color-name": { 2035 | "version": "1.1.4", 2036 | "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", 2037 | "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", 2038 | "dev": true 2039 | }, 2040 | "node_modules/xtend": { 2041 | "version": "4.0.2", 2042 | "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", 2043 | "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", 2044 | "dev": true, 2045 | "engines": { 2046 | "node": ">=0.4" 2047 | } 2048 | }, 2049 | "node_modules/y18n": { 2050 | "version": "5.0.8", 2051 | "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", 2052 | "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", 2053 | "dev": true, 2054 | "engines": { 2055 | "node": ">=10" 2056 | } 2057 | }, 2058 | "node_modules/yallist": { 2059 | "version": "4.0.0", 2060 | "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", 2061 | "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", 2062 | "dev": true 2063 | }, 2064 | "node_modules/yargs": { 2065 | "version": "16.2.0", 2066 | "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", 2067 | "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", 2068 | "dev": true, 2069 | "dependencies": { 2070 | "cliui": "^7.0.2", 2071 | "escalade": "^3.1.1", 2072 | "get-caller-file": "^2.0.5", 2073 | "require-directory": "^2.1.1", 2074 | "string-width": "^4.2.0", 2075 | "y18n": "^5.0.5", 2076 | "yargs-parser": "^20.2.2" 2077 | }, 2078 | "engines": { 2079 | "node": ">=10" 2080 | } 2081 | }, 2082 | "node_modules/yargs-parser": { 2083 | "version": "20.2.9", 2084 | "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.9.tgz", 2085 | "integrity": "sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==", 2086 | "dev": true, 2087 | "engines": { 2088 | "node": ">=10" 2089 | } 2090 | }, 2091 | "node_modules/yocto-queue": { 2092 | "version": "0.1.0", 2093 | "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", 2094 | "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", 2095 | "dev": true, 2096 | "engines": { 2097 | "node": ">=10" 2098 | }, 2099 | "funding": { 2100 | "url": "https://github.com/sponsors/sindresorhus" 2101 | } 2102 | } 2103 | } 2104 | } 2105 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "headless-browser-clusters", 3 | "version": "1.0.17", 4 | "scripts": { 5 | "app:dev": "npm run start && cd ./app && npm run dev", 6 | "app:build": "cd ./app && npm run docker:build", 7 | "app:chromium:install": "cd ./app && npx playwright install-deps chromium", 8 | "browser:build": "cd ./browser && npm run docker:build", 9 | "build": "npm run app:build && npm run browser:build", 10 | "start": "cd .docker/compose && docker compose up -d", 11 | "release": "npx standard-version", 12 | "publish:release": "npm run release && git push --follow-tags origin main" 13 | }, 14 | "devDependencies": { 15 | "standard-version": "^9.5.0" 16 | } 17 | } 18 | --------------------------------------------------------------------------------