diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index dbdcdb24c..6e74ab560 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -64,11 +64,13 @@ jobs: - name: Install NodeJS Dependencies run: | - npm config set spin false - npm ci --no-audit --no-color - npm ci --prefix webapp --no-audit --no-color - npm ci --prefix cli --no-audit --no-color - npm ci --prefix jipt --no-audit --no-color + npm set fund false + npm set audit false + npm set color false + npm ci + npm ci --prefix webapp + npm ci --prefix cli + npm ci --prefix jipt - name: Build webapp production run: npm run build-production-inline --prefix webapp diff --git a/.tool-versions b/.tool-versions index 711a8c7e6..92f045f11 100644 --- a/.tool-versions +++ b/.tool-versions @@ -1,3 +1,3 @@ elixir 1.15.7-otp-26 erlang 26.1.2 -nodejs 16.19.1 +nodejs 21.6.1 diff --git a/Dockerfile b/Dockerfile index 58b63b9e3..3fbbcdc14 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,7 +1,7 @@ # # Build webapp and jipt deps # -FROM node:16.19-bullseye-slim AS webapp-builder +FROM node:21.6.1-bullseye-slim AS webapp-builder RUN apt-get update -y && \ apt-get install -y build-essential git python3 python3-pip && \ apt-get clean && \ @@ -11,7 +11,7 @@ COPY webapp . RUN npm ci --no-audit --no-color && \ npm run build-production -FROM node:16.19-bullseye-slim AS jipt-builder +FROM node:21.6.1-bullseye-slim AS jipt-builder RUN apt-get update -y && \ apt-get install -y build-essential git python3 python3-pip && \ apt-get clean && \ @@ -47,10 +47,11 @@ COPY config config RUN mix deps.get --only prod RUN mix deps.compile --only prod -COPY vendor vendor +COPY vendor/language_tool/priv/ vendor/language_tool/priv/ RUN cd ./vendor/language_tool/priv/native/languagetool && ./gradlew shadowJar RUN cp ./vendor/language_tool/priv/native/languagetool/app/build/libs/language-tool.jar priv/native/language-tool.jar +COPY vendor vendor COPY lib lib RUN mix compile --only prod diff --git a/Makefile b/Makefile index bfea78118..671dc1b22 100644 --- a/Makefile +++ b/Makefile @@ -97,7 +97,7 @@ lint-eslint: .PHONY: lint-prettier lint-prettier: - npx prettier --check './{webapp,jipt,cli}/!(node_modules)/**/*.{js,ts,json,svg,scss,md,hbs}' '*.md' + npx prettier --trailing-comma none --check './{webapp,jipt,cli}/!(node_modules)/**/*.{js,ts,json,svg,scss,md,hbs}' '*.md' .PHONY: lint-template-hbs lint-template-hbs: @@ -110,16 +110,12 @@ type-check: ## Type-check typescript files cd jipt && npx tsc .PHONY: test -test: test-api test-webapp +test: test-api .PHONY: test-api test-api: ## Run the backend test suite mix test -.PHONY: test-webapp -test-webapp: ## Run the frontend test suite - cd webapp && npx ember exam --reporter dot - .PHONY: test-coverage test-coverage: ## Generate the code coverage report mix coveralls @@ -133,7 +129,7 @@ format-elixir: .PHONY: format-prettier format-prettier: - npx prettier --write --single-quote --no-bracket-spacing './{webapp,jipt,cli}/!(node_modules)/**/*.{js,ts,json,svg,scss,md,hbs}' '*.md' + npx prettier --write --single-quote --trailing-comma none --no-bracket-spacing './{webapp,jipt,cli}/!(node_modules)/**/*.{js,ts,json,svg,scss,md,hbs}' '*.md' # Development targets # ------------------- diff --git a/README.md b/README.md index 21bcc141e..4751ade4a 100644 --- a/README.md +++ b/README.md @@ -28,8 +28,8 @@ Accent provides a powerful abstraction around the process maintaining translatio | --------------------------------------------------- | -------------------------------------------------------------------- | | [🚀 Getting started](#-getting-started) | Quickly setup a working app | | [🚧 Requirements](#-requirements) | Dependencies required to run Accent’ stack | -| [🎛 Mix commands](#-executing-mix-commands) | How to execute mix task with the Twelve-Factor pattern | -| [🏎 Quickstart](#-quickstart) | Steps to run the project, from API to webapp, with or without Docker | +| [🎛 Mix commands](#-executing-mix-commands) | How to execute mix task with the Twelve-Factor pattern | +| [🏎 Quickstart](#-quickstart) | Steps to run the project, from API to webapp, with or without Docker | | [🌳 Environment variables](#-environment-variables) | Required and optional env var used | | [✅ Tests](#-tests) | How to run the extensive tests suite | | [🚀 Heroku](#-deploy-on-heroku) | Easy deployment setup with Heroku | @@ -103,13 +103,14 @@ The Makefile should be the main entry for common tasks such as tests, linting, D For the production setup, we use Docker to build an OTP release of the app. With docker-compose, you can run the image locally. Here are the steps to have a working app running locally with Docker: -_When running the production env, you need to provide a valid GOOGLE_API_CLIENT_ID in the `docker-compose.yml` file._ +_When running the production env, you need to provide a valid authentication setup in the `environment` section in `docker-compose.yml` file._ +See the following sections to see available variables to enable third-party logins or dummy login (email only, no password). 1. Run `make build` to build the OTP release with Docker 2. Run `make dev-start-postgresql` to start an instance of Postgresql. The instance will run on port 5432 with the `postgres` user. You can change those values in the `docker-compose.yml` file. 3. Run `make dev-start-application` to start the app! The release hook of the release will execute migrations and seeds before starting the webserver on port 4000 (again you can change the settings in `docker-compose.yml`) -_That’s it! You now have a working Accent instance without installing Elixir or Node!_ +_That’s it! You now have a working Accent instance without installing Elixir or NodeJS!_ ## 🌳 Environment variables @@ -139,24 +140,29 @@ Accent provides a default value for every required environment variable. This me Various login providers are included in Accent using Ueberauth to abstract services. -| Variable | Default | Description | -| -------------------------- | -------------------- | --------------------------------------------------------------------------------------- | -| `DUMMY_LOGIN_ENABLED` | _none_ | If specified, the password-less authentication (with only the email) will be available. | -| `GITHUB_CLIENT_ID` | _none_ | | -| `GITHUB_CLIENT_SECRET` | _none_ | | -| `GITLAB_CLIENT_ID` | _none_ | | -| `GITLAB_CLIENT_SECRET` | _none_ | | -| `GITLAB_SITE_URL` | `https://gitlab.com` | | -| `GOOGLE_API_CLIENT_ID` | _none_ | | -| `GOOGLE_API_CLIENT_SECRET` | _none_ | | -| `SLACK_CLIENT_ID` | _none_ | | -| `SLACK_CLIENT_SECRET` | _none_ | | -| `SLACK_TEAM_ID` | _none_ | | -| `DISCORD_CLIENT_ID` | _none_ | | -| `DISCORD_CLIENT_SECRET` | _none_ | | -| `MICROSOFT_CLIENT_ID` | _none_ | | -| `MICROSOFT_CLIENT_SECRET` | _none_ | | -| `MICROSOFT_TENANT_ID` | _none_ | | +| Variable | Default | Description | +| -------------------------- | ---------------------- | --------------------------------------------------------------------------------------- | +| `DUMMY_LOGIN_ENABLED` | _none_ | If specified, the password-less authentication (with only the email) will be available. | +| `GITHUB_CLIENT_ID` | _none_ | | +| `GITHUB_CLIENT_SECRET` | _none_ | | +| `GITLAB_CLIENT_ID` | _none_ | | +| `GITLAB_CLIENT_SECRET` | _none_ | | +| `GITLAB_SITE_URL` | `https://gitlab.com` | | +| `GOOGLE_API_CLIENT_ID` | _none_ | | +| `GOOGLE_API_CLIENT_SECRET` | _none_ | | +| `SLACK_CLIENT_ID` | _none_ | | +| `SLACK_CLIENT_SECRET` | _none_ | | +| `SLACK_TEAM_ID` | _none_ | | +| `DISCORD_CLIENT_ID` | _none_ | | +| `DISCORD_CLIENT_SECRET` | _none_ | | +| `MICROSOFT_CLIENT_ID` | _none_ | | +| `MICROSOFT_CLIENT_SECRET` | _none_ | | +| `MICROSOFT_TENANT_ID` | _none_ | | +| `OIDC_CLIENT_ID` | _none_ | | +| `OIDC_CLIENT_SECRET` | _none_ | | +| `OIDC_DISCOVERY_URI` | _none_ | | +| `OIDC_UID_FIELD` | `sub` | | +| `OIDC_SCOPE` | `openid profile email` | | ### Email setup diff --git a/cli/README.md b/cli/README.md index 739d57002..74183043d 100644 --- a/cli/README.md +++ b/cli/README.md @@ -20,7 +20,7 @@ $ npm install -g accent-cli $ accent COMMAND running command... $ accent (-v|--version|version) -accent-cli/0.16.0 darwin-arm64 node-v16.19.1 +accent-cli/0.16.2 darwin-arm64 node-v21.6.1 $ accent --help [COMMAND] USAGE $ accent COMMAND @@ -47,7 +47,7 @@ accent-cli reads from a `accent.json` file. The file should contain valid JSON r "source": "localization/fr/*.json", "target": "localization/%slug%/%document_path%.json", "hooks": { - "afterSync": "touch sync-done.txt" + "afterSync": ["touch sync-done.txt", "echo 'Done!'"] } } ] @@ -207,7 +207,7 @@ EXAMPLES $ accent export --order-by=key --version=build.myapp.com:0.12.345 ``` -_See code: [src/commands/export.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/export.ts)_ +_See code: [src/commands/export.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/export.ts)_ ## `accent format` @@ -225,7 +225,7 @@ EXAMPLE $ accent format ``` -_See code: [src/commands/format.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/format.ts)_ +_See code: [src/commands/format.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/format.ts)_ ## `accent help [COMMAND]` @@ -262,7 +262,7 @@ EXAMPLE $ accent jipt ``` -_See code: [src/commands/jipt.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/jipt.ts)_ +_See code: [src/commands/jipt.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/jipt.ts)_ ## `accent lint` @@ -279,7 +279,7 @@ EXAMPLE $ accent lint ``` -_See code: [src/commands/lint.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/lint.ts)_ +_See code: [src/commands/lint.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/lint.ts)_ ## `accent stats` @@ -299,7 +299,7 @@ EXAMPLE $ accent stats ``` -_See code: [src/commands/stats.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/stats.ts)_ +_See code: [src/commands/stats.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/stats.ts)_ ## `accent sync` @@ -331,7 +331,7 @@ EXAMPLES $ accent sync --add-translations --merge-type=smart --order-key=key --version=v0.23 ``` -_See code: [src/commands/sync.ts](https://github.com/mirego/accent/blob/v0.16.0/src/commands/sync.ts)_ +_See code: [src/commands/sync.ts](https://github.com/mirego/accent/blob/v0.16.2/src/commands/sync.ts)_ # GitHub Actions diff --git a/cli/package-lock.json b/cli/package-lock.json index a42e4d1c8..4c5e0e18e 100644 --- a/cli/package-lock.json +++ b/cli/package-lock.json @@ -1,12 +1,12 @@ { "name": "accent-cli", - "version": "0.16.0", + "version": "0.16.2", "lockfileVersion": 2, "requires": true, "packages": { "": { "name": "accent-cli", - "version": "0.14.0", + "version": "0.16.2", "license": "MIT", "dependencies": { "@oclif/command": "^1.8.22", diff --git a/cli/package.json b/cli/package.json index 87673486a..27c73d014 100644 --- a/cli/package.json +++ b/cli/package.json @@ -1,6 +1,6 @@ { "name": "accent-cli", - "version": "0.16.0", + "version": "0.16.2", "author": "Simon Prévost", "description": "Accent CLI", "bin": { diff --git a/cli/src/base.ts b/cli/src/base.ts index 0620aa699..13994a17a 100644 --- a/cli/src/base.ts +++ b/cli/src/base.ts @@ -17,7 +17,7 @@ const sleep = async (ms: number) => export const configFlag = flags.string({ default: 'accent.json', - description: 'Path to the config file', + description: 'Path to the config file' }); const parseConfigFlag = (argv: string[]) => { diff --git a/cli/src/commands/export.ts b/cli/src/commands/export.ts index 989c88912..d1dc90e4a 100644 --- a/cli/src/commands/export.ts +++ b/cli/src/commands/export.ts @@ -19,7 +19,7 @@ export default class Export extends Command { static examples = [ `$ accent export`, - `$ accent export --order-by=key --version=build.myapp.com:0.12.345`, + `$ accent export --order-by=key --version=build.myapp.com:0.12.345` ]; static args = []; @@ -27,13 +27,13 @@ export default class Export extends Command { 'order-by': flags.string({ default: 'index', description: 'Order of the keys', - options: ['index', 'key'], + options: ['index', 'key'] }), version: flags.string({ default: '', - description: 'Fetch a specific version', + description: 'Fetch a specific version' }), - config: configFlag, + config: configFlag }; async run() { diff --git a/cli/src/commands/format.ts b/cli/src/commands/format.ts index 7bf939ae3..7599ad341 100644 --- a/cli/src/commands/format.ts +++ b/cli/src/commands/format.ts @@ -21,9 +21,9 @@ export default class Format extends Command { 'order-by': flags.string({ default: 'index', description: 'Order of the keys', - options: ['index', 'key', '-index', '-key'], + options: ['index', 'key', '-index', '-key'] }), - config: configFlag, + config: configFlag }; async run() { diff --git a/cli/src/commands/jipt.ts b/cli/src/commands/jipt.ts index a91be0312..3fbf99775 100644 --- a/cli/src/commands/jipt.ts +++ b/cli/src/commands/jipt.ts @@ -22,8 +22,8 @@ export default class Jipt extends Command { { description: 'The pseudo language for in-place-translation-editing', name: 'pseudoLanguageName', - required: true, - }, + required: true + } ]; static flags = {config: configFlag}; diff --git a/cli/src/commands/lint.ts b/cli/src/commands/lint.ts index 5477e3969..d08c1b407 100644 --- a/cli/src/commands/lint.ts +++ b/cli/src/commands/lint.ts @@ -31,13 +31,13 @@ export default class Lint extends Command { const {path, language} = target; if (fs.existsSync(path)) { const { - data: {lint_translations: lintTranslations}, + data: {lint_translations: lintTranslations} } = (await document.lint(path, language)) as {data: any}; const lintTranslationsWithLocalPath = lintTranslations.map( (lintTranslation: LintTranslation) => ({ ...lintTranslation, - path, + path }) ); diff --git a/cli/src/commands/stats.ts b/cli/src/commands/stats.ts index 26a037f7f..fc838c14c 100644 --- a/cli/src/commands/stats.ts +++ b/cli/src/commands/stats.ts @@ -19,15 +19,15 @@ export default class Stats extends Command { static flags = { version: flags.string({ default: undefined, - description: 'View stats for a specific version', + description: 'View stats for a specific version' }), 'check-reviewed': flags.boolean({ - description: 'Exit 1 when reviewed percentage is not 100%', + description: 'Exit 1 when reviewed percentage is not 100%' }), 'check-translated': flags.boolean({ - description: 'Exit 1 when translated percentage is not 100%', + description: 'Exit 1 when translated percentage is not 100%' }), - config: configFlag, + config: configFlag }; async run() { diff --git a/cli/src/commands/sync.ts b/cli/src/commands/sync.ts index 45bdd9cac..bc999c404 100644 --- a/cli/src/commands/sync.ts +++ b/cli/src/commands/sync.ts @@ -29,7 +29,7 @@ export default class Sync extends Command { static examples = [ `$ accent sync`, `$ accent sync --dry-run --sync-type=force`, - `$ accent sync --add-translations --merge-type=smart --order-key=key --version=v0.23`, + `$ accent sync --add-translations --merge-type=smart --order-key=key --version=v0.23` ]; static args = []; @@ -37,36 +37,40 @@ export default class Sync extends Command { static flags = { 'add-translations': flags.boolean({ description: - 'Add translations in Accent to help translators if you already have translated strings locally', + 'Add translations in Accent to help translators if you already have translated strings locally' }), - 'dry-run': flags.boolean({ + 'no-local-write': flags.boolean({ default: false, description: - 'Do not write the file from the export _after_ the operation', + 'Do not write to the local files _after_ the sync. Warning: This option could lead to a mismatch between the source of truth (your code repository) and Accent' + }), + 'dry-run': flags.boolean({ + default: false, + description: 'Do not commit the changes in Accent' }), 'merge-type': flags.string({ default: 'passive', description: 'Algorithm to use on existing strings when adding translation', - options: ['smart', 'passive', 'force'], + options: ['smart', 'passive', 'force'] }), 'order-by': flags.string({ default: 'index', description: 'Will be used in the export call as the order of the keys', - options: ['index', 'key'], + options: ['index', 'key'] }), 'sync-type': flags.string({ default: 'smart', description: 'Algorithm to use on existing strings when syncing the main language', - options: ['smart', 'passive'], + options: ['smart', 'passive'] }), version: flags.string({ default: '', description: - 'Sync a specific version, the tag needs to exists in Accent first', + 'Sync a specific version, the tag needs to exists in Accent first' }), - config: configFlag, + config: configFlag }; async run() { @@ -110,6 +114,8 @@ export default class Sync extends Command { return; } + if (flags['no-local-write']) return; + const formatter = new DocumentExportFormatter(); // From all the documentConfigs, do the export, write to local file and log the results. diff --git a/cli/src/services/document-jipt-paths-fetcher.ts b/cli/src/services/document-jipt-paths-fetcher.ts index e14fc5ac5..c35728f5d 100644 --- a/cli/src/services/document-jipt-paths-fetcher.ts +++ b/cli/src/services/document-jipt-paths-fetcher.ts @@ -20,7 +20,7 @@ export default class DocumentJiptPathsFetcher { return { documentPath: path, language: pseudoLanguageName, - path: parsedTarget, + path: parsedTarget }; }); } diff --git a/cli/src/services/document.ts b/cli/src/services/document.ts index 7e7424c55..e64ffa329 100644 --- a/cli/src/services/document.ts +++ b/cli/src/services/document.ts @@ -65,7 +65,7 @@ export default class Document { const response = await fetch(url, { body: formData, headers: this.authorizationHeader(), - method: 'POST', + method: 'POST' }); await throwOnServerError(response); @@ -87,7 +87,7 @@ export default class Document { const response = await fetch(url, { body: formData, headers: this.authorizationHeader(), - method: 'POST', + method: 'POST' }); await throwOnServerError(response); @@ -114,7 +114,7 @@ export default class Document { const response = await fetch(url, { body: formData, headers: this.authorizationHeader(), - method: 'POST', + method: 'POST' }); await throwOnServerError(response); @@ -146,7 +146,7 @@ export default class Document { const response = await fetch(url, { body: formData, headers: this.authorizationHeader(), - method: 'POST', + method: 'POST' }); await throwOnServerError(response); @@ -174,7 +174,7 @@ export default class Document { ['document_path', documentPath], ['document_format', this.config.format], ['order_by', options['order-by']], - ['language', language], + ['language', language] ]; if (options.version) query.push(['version', options.version]); @@ -182,7 +182,7 @@ export default class Document { const url = `${this.apiUrl}/export?${this.encodeQuery(query)}`; const response = await fetch(url, { - headers: this.authorizationHeader(), + headers: this.authorizationHeader() }); await throwOnServerError(response); @@ -192,7 +192,7 @@ export default class Document { async exportJipt(file: string, documentPath: string) { const query = [ ['document_path', documentPath], - ['document_format', this.config.format], + ['document_format', this.config.format] ]; if (this.projectId) query.push(['project_id', this.projectId]); @@ -200,7 +200,7 @@ export default class Document { const url = `${this.apiUrl}/jipt-export?${this.encodeQuery(query)}`; const response = await fetch(url, { - headers: this.authorizationHeader(), + headers: this.authorizationHeader() }); await throwOnServerError(response); diff --git a/cli/src/services/formatters/hook-runner.ts b/cli/src/services/formatters/hook-runner.ts index ffd7ba626..4840b769d 100644 --- a/cli/src/services/formatters/hook-runner.ts +++ b/cli/src/services/formatters/hook-runner.ts @@ -8,10 +8,30 @@ const capitalizeFirstLetter = (str: string) => export default class HookRunnerFomatter { log(name: string, commands: string[]) { const operation = capitalizeFirstLetter(decamelize(name, {separator: ' '})); - console.log(chalk.yellow('➤ '), chalk.bold(chalk.yellow(`${operation}:`))); + console.log(chalk.yellow('➤'), chalk.bold(chalk.yellow(`${operation}:`))); + commands.forEach((command) => { - console.log(' ', chalk.yellow(command)); + console.log(' ', chalk.yellow(command)); + }); + + console.log(''); + } + + error(command: string, errors: string[]) { + console.log(chalk.red('⚠'), chalk.bold(chalk.red(`${command}:`))); + + errors.forEach((error) => { + console.log(' ', chalk.red(error)); }); + + console.log(''); + } + + success(command: string, message: string) { + console.log(chalk.green('✓'), chalk.bold(chalk.green(`${command}:`))); + + console.log(' ', chalk.green(message)); + console.log(''); } } diff --git a/cli/src/services/formatters/project-stats.ts b/cli/src/services/formatters/project-stats.ts index 3588730c8..5b8e006f7 100644 --- a/cli/src/services/formatters/project-stats.ts +++ b/cli/src/services/formatters/project-stats.ts @@ -8,13 +8,13 @@ import { Document, Project, Revision, - Version, + Version } from '../../types/project'; // Services import { fetchFromRevision, - fetchNameFromRevision, + fetchNameFromRevision } from '../revision-slug-fetcher'; import Base from './base'; diff --git a/cli/src/services/hook-runner.ts b/cli/src/services/hook-runner.ts index cbb0bc196..1ba8255c1 100644 --- a/cli/src/services/hook-runner.ts +++ b/cli/src/services/hook-runner.ts @@ -23,9 +23,25 @@ export default class HookRunner { const hooks = this.hooks[name]; if (hooks) { - new Formatter().log(name, hooks); - - hooks.forEach(execSync); + const formatter = new Formatter(); + formatter.log(name, hooks); + + hooks.forEach((hook) => { + try { + const output = execSync(hook, {stdio: 'pipe'}).toString(); + if (output.length > 0) formatter.success(hook, output); + } catch (error: any) { + const output = error.stderr.toString(); + + if (output.length > 0) { + formatter.error(hook, [output]); + } else { + formatter.error(hook, [`Exit status: ${error.status}`]); + } + + process.exit(error.status); + } + }); } return this.document.refreshPaths(); diff --git a/cli/src/services/project-fetcher.ts b/cli/src/services/project-fetcher.ts index 031081d12..e72b9f8e4 100644 --- a/cli/src/services/project-fetcher.ts +++ b/cli/src/services/project-fetcher.ts @@ -115,9 +115,9 @@ export default class ProjectFetcher { body: JSON.stringify({query, variables}), headers: { 'Content-Type': 'application/json', - authorization: `Bearer ${config.apiKey}`, + authorization: `Bearer ${config.apiKey}` }, - method: 'POST', + method: 'POST' }); } } diff --git a/cli/src/types/document-config.ts b/cli/src/types/document-config.ts index 9ba9df482..8b0bf5bde 100644 --- a/cli/src/types/document-config.ts +++ b/cli/src/types/document-config.ts @@ -4,14 +4,14 @@ export enum Hooks { beforeExport = 'beforeExport', afterExport = 'afterExport', beforeSync = 'beforeSync', - afterSync = 'afterSync', + afterSync = 'afterSync' } export enum NamePattern { file = 'file', fileWithSlugSuffix = 'fileWithSlugSuffix', parentDirectory = 'parentDirectory', - fileWithParentDirectory = 'fileWithParentDirectory', + fileWithParentDirectory = 'fileWithParentDirectory' } export interface HookConfig { diff --git a/config/config.exs b/config/config.exs index 1596009fc..609ea7c05 100644 --- a/config/config.exs +++ b/config/config.exs @@ -13,17 +13,14 @@ if config_env() == :dev do end if config_env() == :test do - events = ~w(sync add_translations create_collaborator create_comment complete_review new_conflicts) - - config :accent, Accent.Hook, outbounds: [{Accent.Hook.Outbounds.Mock, events: events}] + config :accent, Accent.Hook, outbounds: [Accent.Hook.Outbounds.Mock] else config :accent, Accent.Hook, outbounds: [ - {Accent.Hook.Outbounds.Discord, events: ~w(sync complete_review new_conflicts)}, - {Accent.Hook.Outbounds.Email, events: ~w(create_collaborator create_comment)}, - {Accent.Hook.Outbounds.Slack, events: ~w(sync complete_review new_conflicts)}, - {Accent.Hook.Outbounds.Websocket, - events: ~w(sync create_collaborator create_comment complete_review new_conflicts)} + Accent.Hook.Outbounds.Discord, + Accent.Hook.Outbounds.Email, + Accent.Hook.Outbounds.Slack, + Accent.Hook.Outbounds.Websocket ] end diff --git a/config/dev.exs b/config/dev.exs index 2521ba125..8929d095a 100644 --- a/config/dev.exs +++ b/config/dev.exs @@ -9,6 +9,11 @@ watchers = "run", "build-dev", cd: Path.expand("../webapp", __DIR__) + ], + npm: [ + "run", + "watch-production-inline", + cd: Path.expand("../jipt", __DIR__) ] ] end diff --git a/config/runtime.exs b/config/runtime.exs index 065c7f72c..94efb0bd8 100644 --- a/config/runtime.exs +++ b/config/runtime.exs @@ -31,17 +31,23 @@ else debug_errors: get_env("DEBUG_ERRORS", :boolean) end -ecto_ipv6? = get_env("ECTO_IPV6", :boolean) - -config :accent, Accent.Repo, - timeout: get_env("DATABASE_TIMEOUT", :integer) || 29_000, - queue_target: get_env("DATABASE_QUEUE_TARGET", :integer) || 500, - queue_interval: get_env("DATABASE_QUEUE_INTERVAL", :integer) || 2000, - pool_size: get_env("DATABASE_POOL_SIZE", :integer), - ssl: get_env("DATABASE_SSL", :boolean), - ssl_opts: [verify: :verify_none], - url: get_env("DATABASE_URL") || "postgres://localhost/accent_development", - socket_options: if(ecto_ipv6?, do: [:inet6], else: []) +if config_env() === :test do + config :accent, Accent.Repo, + pool_size: System.schedulers_online() * 2, + url: get_env("DATABASE_URL") +else + ecto_ipv6? = get_env("ECTO_IPV6", :boolean) + + config :accent, Accent.Repo, + timeout: get_env("DATABASE_TIMEOUT", :integer) || 29_000, + queue_target: get_env("DATABASE_QUEUE_TARGET", :integer) || 500, + queue_interval: get_env("DATABASE_QUEUE_INTERVAL", :integer) || 2000, + pool_size: get_env("DATABASE_POOL_SIZE", :integer), + ssl: get_env("DATABASE_SSL", :boolean), + ssl_opts: [verify: :verify_none], + url: get_env("DATABASE_URL") || "postgres://localhost/accent_development", + socket_options: if(ecto_ipv6?, do: [:inet6], else: []) +end config :accent, Accent.MachineTranslations, default_providers_config: %{ @@ -49,6 +55,11 @@ config :accent, Accent.MachineTranslations, "deepl" => %{"key" => get_env("DEEPL_TRANSLATIONS_KEY")} } +config :accent, Accent.Prompts, + default_providers_config: %{ + "openai" => %{"key" => get_env("OPENAI_API_KEY")} + } + config :accent, LanguageTool, languages: get_env("LANGUAGE_TOOL_LANGUAGES", :comma_separated_list) providers = [] @@ -88,6 +99,11 @@ providers = providers = if get_env("AUTH0_CLIENT_ID"), do: [{:auth0, {Ueberauth.Strategy.Auth0, []}} | providers], else: providers +providers = + if get_env("OIDC_CLIENT_ID"), + do: [{:oidc, {Ueberauth.Strategy.OIDC, [default: [provider: :default_oidc, uid_field: :sub]]}} | providers], + else: providers + providers = if get_env("DUMMY_LOGIN_ENABLED"), do: [{:dummy, {Accent.Auth.Ueberauth.DummyStrategy, []}} | providers], @@ -129,6 +145,18 @@ config :ueberauth, Ueberauth.Strategy.Microsoft.OAuth, client_secret: get_env("MICROSOFT_CLIENT_SECRET"), tenant_id: get_env("MICROSOFT_TENANT_ID") +config :ueberauth, Ueberauth.Strategy.OIDC, + default_oidc: [ + fetch_userinfo: true, + uid_field: get_env("OIDC_UID_FIELD") || "sub", + client_id: get_env("OIDC_CLIENT_ID"), + client_secret: get_env("OIDC_CLIENT_SECRET"), + discovery_document_uri: get_env("OIDC_DISCOVERY_URI"), + redirect_uri: "#{static_uri}/auth/oidc/callback", + response_type: "code", + scope: get_env("OIDC_SCOPE") || "openid profile email" + ] + config :accent, Accent.WebappView, path: "priv/static/webapp/index.html", sentry_dsn: get_env("WEBAPP_SENTRY_DSN") || "", diff --git a/docker-compose.yml b/docker-compose.yml index 399c6476a..05a282d95 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -1,8 +1,8 @@ version: '3.7' services: application: - image: mirego/accent container_name: accent + network_mode: "host" build: . ports: - 4000:4000 @@ -10,9 +10,9 @@ services: - postgresql environment: - PORT=4000 - - DATABASE_URL=postgres://postgres@postgresql:5432/accent_development + - DATABASE_URL=postgres://postgres@localhost:5432/accent_development postgresql: - image: postgres:10.3 + image: postgres:15.6 container_name: accent-postgres environment: - POSTGRES_DB=accent_development diff --git a/jipt/index.html b/jipt/index.html index 631a0ad7c..98d1feaa1 100644 --- a/jipt/index.html +++ b/jipt/index.html @@ -1,7 +1,7 @@
-

{^demo-accent.foo@session}

+

{^accept-ticket.error@ticketss}

diff --git a/jipt/package-lock.json b/jipt/package-lock.json index b0231b909..1d6bd0274 100644 --- a/jipt/package-lock.json +++ b/jipt/package-lock.json @@ -1,35 +1,44 @@ { "name": "jipt", "version": "1.0.0", - "lockfileVersion": 1, + "lockfileVersion": 3, "requires": true, - "dependencies": { - "@babel/code-frame": { + "packages": { + "": { + "name": "jipt", + "version": "1.0.0", + "license": "ISC", + "devDependencies": { + "parcel-bundler": "1.12.4", + "typescript": "3.8.3" + } + }, + "node_modules/@babel/code-frame": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.8.3.tgz", "integrity": "sha512-a9gxpmdXtZEInkCSHUJDLHZVBgb1QS0jhss4cPP93EW7s+uC5bikET2twEF3KV+7rDblJcmNvTR7VJejqd2C2g==", "dev": true, - "requires": { + "dependencies": { "@babel/highlight": "^7.8.3" } }, - "@babel/compat-data": { + "node_modules/@babel/compat-data": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.9.0.tgz", "integrity": "sha512-zeFQrr+284Ekvd9e7KAX954LkapWiOmQtsfHirhxqfdlX6MEC32iRE+pqUGlYIBchdevaCwvzxWGSy/YBNI85g==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.9.1", "invariant": "^2.2.4", "semver": "^5.5.0" } }, - "@babel/core": { + "node_modules/@babel/core": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.9.0.tgz", "integrity": "sha512-kWc7L0fw1xwvI0zi8OKVBuxRVefwGOrKSQMvrQ3dW+bIIavBY3/NpXmpjMy7bQnLgwgzWQZ8TlM57YHpHNHz4w==", "dev": true, - "requires": { + "dependencies": { "@babel/code-frame": "^7.8.3", "@babel/generator": "^7.9.0", "@babel/helper-module-transforms": "^7.9.0", @@ -47,182 +56,203 @@ "semver": "^5.4.1", "source-map": "^0.5.0" }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@babel/core/node_modules/json5": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", + "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", + "dev": true, "dependencies": { - "json5": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", - "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", - "dev": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } + "minimist": "^1.2.5" + }, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/@babel/core/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true, + "engines": { + "node": ">=0.10.0" } }, - "@babel/generator": { + "node_modules/@babel/generator": { "version": "7.9.4", "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.9.4.tgz", "integrity": "sha512-rjP8ahaDy/ouhrvCoU1E5mqaitWrxwuNGU+dy1EpaoK48jZay4MdkskKGIMHLZNewg8sAsqpGSREJwP0zH3YQA==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.9.0", "jsesc": "^2.5.1", "lodash": "^4.17.13", "source-map": "^0.5.0" - }, - "dependencies": { - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } } }, - "@babel/helper-annotate-as-pure": { + "node_modules/@babel/generator/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.8.3.tgz", "integrity": "sha512-6o+mJrZBxOoEX77Ezv9zwW7WV8DdluouRKNY/IR5u/YTMuKHgugHOzYWlYvYLpLA9nPsQCAAASpCIbjI9Mv+Uw==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-builder-binary-assignment-operator-visitor": { + "node_modules/@babel/helper-builder-binary-assignment-operator-visitor": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.8.3.tgz", "integrity": "sha512-5eFOm2SyFPK4Rh3XMMRDjN7lBH0orh3ss0g3rTYZnBQ+r6YPj7lgDyCvPphynHvUrobJmeMignBr6Acw9mAPlw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-explode-assignable-expression": "^7.8.3", "@babel/types": "^7.8.3" } }, - "@babel/helper-builder-react-jsx": { + "node_modules/@babel/helper-builder-react-jsx": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/helper-builder-react-jsx/-/helper-builder-react-jsx-7.9.0.tgz", "integrity": "sha512-weiIo4gaoGgnhff54GQ3P5wsUQmnSwpkvU0r6ZHq6TzoSzKy4JxHEgnxNytaKbov2a9z/CVNyzliuCOUPEX3Jw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/types": "^7.9.0" } }, - "@babel/helper-builder-react-jsx-experimental": { + "node_modules/@babel/helper-builder-react-jsx-experimental": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/helper-builder-react-jsx-experimental/-/helper-builder-react-jsx-experimental-7.9.0.tgz", "integrity": "sha512-3xJEiyuYU4Q/Ar9BsHisgdxZsRlsShMe90URZ0e6przL26CCs8NJbDoxH94kKT17PcxlMhsCAwZd90evCo26VQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/helper-module-imports": "^7.8.3", "@babel/types": "^7.9.0" } }, - "@babel/helper-compilation-targets": { + "node_modules/@babel/helper-compilation-targets": { "version": "7.8.7", "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.8.7.tgz", "integrity": "sha512-4mWm8DCK2LugIS+p1yArqvG1Pf162upsIsjE7cNBjez+NjliQpVhj20obE520nao0o14DaTnFJv+Fw5a0JpoUw==", "dev": true, - "requires": { + "dependencies": { "@babel/compat-data": "^7.8.6", "browserslist": "^4.9.1", "invariant": "^2.2.4", "levenary": "^1.1.1", "semver": "^5.5.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" } }, - "@babel/helper-create-regexp-features-plugin": { + "node_modules/@babel/helper-create-regexp-features-plugin": { "version": "7.8.8", "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.8.8.tgz", "integrity": "sha512-LYVPdwkrQEiX9+1R29Ld/wTrmQu1SSKYnuOk3g0CkcZMA1p0gsNxJFj/3gBdaJ7Cg0Fnek5z0DsMULePP7Lrqg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/helper-regex": "^7.8.3", "regexpu-core": "^4.7.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" } }, - "@babel/helper-define-map": { + "node_modules/@babel/helper-define-map": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.8.3.tgz", "integrity": "sha512-PoeBYtxoZGtct3md6xZOCWPcKuMuk3IHhgxsRRNtnNShebf4C8YonTSblsK4tvDbm+eJAw2HAPOfCr+Q/YRG/g==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-function-name": "^7.8.3", "@babel/types": "^7.8.3", "lodash": "^4.17.13" } }, - "@babel/helper-explode-assignable-expression": { + "node_modules/@babel/helper-explode-assignable-expression": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.8.3.tgz", "integrity": "sha512-N+8eW86/Kj147bO9G2uclsg5pwfs/fqqY5rwgIL7eTBklgXjcOJ3btzS5iM6AitJcftnY7pm2lGsrJVYLGjzIw==", "dev": true, - "requires": { + "dependencies": { "@babel/traverse": "^7.8.3", "@babel/types": "^7.8.3" } }, - "@babel/helper-function-name": { + "node_modules/@babel/helper-function-name": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.8.3.tgz", "integrity": "sha512-BCxgX1BC2hD/oBlIFUgOCQDOPV8nSINxCwM3o93xP4P9Fq6aV5sgv2cOOITDMtCfQ+3PvHp3l689XZvAM9QyOA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-get-function-arity": "^7.8.3", "@babel/template": "^7.8.3", "@babel/types": "^7.8.3" } }, - "@babel/helper-get-function-arity": { + "node_modules/@babel/helper-get-function-arity": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.8.3.tgz", "integrity": "sha512-FVDR+Gd9iLjUMY1fzE2SR0IuaJToR4RkCDARVfsBBPSP53GEqSFjD8gNyxg246VUyc/ALRxFaAK8rVG7UT7xRA==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-hoist-variables": { + "node_modules/@babel/helper-hoist-variables": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.8.3.tgz", "integrity": "sha512-ky1JLOjcDUtSc+xkt0xhYff7Z6ILTAHKmZLHPxAhOP0Nd77O+3nCsd6uSVYur6nJnCI029CrNbYlc0LoPfAPQg==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-member-expression-to-functions": { + "node_modules/@babel/helper-member-expression-to-functions": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.8.3.tgz", "integrity": "sha512-fO4Egq88utkQFjbPrSHGmGLFqmrshs11d46WI+WZDESt7Wu7wN2G2Iu+NMMZJFDOVRHAMIkB5SNh30NtwCA7RA==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-module-imports": { + "node_modules/@babel/helper-module-imports": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.8.3.tgz", "integrity": "sha512-R0Bx3jippsbAEtzkpZ/6FIiuzOURPcMjHp+Z6xPe6DtApDJx+w7UYyOLanZqO8+wKR9G10s/FmHXvxaMd9s6Kg==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-module-transforms": { + "node_modules/@babel/helper-module-transforms": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.9.0.tgz", "integrity": "sha512-0FvKyu0gpPfIQ8EkxlrAydOWROdHpBmiCiRwLkUiBGhCUPRRbVD2/tm3sFr/c/GWFrQ/ffutGUAnx7V0FzT2wA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-module-imports": "^7.8.3", "@babel/helper-replace-supers": "^7.8.6", "@babel/helper-simple-access": "^7.8.3", @@ -232,36 +262,36 @@ "lodash": "^4.17.13" } }, - "@babel/helper-optimise-call-expression": { + "node_modules/@babel/helper-optimise-call-expression": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.8.3.tgz", "integrity": "sha512-Kag20n86cbO2AvHca6EJsvqAd82gc6VMGule4HwebwMlwkpXuVqrNRj6CkCV2sKxgi9MyAUnZVnZ6lJ1/vKhHQ==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-plugin-utils": { + "node_modules/@babel/helper-plugin-utils": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.8.3.tgz", "integrity": "sha512-j+fq49Xds2smCUNYmEHF9kGNkhbet6yVIBp4e6oeQpH1RUs/Ir06xUKzDjDkGcaaokPiTNs2JBWHjaE4csUkZQ==", "dev": true }, - "@babel/helper-regex": { + "node_modules/@babel/helper-regex": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.8.3.tgz", "integrity": "sha512-BWt0QtYv/cg/NecOAZMdcn/waj/5P26DR4mVLXfFtDokSR6fyuG0Pj+e2FqtSME+MqED1khnSMulkmGl8qWiUQ==", "dev": true, - "requires": { + "dependencies": { "lodash": "^4.17.13" } }, - "@babel/helper-remap-async-to-generator": { + "node_modules/@babel/helper-remap-async-to-generator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.8.3.tgz", "integrity": "sha512-kgwDmw4fCg7AVgS4DukQR/roGp+jP+XluJE5hsRZwxCYGg+Rv9wSGErDWhlI90FODdYfd4xG4AQRiMDjjN0GzA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/helper-wrap-function": "^7.8.3", "@babel/template": "^7.8.3", @@ -269,318 +299,408 @@ "@babel/types": "^7.8.3" } }, - "@babel/helper-replace-supers": { + "node_modules/@babel/helper-replace-supers": { "version": "7.8.6", "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.8.6.tgz", "integrity": "sha512-PeMArdA4Sv/Wf4zXwBKPqVj7n9UF/xg6slNRtZW84FM7JpE1CbG8B612FyM4cxrf4fMAMGO0kR7voy1ForHHFA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-member-expression-to-functions": "^7.8.3", "@babel/helper-optimise-call-expression": "^7.8.3", "@babel/traverse": "^7.8.6", "@babel/types": "^7.8.6" } }, - "@babel/helper-simple-access": { + "node_modules/@babel/helper-simple-access": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.8.3.tgz", "integrity": "sha512-VNGUDjx5cCWg4vvCTR8qQ7YJYZ+HBjxOgXEl7ounz+4Sn7+LMD3CFrCTEU6/qXKbA2nKg21CwhhBzO0RpRbdCw==", "dev": true, - "requires": { + "dependencies": { "@babel/template": "^7.8.3", "@babel/types": "^7.8.3" } }, - "@babel/helper-split-export-declaration": { + "node_modules/@babel/helper-split-export-declaration": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.8.3.tgz", "integrity": "sha512-3x3yOeyBhW851hroze7ElzdkeRXQYQbFIb7gLK1WQYsw2GWDay5gAJNw1sWJ0VFP6z5J1whqeXH/WCdCjZv6dA==", "dev": true, - "requires": { + "dependencies": { "@babel/types": "^7.8.3" } }, - "@babel/helper-validator-identifier": { + "node_modules/@babel/helper-validator-identifier": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.9.0.tgz", "integrity": "sha512-6G8bQKjOh+of4PV/ThDm/rRqlU7+IGoJuofpagU5GlEl29Vv0RGqqt86ZGRV8ZuSOY3o+8yXl5y782SMcG7SHw==", "dev": true }, - "@babel/helper-wrap-function": { + "node_modules/@babel/helper-wrap-function": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.8.3.tgz", "integrity": "sha512-LACJrbUET9cQDzb6kG7EeD7+7doC3JNvUgTEQOx2qaO1fKlzE/Bf05qs9w1oXQMmXlPO65lC3Tq9S6gZpTErEQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-function-name": "^7.8.3", "@babel/template": "^7.8.3", "@babel/traverse": "^7.8.3", "@babel/types": "^7.8.3" } }, - "@babel/helpers": { + "node_modules/@babel/helpers": { "version": "7.9.2", "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.9.2.tgz", "integrity": "sha512-JwLvzlXVPjO8eU9c/wF9/zOIN7X6h8DYf7mG4CiFRZRvZNKEF5dQ3H3V+ASkHoIB3mWhatgl5ONhyqHRI6MppA==", "dev": true, - "requires": { + "dependencies": { "@babel/template": "^7.8.3", "@babel/traverse": "^7.9.0", "@babel/types": "^7.9.0" } }, - "@babel/highlight": { + "node_modules/@babel/highlight": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.9.0.tgz", "integrity": "sha512-lJZPilxX7Op3Nv/2cvFdnlepPXDxi29wxteT57Q965oc5R9v86ztx0jfxVrTcBk8C2kcPkkDa2Z4T3ZsPPVWsQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-validator-identifier": "^7.9.0", "chalk": "^2.0.0", "js-tokens": "^4.0.0" } }, - "@babel/parser": { + "node_modules/@babel/parser": { "version": "7.9.4", "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.9.4.tgz", "integrity": "sha512-bC49otXX6N0/VYhgOMh4gnP26E9xnDZK3TmbNpxYzzz9BQLBosQwfyOe9/cXUU3txYhTzLCbcqd5c8y/OmCjHA==", - "dev": true + "dev": true, + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } }, - "@babel/plugin-proposal-async-generator-functions": { + "node_modules/@babel/plugin-proposal-async-generator-functions": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.8.3.tgz", "integrity": "sha512-NZ9zLv848JsV3hs8ryEh7Uaz/0KsmPLqv0+PdkDJL1cJy0K4kOCFa8zc1E3mp+RHPQcpdfb/6GovEsW4VDrOMw==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-async-generator-functions instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/helper-remap-async-to-generator": "^7.8.3", "@babel/plugin-syntax-async-generators": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-dynamic-import": { + "node_modules/@babel/plugin-proposal-dynamic-import": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.8.3.tgz", "integrity": "sha512-NyaBbyLFXFLT9FP+zk0kYlUlA8XtCUbehs67F0nnEg7KICgMc2mNkIeu9TYhKzyXMkrapZFwAhXLdnt4IYHy1w==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-dynamic-import instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-dynamic-import": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-json-strings": { + "node_modules/@babel/plugin-proposal-json-strings": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.8.3.tgz", "integrity": "sha512-KGhQNZ3TVCQG/MjRbAUwuH+14y9q0tpxs1nWWs3pbSleRdDro9SAMMDyye8HhY1gqZ7/NqIc8SKhya0wRDgP1Q==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-json-strings instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-json-strings": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-nullish-coalescing-operator": { + "node_modules/@babel/plugin-proposal-nullish-coalescing-operator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.8.3.tgz", "integrity": "sha512-TS9MlfzXpXKt6YYomudb/KU7nQI6/xnapG6in1uZxoxDghuSMZsPb6D2fyUwNYSAp4l1iR7QtFOjkqcRYcUsfw==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-nullish-coalescing-operator instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-numeric-separator": { + "node_modules/@babel/plugin-proposal-numeric-separator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.8.3.tgz", "integrity": "sha512-jWioO1s6R/R+wEHizfaScNsAx+xKgwTLNXSh7tTC4Usj3ItsPEhYkEpU4h+lpnBwq7NBVOJXfO6cRFYcX69JUQ==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-numeric-separator instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-numeric-separator": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-object-rest-spread": { + "node_modules/@babel/plugin-proposal-object-rest-spread": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.9.0.tgz", "integrity": "sha512-UgqBv6bjq4fDb8uku9f+wcm1J7YxJ5nT7WO/jBr0cl0PLKb7t1O6RNR1kZbjgx2LQtsDI9hwoQVmn0yhXeQyow==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-object-rest-spread instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-object-rest-spread": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-optional-catch-binding": { + "node_modules/@babel/plugin-proposal-optional-catch-binding": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.8.3.tgz", "integrity": "sha512-0gkX7J7E+AtAw9fcwlVQj8peP61qhdg/89D5swOkjYbkboA2CVckn3kiyum1DE0wskGb7KJJxBdyEBApDLLVdw==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-optional-catch-binding instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-optional-catch-binding": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-optional-chaining": { + "node_modules/@babel/plugin-proposal-optional-chaining": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.9.0.tgz", "integrity": "sha512-NDn5tu3tcv4W30jNhmc2hyD5c56G6cXx4TesJubhxrJeCvuuMpttxr0OnNCqbZGhFjLrg+NIhxxC+BK5F6yS3w==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-optional-chaining instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-optional-chaining": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-proposal-unicode-property-regex": { + "node_modules/@babel/plugin-proposal-unicode-property-regex": { "version": "7.8.8", "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.8.8.tgz", "integrity": "sha512-EVhjVsMpbhLw9ZfHWSx2iy13Q8Z/eg8e8ccVWt23sWQK5l1UdkoLJPN5w69UA4uITGBnEZD2JOe4QOHycYKv8A==", + "deprecated": "This proposal has been merged to the ECMAScript standard and thus this plugin is no longer maintained. Please use @babel/plugin-transform-unicode-property-regex instead.", "dev": true, - "requires": { + "dependencies": { "@babel/helper-create-regexp-features-plugin": "^7.8.8", "@babel/helper-plugin-utils": "^7.8.3" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-async-generators": { + "node_modules/@babel/plugin-syntax-async-generators": { "version": "7.8.4", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-async-generators/-/plugin-syntax-async-generators-7.8.4.tgz", "integrity": "sha512-tycmZxkGfZaxhMRbXlPXuVFpdWlXpir2W4AMhSJgRKzk/eDlIXOhb2LHWoLpDF7TEHylV5zNhykX6KAgHJmTNw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-dynamic-import": { + "node_modules/@babel/plugin-syntax-dynamic-import": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-dynamic-import/-/plugin-syntax-dynamic-import-7.8.3.tgz", "integrity": "sha512-5gdGbFon+PszYzqs83S3E5mpi7/y/8M9eC90MRTZfduQOYW76ig6SOSPNe41IG5LoP3FGBn2N0RjVDSQiS94kQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-flow": { + "node_modules/@babel/plugin-syntax-flow": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-flow/-/plugin-syntax-flow-7.8.3.tgz", "integrity": "sha512-innAx3bUbA0KSYj2E2MNFSn9hiCeowOFLxlsuhXzw8hMQnzkDomUr9QCD7E9VF60NmnG1sNTuuv6Qf4f8INYsg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-json-strings": { + "node_modules/@babel/plugin-syntax-json-strings": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-json-strings/-/plugin-syntax-json-strings-7.8.3.tgz", "integrity": "sha512-lY6kdGpWHvjoe2vk4WrAapEuBR69EMxZl+RoGRhrFGNYVK8mOPAW8VfbT/ZgrFbXlDNiiaxQnAtgVCZ6jv30EA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-jsx": { + "node_modules/@babel/plugin-syntax-jsx": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-jsx/-/plugin-syntax-jsx-7.8.3.tgz", "integrity": "sha512-WxdW9xyLgBdefoo0Ynn3MRSkhe5tFVxxKNVdnZSh318WrG2e2jH+E9wd/++JsqcLJZPfz87njQJ8j2Upjm0M0A==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-nullish-coalescing-operator": { + "node_modules/@babel/plugin-syntax-nullish-coalescing-operator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-nullish-coalescing-operator/-/plugin-syntax-nullish-coalescing-operator-7.8.3.tgz", "integrity": "sha512-aSff4zPII1u2QD7y+F8oDsz19ew4IGEJg9SVW+bqwpwtfFleiQDMdzA/R+UlWDzfnHFCxxleFT0PMIrR36XLNQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-numeric-separator": { + "node_modules/@babel/plugin-syntax-numeric-separator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.8.3.tgz", "integrity": "sha512-H7dCMAdN83PcCmqmkHB5dtp+Xa9a6LKSvA2hiFBC/5alSHxM5VgWZXFqDi0YFe8XNGT6iCa+z4V4zSt/PdZ7Dw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-object-rest-spread": { + "node_modules/@babel/plugin-syntax-object-rest-spread": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-object-rest-spread/-/plugin-syntax-object-rest-spread-7.8.3.tgz", "integrity": "sha512-XoqMijGZb9y3y2XskN+P1wUGiVwWZ5JmoDRwx5+3GmEplNyVM2s2Dg8ILFQm8rWM48orGy5YpI5Bl8U1y7ydlA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-optional-catch-binding": { + "node_modules/@babel/plugin-syntax-optional-catch-binding": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-catch-binding/-/plugin-syntax-optional-catch-binding-7.8.3.tgz", "integrity": "sha512-6VPD0Pc1lpTqw0aKoeRTMiB+kWhAoT24PA+ksWSBrFtl5SIRVpZlwN3NNPQjehA2E/91FV3RjLWoVTglWcSV3Q==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-optional-chaining": { + "node_modules/@babel/plugin-syntax-optional-chaining": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-optional-chaining/-/plugin-syntax-optional-chaining-7.8.3.tgz", "integrity": "sha512-KoK9ErH1MBlCPxV0VANkXW2/dw4vlbGDrFgz8bmUsBGYkFRcbRwMh6cIJubdPrkxRwuGdtCk0v/wPTKbQgBjkg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-syntax-top-level-await": { + "node_modules/@babel/plugin-syntax-top-level-await": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.8.3.tgz", "integrity": "sha512-kwj1j9lL/6Wd0hROD3b/OZZ7MSrZLqqn9RAZ5+cYYsflQ9HZBIKCUkr3+uL1MEJ1NePiUbf98jjiMQSv0NMR9g==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-arrow-functions": { + "node_modules/@babel/plugin-transform-arrow-functions": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.8.3.tgz", "integrity": "sha512-0MRF+KC8EqH4dbuITCWwPSzsyO3HIWWlm30v8BbbpOrS1B++isGxPnnuq/IZvOX5J2D/p7DQalQm+/2PnlKGxg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-async-to-generator": { + "node_modules/@babel/plugin-transform-async-to-generator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.8.3.tgz", "integrity": "sha512-imt9tFLD9ogt56Dd5CI/6XgpukMwd/fLGSrix2httihVe7LOGVPhyhMh1BU5kDM7iHD08i8uUtmV2sWaBFlHVQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-module-imports": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3", "@babel/helper-remap-async-to-generator": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-block-scoped-functions": { + "node_modules/@babel/plugin-transform-block-scoped-functions": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.8.3.tgz", "integrity": "sha512-vo4F2OewqjbB1+yaJ7k2EJFHlTP3jR634Z9Cj9itpqNjuLXvhlVxgnjsHsdRgASR8xYDrx6onw4vW5H6We0Jmg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-block-scoping": { + "node_modules/@babel/plugin-transform-block-scoping": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.8.3.tgz", "integrity": "sha512-pGnYfm7RNRgYRi7bids5bHluENHqJhrV4bCZRwc5GamaWIIs07N4rZECcmJL6ZClwjDz1GbdMZFtPs27hTB06w==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "lodash": "^4.17.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-classes": { + "node_modules/@babel/plugin-transform-classes": { "version": "7.9.2", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.9.2.tgz", "integrity": "sha512-TC2p3bPzsfvSsqBZo0kJnuelnoK9O3welkUpqSqBQuBF6R5MN2rysopri8kNvtlGIb2jmUO7i15IooAZJjZuMQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/helper-define-map": "^7.8.3", "@babel/helper-function-name": "^7.8.3", @@ -589,287 +709,374 @@ "@babel/helper-replace-supers": "^7.8.6", "@babel/helper-split-export-declaration": "^7.8.3", "globals": "^11.1.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-computed-properties": { + "node_modules/@babel/plugin-transform-computed-properties": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.8.3.tgz", "integrity": "sha512-O5hiIpSyOGdrQZRQ2ccwtTVkgUDBBiCuK//4RJ6UfePllUTCENOzKxfh6ulckXKc0DixTFLCfb2HVkNA7aDpzA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-destructuring": { + "node_modules/@babel/plugin-transform-destructuring": { "version": "7.8.8", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.8.8.tgz", "integrity": "sha512-eRJu4Vs2rmttFCdhPUM3bV0Yo/xPSdPw6ML9KHs/bjB4bLA5HXlbvYXPOD5yASodGod+krjYx21xm1QmL8dCJQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-dotall-regex": { + "node_modules/@babel/plugin-transform-dotall-regex": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.8.3.tgz", "integrity": "sha512-kLs1j9Nn4MQoBYdRXH6AeaXMbEJFaFu/v1nQkvib6QzTj8MZI5OQzqmD83/2jEM1z0DLilra5aWO5YpyC0ALIw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-create-regexp-features-plugin": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-duplicate-keys": { + "node_modules/@babel/plugin-transform-duplicate-keys": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.8.3.tgz", "integrity": "sha512-s8dHiBUbcbSgipS4SMFuWGqCvyge5V2ZeAWzR6INTVC3Ltjig/Vw1G2Gztv0vU/hRG9X8IvKvYdoksnUfgXOEQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-exponentiation-operator": { + "node_modules/@babel/plugin-transform-exponentiation-operator": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.8.3.tgz", "integrity": "sha512-zwIpuIymb3ACcInbksHaNcR12S++0MDLKkiqXHl3AzpgdKlFNhog+z/K0+TGW+b0w5pgTq4H6IwV/WhxbGYSjQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-builder-binary-assignment-operator-visitor": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-flow-strip-types": { + "node_modules/@babel/plugin-transform-flow-strip-types": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-flow-strip-types/-/plugin-transform-flow-strip-types-7.9.0.tgz", "integrity": "sha512-7Qfg0lKQhEHs93FChxVLAvhBshOPQDtJUTVHr/ZwQNRccCm4O9D79r9tVSoV8iNwjP1YgfD+e/fgHcPkN1qEQg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-flow": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-for-of": { + "node_modules/@babel/plugin-transform-for-of": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.9.0.tgz", "integrity": "sha512-lTAnWOpMwOXpyDx06N+ywmF3jNbafZEqZ96CGYabxHrxNX8l5ny7dt4bK/rGwAh9utyP2b2Hv7PlZh1AAS54FQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-function-name": { + "node_modules/@babel/plugin-transform-function-name": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.8.3.tgz", "integrity": "sha512-rO/OnDS78Eifbjn5Py9v8y0aR+aSYhDhqAwVfsTl0ERuMZyr05L1aFSCJnbv2mmsLkit/4ReeQ9N2BgLnOcPCQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-function-name": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-literals": { + "node_modules/@babel/plugin-transform-literals": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.8.3.tgz", "integrity": "sha512-3Tqf8JJ/qB7TeldGl+TT55+uQei9JfYaregDcEAyBZ7akutriFrt6C/wLYIer6OYhleVQvH/ntEhjE/xMmy10A==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-member-expression-literals": { + "node_modules/@babel/plugin-transform-member-expression-literals": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.8.3.tgz", "integrity": "sha512-3Wk2EXhnw+rP+IDkK6BdtPKsUE5IeZ6QOGrPYvw52NwBStw9V1ZVzxgK6fSKSxqUvH9eQPR3tm3cOq79HlsKYA==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-modules-amd": { + "node_modules/@babel/plugin-transform-modules-amd": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.9.0.tgz", "integrity": "sha512-vZgDDF003B14O8zJy0XXLnPH4sg+9X5hFBBGN1V+B2rgrB+J2xIypSN6Rk9imB2hSTHQi5OHLrFWsZab1GMk+Q==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-module-transforms": "^7.9.0", "@babel/helper-plugin-utils": "^7.8.3", "babel-plugin-dynamic-import-node": "^2.3.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-modules-commonjs": { + "node_modules/@babel/plugin-transform-modules-commonjs": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.9.0.tgz", "integrity": "sha512-qzlCrLnKqio4SlgJ6FMMLBe4bySNis8DFn1VkGmOcxG9gqEyPIOzeQrA//u0HAKrWpJlpZbZMPB1n/OPa4+n8g==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-module-transforms": "^7.9.0", "@babel/helper-plugin-utils": "^7.8.3", "@babel/helper-simple-access": "^7.8.3", "babel-plugin-dynamic-import-node": "^2.3.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-modules-systemjs": { + "node_modules/@babel/plugin-transform-modules-systemjs": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.9.0.tgz", "integrity": "sha512-FsiAv/nao/ud2ZWy4wFacoLOm5uxl0ExSQ7ErvP7jpoihLR6Cq90ilOFyX9UXct3rbtKsAiZ9kFt5XGfPe/5SQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-hoist-variables": "^7.8.3", "@babel/helper-module-transforms": "^7.9.0", "@babel/helper-plugin-utils": "^7.8.3", "babel-plugin-dynamic-import-node": "^2.3.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-modules-umd": { + "node_modules/@babel/plugin-transform-modules-umd": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.9.0.tgz", "integrity": "sha512-uTWkXkIVtg/JGRSIABdBoMsoIeoHQHPTL0Y2E7xf5Oj7sLqwVsNXOkNk0VJc7vF0IMBsPeikHxFjGe+qmwPtTQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-module-transforms": "^7.9.0", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-named-capturing-groups-regex": { + "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.8.3.tgz", "integrity": "sha512-f+tF/8UVPU86TrCb06JoPWIdDpTNSGGcAtaD9mLP0aYGA0OS0j7j7DHJR0GTFrUZPUU6loZhbsVZgTh0N+Qdnw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-create-regexp-features-plugin": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" } }, - "@babel/plugin-transform-new-target": { + "node_modules/@babel/plugin-transform-new-target": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.8.3.tgz", "integrity": "sha512-QuSGysibQpyxexRyui2vca+Cmbljo8bcRckgzYV4kRIsHpVeyeC3JDO63pY+xFZ6bWOBn7pfKZTqV4o/ix9sFw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-object-super": { + "node_modules/@babel/plugin-transform-object-super": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.8.3.tgz", "integrity": "sha512-57FXk+gItG/GejofIyLIgBKTas4+pEU47IXKDBWFTxdPd7F80H8zybyAY7UoblVfBhBGs2EKM+bJUu2+iUYPDQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/helper-replace-supers": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-parameters": { + "node_modules/@babel/plugin-transform-parameters": { "version": "7.9.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.9.3.tgz", "integrity": "sha512-fzrQFQhp7mIhOzmOtPiKffvCYQSK10NR8t6BBz2yPbeUHb9OLW8RZGtgDRBn8z2hGcwvKDL3vC7ojPTLNxmqEg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-get-function-arity": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-property-literals": { + "node_modules/@babel/plugin-transform-property-literals": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.8.3.tgz", "integrity": "sha512-uGiiXAZMqEoQhRWMK17VospMZh5sXWg+dlh2soffpkAl96KAm+WZuJfa6lcELotSRmooLqg0MWdH6UUq85nmmg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-react-jsx": { + "node_modules/@babel/plugin-transform-react-jsx": { "version": "7.9.4", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-react-jsx/-/plugin-transform-react-jsx-7.9.4.tgz", "integrity": "sha512-Mjqf3pZBNLt854CK0C/kRuXAnE6H/bo7xYojP+WGtX8glDGSibcwnsWwhwoSuRg0+EBnxPC1ouVnuetUIlPSAw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-builder-react-jsx": "^7.9.0", "@babel/helper-builder-react-jsx-experimental": "^7.9.0", "@babel/helper-plugin-utils": "^7.8.3", "@babel/plugin-syntax-jsx": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-regenerator": { + "node_modules/@babel/plugin-transform-regenerator": { "version": "7.8.7", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.8.7.tgz", "integrity": "sha512-TIg+gAl4Z0a3WmD3mbYSk+J9ZUH6n/Yc57rtKRnlA/7rcCvpekHXe0CMZHP1gYp7/KLe9GHTuIba0vXmls6drA==", "dev": true, - "requires": { + "dependencies": { "regenerator-transform": "^0.14.2" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-reserved-words": { + "node_modules/@babel/plugin-transform-reserved-words": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.8.3.tgz", "integrity": "sha512-mwMxcycN3omKFDjDQUl+8zyMsBfjRFr0Zn/64I41pmjv4NJuqcYlEtezwYtw9TFd9WR1vN5kiM+O0gMZzO6L0A==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-shorthand-properties": { + "node_modules/@babel/plugin-transform-shorthand-properties": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.8.3.tgz", "integrity": "sha512-I9DI6Odg0JJwxCHzbzW08ggMdCezoWcuQRz3ptdudgwaHxTjxw5HgdFJmZIkIMlRymL6YiZcped4TTCB0JcC8w==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-spread": { + "node_modules/@babel/plugin-transform-spread": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.8.3.tgz", "integrity": "sha512-CkuTU9mbmAoFOI1tklFWYYbzX5qCIZVXPVy0jpXgGwkplCndQAa58s2jr66fTeQnA64bDox0HL4U56CFYoyC7g==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-sticky-regex": { + "node_modules/@babel/plugin-transform-sticky-regex": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.8.3.tgz", "integrity": "sha512-9Spq0vGCD5Bb4Z/ZXXSK5wbbLFMG085qd2vhL1JYu1WcQ5bXqZBAYRzU1d+p79GcHs2szYv5pVQCX13QgldaWw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3", "@babel/helper-regex": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-template-literals": { + "node_modules/@babel/plugin-transform-template-literals": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.8.3.tgz", "integrity": "sha512-820QBtykIQOLFT8NZOcTRJ1UNuztIELe4p9DCgvj4NK+PwluSJ49we7s9FB1HIGNIYT7wFUJ0ar2QpCDj0escQ==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-annotate-as-pure": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-typeof-symbol": { + "node_modules/@babel/plugin-transform-typeof-symbol": { "version": "7.8.4", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.8.4.tgz", "integrity": "sha512-2QKyfjGdvuNfHsb7qnBBlKclbD4CfshH2KvDabiijLMGXPHJXGxtDzwIF7bQP+T0ysw8fYTtxPafgfs/c1Lrqg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/plugin-transform-unicode-regex": { + "node_modules/@babel/plugin-transform-unicode-regex": { "version": "7.8.3", "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.8.3.tgz", "integrity": "sha512-+ufgJjYdmWfSQ+6NS9VGUR2ns8cjJjYbrbi11mZBTaWm+Fui/ncTLFF28Ei1okavY+xkojGr1eJxNsWYeA5aZw==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-create-regexp-features-plugin": "^7.8.3", "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/preset-env": { + "node_modules/@babel/preset-env": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.9.0.tgz", "integrity": "sha512-712DeRXT6dyKAM/FMbQTV/FvRCms2hPCx+3weRjZ8iQVQWZejWWk1wwG6ViWMyqb/ouBbGOl5b6aCk0+j1NmsQ==", "dev": true, - "requires": { + "dependencies": { "@babel/compat-data": "^7.9.0", "@babel/helper-compilation-targets": "^7.8.7", "@babel/helper-module-imports": "^7.8.3", @@ -930,47 +1137,53 @@ "invariant": "^2.2.2", "levenary": "^1.1.1", "semver": "^5.5.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/preset-modules": { + "node_modules/@babel/preset-modules": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.3.tgz", "integrity": "sha512-Ra3JXOHBq2xd56xSF7lMKXdjBn3T772Y1Wet3yWnkDly9zHvJki029tAFzvAAK5cf4YV3yoxuP61crYRol6SVg==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-plugin-utils": "^7.0.0", "@babel/plugin-proposal-unicode-property-regex": "^7.4.4", "@babel/plugin-transform-dotall-regex": "^7.4.4", "@babel/types": "^7.4.4", "esutils": "^2.0.2" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" } }, - "@babel/runtime": { + "node_modules/@babel/runtime": { "version": "7.9.2", "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.9.2.tgz", "integrity": "sha512-NE2DtOdufG7R5vnfQUTehdTfNycfUANEtCa9PssN9O/xmTzP4E08UI797ixaei6hBEVL9BI/PsdJS5x7mWoB9Q==", "dev": true, - "requires": { + "dependencies": { "regenerator-runtime": "^0.13.4" } }, - "@babel/template": { + "node_modules/@babel/template": { "version": "7.8.6", "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.8.6.tgz", "integrity": "sha512-zbMsPMy/v0PWFZEhQJ66bqjhH+z0JgMoBWuikXybgG3Gkd/3t5oQ1Rw2WQhnSrsOmsKXnZOx15tkC4qON/+JPg==", "dev": true, - "requires": { + "dependencies": { "@babel/code-frame": "^7.8.3", "@babel/parser": "^7.8.6", "@babel/types": "^7.8.6" } }, - "@babel/traverse": { + "node_modules/@babel/traverse": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.9.0.tgz", "integrity": "sha512-jAZQj0+kn4WTHO5dUZkZKhbFrqZE7K5LAQ5JysMnmvGij+wOdr+8lWqPeW0BcF4wFwrEXXtdGO7wcV6YPJcf3w==", "dev": true, - "requires": { + "dependencies": { "@babel/code-frame": "^7.8.3", "@babel/generator": "^7.9.0", "@babel/helper-function-name": "^7.8.3", @@ -982,398 +1195,467 @@ "lodash": "^4.17.13" } }, - "@babel/types": { + "node_modules/@babel/types": { "version": "7.9.0", "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.9.0.tgz", "integrity": "sha512-BS9JKfXkzzJl8RluW4JGknzpiUV7ZrvTayM6yfqLTVBEnFtyowVIOu6rqxRd5cVO6yGoWf4T8u8dgK9oB+GCng==", "dev": true, - "requires": { + "dependencies": { "@babel/helper-validator-identifier": "^7.9.0", "lodash": "^4.17.13", "to-fast-properties": "^2.0.0" } }, - "@iarna/toml": { + "node_modules/@iarna/toml": { "version": "2.2.3", "resolved": "https://registry.npmjs.org/@iarna/toml/-/toml-2.2.3.tgz", "integrity": "sha512-FmuxfCuolpLl0AnQ2NHSzoUKWEJDFl63qXjzdoWBVyFCXzMGm1spBzk7LeHNoVCiWCF7mRVms9e6jEV9+MoPbg==", "dev": true }, - "@mrmlnc/readdir-enhanced": { + "node_modules/@mrmlnc/readdir-enhanced": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/@mrmlnc/readdir-enhanced/-/readdir-enhanced-2.2.1.tgz", "integrity": "sha512-bPHp6Ji8b41szTOcaP63VlnbbO5Ny6dwAATtY6JTjh5N2OLrb5Qk/Th5cRkRQhkWCt+EJsYrNB0MiL+Gpn6e3g==", "dev": true, - "requires": { + "dependencies": { "call-me-maybe": "^1.0.1", "glob-to-regexp": "^0.3.0" + }, + "engines": { + "node": ">=4" } }, - "@nodelib/fs.stat": { + "node_modules/@nodelib/fs.stat": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-1.1.3.tgz", "integrity": "sha512-shAmDyaQC4H92APFoIaVDHCx5bStIocgvbwQyxPRrbUY20V1EYTbSDchWbuwlMG3V17cprZhA6+78JfB+3DTPw==", - "dev": true + "dev": true, + "engines": { + "node": ">= 6" + } }, - "@parcel/fs": { + "node_modules/@parcel/fs": { "version": "1.11.0", "resolved": "https://registry.npmjs.org/@parcel/fs/-/fs-1.11.0.tgz", "integrity": "sha512-86RyEqULbbVoeo8OLcv+LQ1Vq2PKBAvWTU9fCgALxuCTbbs5Ppcvll4Vr+Ko1AnmMzja/k++SzNAwJfeQXVlpA==", "dev": true, - "requires": { + "dependencies": { "@parcel/utils": "^1.11.0", "mkdirp": "^0.5.1", "rimraf": "^2.6.2" + }, + "engines": { + "node": ">= 6.0.0" } }, - "@parcel/logger": { + "node_modules/@parcel/logger": { "version": "1.11.1", "resolved": "https://registry.npmjs.org/@parcel/logger/-/logger-1.11.1.tgz", "integrity": "sha512-9NF3M6UVeP2udOBDILuoEHd8VrF4vQqoWHEafymO1pfSoOMfxrSJZw1MfyAAIUN/IFp9qjcpDCUbDZB+ioVevA==", "dev": true, - "requires": { + "dependencies": { "@parcel/workers": "^1.11.0", "chalk": "^2.1.0", "grapheme-breaker": "^0.3.2", "ora": "^2.1.0", "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">= 6.0.0" } }, - "@parcel/utils": { + "node_modules/@parcel/utils": { "version": "1.11.0", "resolved": "https://registry.npmjs.org/@parcel/utils/-/utils-1.11.0.tgz", "integrity": "sha512-cA3p4jTlaMeOtAKR/6AadanOPvKeg8VwgnHhOyfi0yClD0TZS/hi9xu12w4EzA/8NtHu0g6o4RDfcNjqN8l1AQ==", - "dev": true + "dev": true, + "engines": { + "node": ">= 6.0.0" + } }, - "@parcel/watcher": { + "node_modules/@parcel/watcher": { "version": "1.12.1", "resolved": "https://registry.npmjs.org/@parcel/watcher/-/watcher-1.12.1.tgz", "integrity": "sha512-od+uCtCxC/KoNQAIE1vWx1YTyKYY+7CTrxBJPRh3cDWw/C0tCtlBMVlrbplscGoEpt6B27KhJDCv82PBxOERNA==", "dev": true, - "requires": { + "dependencies": { "@parcel/utils": "^1.11.0", "chokidar": "^2.1.5" } }, - "@parcel/workers": { + "node_modules/@parcel/workers": { "version": "1.11.0", "resolved": "https://registry.npmjs.org/@parcel/workers/-/workers-1.11.0.tgz", "integrity": "sha512-USSjRAAQYsZFlv43FUPdD+jEGML5/8oLF0rUzPQTtK4q9kvaXr49F5ZplyLz5lox78cLZ0TxN2bIDQ1xhOkulQ==", "dev": true, - "requires": { + "dependencies": { "@parcel/utils": "^1.11.0", "physical-cpu-count": "^2.0.0" + }, + "engines": { + "node": ">= 6.0.0" } }, - "@types/q": { + "node_modules/@types/q": { "version": "1.5.2", "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.2.tgz", "integrity": "sha512-ce5d3q03Ex0sy4R14722Rmt6MT07Ua+k4FwDfdcToYJcMKNtRVQvJ6JCAPdAmAnbRb6CsX6aYb9m96NGod9uTw==", "dev": true }, - "abab": { + "node_modules/abab": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.3.tgz", "integrity": "sha512-tsFzPpcttalNjFBCFMqsKYQcWxxen1pgJR56by//QwvJc4/OUS3kPOOttx2tSIfjsylB0pYu7f5D3K1RCxUnUg==", + "deprecated": "Use your platform's native atob() and btoa() methods instead", "dev": true }, - "acorn": { + "node_modules/acorn": { "version": "7.1.1", "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.1.1.tgz", "integrity": "sha512-add7dgA5ppRPxCFJoAGfMDi7PIBXq1RtGo7BhbLaxwrXPOmw8gq48Y9ozT01hUKy9byMjlR20EJhu5zlkErEkg==", - "dev": true + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } }, - "acorn-globals": { + "node_modules/acorn-globals": { "version": "4.3.4", "resolved": "https://registry.npmjs.org/acorn-globals/-/acorn-globals-4.3.4.tgz", "integrity": "sha512-clfQEh21R+D0leSbUdWf3OcfqyaCSAQ8Ryq00bofSekfr9W8u1jyYZo6ir0xu9Gtcf7BjcHJpnbZH7JOCpP60A==", "dev": true, - "requires": { + "dependencies": { "acorn": "^6.0.1", "acorn-walk": "^6.0.1" + } + }, + "node_modules/acorn-globals/node_modules/acorn": { + "version": "6.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.1.tgz", + "integrity": "sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==", + "dev": true, + "bin": { + "acorn": "bin/acorn" }, - "dependencies": { - "acorn": { - "version": "6.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.1.tgz", - "integrity": "sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==", - "dev": true - } + "engines": { + "node": ">=0.4.0" } }, - "acorn-walk": { + "node_modules/acorn-walk": { "version": "6.2.0", "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-6.2.0.tgz", "integrity": "sha512-7evsyfH1cLOCdAzZAd43Cic04yKydNx0cF+7tiA19p1XnLLPU4dpCQOqpjqwokFe//vS0QqfqqjCS2JkiIs0cA==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4.0" + } }, - "ajv": { + "node_modules/ajv": { "version": "6.12.0", "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.0.tgz", "integrity": "sha512-D6gFiFA0RRLyUbvijN74DWAjXSFxWKaWP7mldxkVhyhAV3+SWA9HEJPHQ2c9soIeTFJqcSdFDGFgdqs1iUU2Hw==", "dev": true, - "requires": { + "dependencies": { "fast-deep-equal": "^3.1.1", "fast-json-stable-stringify": "^2.0.0", "json-schema-traverse": "^0.4.1", "uri-js": "^4.2.2" } }, - "alphanum-sort": { + "node_modules/alphanum-sort": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/alphanum-sort/-/alphanum-sort-1.0.2.tgz", "integrity": "sha1-l6ERlkmyEa0zaR2fn0hqjsn74KM=", "dev": true }, - "ansi-regex": { + "node_modules/ansi-regex": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "ansi-styles": { + "node_modules/ansi-styles": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz", "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==", "dev": true, - "requires": { + "dependencies": { "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" } }, - "ansi-to-html": { + "node_modules/ansi-to-html": { "version": "0.6.14", "resolved": "https://registry.npmjs.org/ansi-to-html/-/ansi-to-html-0.6.14.tgz", "integrity": "sha512-7ZslfB1+EnFSDO5Ju+ue5Y6It19DRnZXWv8jrGHgIlPna5Mh4jz7BV5jCbQneXNFurQcKoolaaAjHtgSBfOIuA==", "dev": true, - "requires": { + "dependencies": { "entities": "^1.1.2" + }, + "bin": { + "ansi-to-html": "bin/ansi-to-html" + }, + "engines": { + "node": "*" } }, - "anymatch": { + "node_modules/anymatch": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-2.0.0.tgz", "integrity": "sha512-5teOsQWABXHHBFP9y3skS5P3d/WfWXpv3FUpy+LorMrNYaT9pI4oLMQX7jzQ2KklNpGpWHzdCXTDT2Y3XGlZBw==", "dev": true, - "requires": { + "dependencies": { "micromatch": "^3.1.4", "normalize-path": "^2.1.1" - }, + } + }, + "node_modules/anymatch/node_modules/normalize-path": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", + "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", + "dev": true, "dependencies": { - "normalize-path": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-2.1.1.tgz", - "integrity": "sha1-GrKLVW4Zg2Oowab35vogE3/mrtk=", - "dev": true, - "requires": { - "remove-trailing-separator": "^1.0.1" - } - } + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "argparse": { + "node_modules/argparse": { "version": "1.0.10", "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", "dev": true, - "requires": { + "dependencies": { "sprintf-js": "~1.0.2" } }, - "arr-diff": { + "node_modules/arr-diff": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", "integrity": "sha1-1kYQdP6/7HHn4VI1dhoyml3HxSA=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "arr-flatten": { + "node_modules/arr-flatten": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/arr-flatten/-/arr-flatten-1.1.0.tgz", "integrity": "sha512-L3hKV5R/p5o81R7O02IGnwpDmkp6E982XhtbuwSe3O4qOtMMMtodicASA1Cny2U+aCXcNpml+m4dPsvsJ3jatg==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "arr-union": { + "node_modules/arr-union": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/arr-union/-/arr-union-3.1.0.tgz", "integrity": "sha1-45sJrqne+Gao8gbiiK9jkZuuOcQ=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "array-equal": { + "node_modules/array-equal": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.0.tgz", "integrity": "sha1-jCpe8kcv2ep0KwTHenUJO6J1fJM=", "dev": true }, - "array-unique": { + "node_modules/array-unique": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/array-unique/-/array-unique-0.3.2.tgz", "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "asn1": { + "node_modules/asn1": { "version": "0.2.4", "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", "integrity": "sha512-jxwzQpLQjSmWXgwaCZE9Nz+glAG01yF1QnWgbhGwHI5A6FRIEY6IVqtHhIepHqI7/kyEyQEagBC5mBEFlIYvdg==", "dev": true, - "requires": { + "dependencies": { "safer-buffer": "~2.1.0" } }, - "asn1.js": { + "node_modules/asn1.js": { "version": "4.10.1", "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-4.10.1.tgz", "integrity": "sha512-p32cOF5q0Zqs9uBiONKYLm6BClCoBCM5O9JfeUSlnQLBTxYdTK+pW+nXflm8UkKd2UYlEbYz5qEi0JuZR9ckSw==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.0.0", "inherits": "^2.0.1", "minimalistic-assert": "^1.0.0" } }, - "assert": { + "node_modules/assert": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/assert/-/assert-1.5.0.tgz", "integrity": "sha512-EDsgawzwoun2CZkCgtxJbv392v4nbk9XDD06zI+kQYoBM/3RBWLlEyJARDOmhAAosBjWACEkKL6S+lIZtcAubA==", "dev": true, - "requires": { + "dependencies": { "object-assign": "^4.1.1", "util": "0.10.3" - }, - "dependencies": { - "inherits": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", - "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", - "dev": true - }, - "util": { - "version": "0.10.3", - "resolved": "https://registry.npmjs.org/util/-/util-0.10.3.tgz", - "integrity": "sha1-evsa/lCAUkZInj23/g7TeTNqwPk=", - "dev": true, - "requires": { - "inherits": "2.0.1" - } - } } }, - "assert-plus": { + "node_modules/assert-plus": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/assert-plus/-/assert-plus-1.0.0.tgz", "integrity": "sha1-8S4PPF13sLHN2RRpQuTpbB5N1SU=", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/assert/node_modules/inherits": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.1.tgz", + "integrity": "sha1-sX0I0ya0Qj5Wjv9xn5GwscvfafE=", "dev": true }, - "assign-symbols": { + "node_modules/assert/node_modules/util": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/util/-/util-0.10.3.tgz", + "integrity": "sha1-evsa/lCAUkZInj23/g7TeTNqwPk=", + "dev": true, + "dependencies": { + "inherits": "2.0.1" + } + }, + "node_modules/assign-symbols": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/assign-symbols/-/assign-symbols-1.0.0.tgz", "integrity": "sha1-WWZ/QfrdTyDMvCu5a41Pf3jsA2c=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "async-each": { + "node_modules/async-each": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/async-each/-/async-each-1.0.3.tgz", "integrity": "sha512-z/WhQ5FPySLdvREByI2vZiTWwCnF0moMJ1hK9YQwDTHKh6I7/uSckMetoRGb5UBZPC1z0jlw+n/XCgjeH7y1AQ==", "dev": true }, - "async-limiter": { + "node_modules/async-limiter": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz", "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==", "dev": true }, - "asynckit": { + "node_modules/asynckit": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", "dev": true }, - "atob": { + "node_modules/atob": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", - "dev": true + "dev": true, + "bin": { + "atob": "bin/atob.js" + }, + "engines": { + "node": ">= 4.5.0" + } }, - "aws-sign2": { + "node_modules/aws-sign2": { "version": "0.7.0", "resolved": "https://registry.npmjs.org/aws-sign2/-/aws-sign2-0.7.0.tgz", "integrity": "sha1-tG6JCTSpWR8tL2+G1+ap8bP+dqg=", - "dev": true + "dev": true, + "engines": { + "node": "*" + } }, - "aws4": { + "node_modules/aws4": { "version": "1.9.1", "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.9.1.tgz", "integrity": "sha512-wMHVg2EOHaMRxbzgFJ9gtjOOCrI80OHLG14rxi28XwOW8ux6IiEbRCGGGqCtdAIg4FQCbW20k9RsT4y3gJlFug==", "dev": true }, - "babel-plugin-dynamic-import-node": { + "node_modules/babel-plugin-dynamic-import-node": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.0.tgz", "integrity": "sha512-o6qFkpeQEBxcqt0XYlWzAVxNCSCZdUgcR8IRlhD/8DylxjjO4foPcvTW0GGKa/cVt3rvxZ7o5ippJ+/0nvLhlQ==", "dev": true, - "requires": { + "dependencies": { "object.assign": "^4.1.0" } }, - "babel-runtime": { + "node_modules/babel-runtime": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-runtime/-/babel-runtime-6.26.0.tgz", "integrity": "sha1-llxwWGaOgrVde/4E/yM3vItWR/4=", "dev": true, - "requires": { + "dependencies": { "core-js": "^2.4.0", "regenerator-runtime": "^0.11.0" - }, - "dependencies": { - "regenerator-runtime": { - "version": "0.11.1", - "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz", - "integrity": "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg==", - "dev": true - } } }, - "babel-types": { + "node_modules/babel-runtime/node_modules/regenerator-runtime": { + "version": "0.11.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.11.1.tgz", + "integrity": "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg==", + "dev": true + }, + "node_modules/babel-types": { "version": "6.26.0", "resolved": "https://registry.npmjs.org/babel-types/-/babel-types-6.26.0.tgz", "integrity": "sha1-o7Bz+Uq0nrb6Vc1lInozQ4BjJJc=", "dev": true, - "requires": { + "dependencies": { "babel-runtime": "^6.26.0", "esutils": "^2.0.2", "lodash": "^4.17.4", "to-fast-properties": "^1.0.3" - }, - "dependencies": { - "to-fast-properties": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-1.0.3.tgz", - "integrity": "sha1-uDVx+k2MJbguIxsG46MFXeTKGkc=", - "dev": true - } } }, - "babylon-walk": { + "node_modules/babel-types/node_modules/to-fast-properties": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-1.0.3.tgz", + "integrity": "sha1-uDVx+k2MJbguIxsG46MFXeTKGkc=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/babylon-walk": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/babylon-walk/-/babylon-walk-1.0.2.tgz", "integrity": "sha1-OxWl3btIKni0zpwByLoYFwLZ1s4=", "dev": true, - "requires": { + "dependencies": { "babel-runtime": "^6.11.6", "babel-types": "^6.15.0", "lodash.clone": "^4.5.0" } }, - "balanced-match": { + "node_modules/balanced-match": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", "dev": true }, - "base": { + "node_modules/base": { "version": "0.11.2", "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", "dev": true, - "requires": { + "dependencies": { "cache-base": "^1.0.1", "class-utils": "^0.3.5", "component-emitter": "^1.2.1", @@ -1382,105 +1664,123 @@ "mixin-deep": "^1.2.0", "pascalcase": "^0.1.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/base/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "base64-js": { + "node_modules/base64-js": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.3.1.tgz", "integrity": "sha512-mLQ4i2QO1ytvGWFWmcngKO//JXAQueZvwEKtjgQFM4jIK0kU+ytMfplL8j+n5mspOfjHwoAg+9yhb7BwAHm36g==", "dev": true }, - "bcrypt-pbkdf": { + "node_modules/bcrypt-pbkdf": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/bcrypt-pbkdf/-/bcrypt-pbkdf-1.0.2.tgz", "integrity": "sha1-pDAdOJtqQ/m2f/PKEaP2Y342Dp4=", "dev": true, - "requires": { + "dependencies": { "tweetnacl": "^0.14.3" } }, - "binary-extensions": { + "node_modules/binary-extensions": { "version": "1.13.1", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-1.13.1.tgz", "integrity": "sha512-Un7MIEDdUC5gNpcGDV97op1Ywk748MpHcFTHoYs6qnj1Z3j7I53VG3nwZhKzoBZmbdRNnb6WRdFlwl7tSDuZGw==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "bindings": { + "node_modules/bindings": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/bindings/-/bindings-1.5.0.tgz", "integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==", "dev": true, - "requires": { + "dependencies": { "file-uri-to-path": "1.0.0" } }, - "bn.js": { + "node_modules/bn.js": { "version": "4.11.8", "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.8.tgz", "integrity": "sha512-ItfYfPLkWHUjckQCk8xC+LwxgK8NYcXywGigJgSwOP8Y2iyWT4f2vsZnoOXTTbo+o5yXmIUJ4gn5538SO5S3gA==", "dev": true }, - "boolbase": { + "node_modules/boolbase": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/boolbase/-/boolbase-1.0.0.tgz", "integrity": "sha1-aN/1++YMUes3cl6p4+0xDcwed24=", "dev": true }, - "brace-expansion": { + "node_modules/brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", "dev": true, - "requires": { + "dependencies": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" } }, - "braces": { + "node_modules/braces": { "version": "2.3.2", "resolved": "https://registry.npmjs.org/braces/-/braces-2.3.2.tgz", "integrity": "sha512-aNdbnj9P8PjdXU4ybaWLK2IF3jc/EoDYbC7AazW6to3TRsfXxscC9UXOB5iDiEQrkyIbWp2SLQda4+QAa7nc3w==", "dev": true, - "requires": { + "dependencies": { "arr-flatten": "^1.1.0", "array-unique": "^0.3.2", "extend-shallow": "^2.0.1", @@ -1492,48 +1792,55 @@ "split-string": "^3.0.2", "to-regex": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/braces/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "brfs": { + "node_modules/brfs": { "version": "1.6.1", "resolved": "https://registry.npmjs.org/brfs/-/brfs-1.6.1.tgz", "integrity": "sha512-OfZpABRQQf+Xsmju8XE9bDjs+uU4vLREGolP7bDgcpsI17QREyZ4Bl+2KLxxx1kCgA0fAIhKQBaBYh+PEcCqYQ==", "dev": true, - "requires": { + "dependencies": { "quote-stream": "^1.0.1", "resolve": "^1.1.5", "static-module": "^2.2.0", "through2": "^2.0.0" + }, + "bin": { + "brfs": "bin/cmd.js" } }, - "brorand": { + "node_modules/brorand": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", "dev": true }, - "browser-process-hrtime": { + "node_modules/browser-process-hrtime": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/browser-process-hrtime/-/browser-process-hrtime-1.0.0.tgz", "integrity": "sha512-9o5UecI3GhkpM6DrXr69PblIuWxPKk9Y0jHBRhdocZ2y7YECBFCsHm79Pr3OyR2AvjhDkabFJaDJMYRazHgsow==", "dev": true }, - "browserify-aes": { + "node_modules/browserify-aes": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/browserify-aes/-/browserify-aes-1.2.0.tgz", "integrity": "sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==", "dev": true, - "requires": { + "dependencies": { "buffer-xor": "^1.0.3", "cipher-base": "^1.0.0", "create-hash": "^1.1.0", @@ -1542,45 +1849,45 @@ "safe-buffer": "^5.0.1" } }, - "browserify-cipher": { + "node_modules/browserify-cipher": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/browserify-cipher/-/browserify-cipher-1.0.1.tgz", "integrity": "sha512-sPhkz0ARKbf4rRQt2hTpAHqn47X3llLkUGn+xEJzLjwY8LRs2p0v7ljvI5EyoRO/mexrNunNECisZs+gw2zz1w==", "dev": true, - "requires": { + "dependencies": { "browserify-aes": "^1.0.4", "browserify-des": "^1.0.0", "evp_bytestokey": "^1.0.0" } }, - "browserify-des": { + "node_modules/browserify-des": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/browserify-des/-/browserify-des-1.0.2.tgz", "integrity": "sha512-BioO1xf3hFwz4kc6iBhI3ieDFompMhrMlnDFC4/0/vd5MokpuAc3R+LYbwTA9A5Yc9pq9UYPqffKpW2ObuwX5A==", "dev": true, - "requires": { + "dependencies": { "cipher-base": "^1.0.1", "des.js": "^1.0.0", "inherits": "^2.0.1", "safe-buffer": "^5.1.2" } }, - "browserify-rsa": { + "node_modules/browserify-rsa": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", "integrity": "sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.1.0", "randombytes": "^2.0.1" } }, - "browserify-sign": { + "node_modules/browserify-sign": { "version": "4.0.4", "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.0.4.tgz", "integrity": "sha1-qk62jl17ZYuqa/alfmMMvXqT0pg=", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.1.1", "browserify-rsa": "^4.0.0", "create-hash": "^1.1.0", @@ -1590,76 +1897,84 @@ "parse-asn1": "^5.0.0" } }, - "browserify-zlib": { + "node_modules/browserify-zlib": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/browserify-zlib/-/browserify-zlib-0.2.0.tgz", "integrity": "sha512-Z942RysHXmJrhqk88FmKBVq/v5tqmSkDz7p54G/MGyjMnCFFnC79XWNbg+Vta8W6Wb2qtSZTSxIGkJrRpCFEiA==", "dev": true, - "requires": { - "pako": "~1.0.5" - }, "dependencies": { - "pako": { - "version": "1.0.11", - "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", - "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", - "dev": true - } + "pako": "~1.0.5" } }, - "browserslist": { + "node_modules/browserify-zlib/node_modules/pako": { + "version": "1.0.11", + "resolved": "https://registry.npmjs.org/pako/-/pako-1.0.11.tgz", + "integrity": "sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw==", + "dev": true + }, + "node_modules/browserslist": { "version": "4.11.1", "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.11.1.tgz", "integrity": "sha512-DCTr3kDrKEYNw6Jb9HFxVLQNaue8z+0ZfRBRjmCunKDEXEBajKDj2Y+Uelg+Pi29OnvaSGwjOsnRyNEkXzHg5g==", "dev": true, - "requires": { + "dependencies": { "caniuse-lite": "^1.0.30001038", "electron-to-chromium": "^1.3.390", "node-releases": "^1.1.53", "pkg-up": "^2.0.0" + }, + "bin": { + "browserslist": "cli.js" + }, + "funding": { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/browserslist" } }, - "buffer": { + "node_modules/buffer": { "version": "4.9.2", "resolved": "https://registry.npmjs.org/buffer/-/buffer-4.9.2.tgz", "integrity": "sha512-xq+q3SRMOxGivLhBNaUdC64hDTQwejJ+H0T/NB1XMtTVEwNTrfFF3gAxiyW0Bu/xWEGhjVKgUcMhCrUy2+uCWg==", "dev": true, - "requires": { + "dependencies": { "base64-js": "^1.0.2", "ieee754": "^1.1.4", "isarray": "^1.0.0" } }, - "buffer-equal": { + "node_modules/buffer-equal": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/buffer-equal/-/buffer-equal-0.0.1.tgz", "integrity": "sha1-kbx0sR6kBbyRa8aqkI+q+ltKrEs=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4.0" + } }, - "buffer-from": { + "node_modules/buffer-from": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz", "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==", "dev": true }, - "buffer-xor": { + "node_modules/buffer-xor": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/buffer-xor/-/buffer-xor-1.0.3.tgz", "integrity": "sha1-JuYe0UIvtw3ULm42cp7VHYVf6Nk=", "dev": true }, - "builtin-status-codes": { + "node_modules/builtin-status-codes": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/builtin-status-codes/-/builtin-status-codes-3.0.0.tgz", "integrity": "sha1-hZgoeOIbmOHGZCXgPQF0eI9Wnug=", "dev": true }, - "cache-base": { + "node_modules/cache-base": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/cache-base/-/cache-base-1.0.1.tgz", "integrity": "sha512-AKcdTnFSWATd5/GCPRxr2ChwIJ85CeyrEyjRHlKxQ56d4XJMGym0uAiKn0xbLOGOl3+yRpOTi484dVCEc5AUzQ==", "dev": true, - "requires": { + "dependencies": { "collection-visit": "^1.0.0", "component-emitter": "^1.2.1", "get-value": "^2.0.6", @@ -1669,89 +1984,121 @@ "to-object-path": "^0.3.0", "union-value": "^1.0.0", "unset-value": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "call-me-maybe": { + "node_modules/call-me-maybe": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=", "dev": true }, - "caller-callsite": { + "node_modules/caller-callsite": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/caller-callsite/-/caller-callsite-2.0.0.tgz", "integrity": "sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=", "dev": true, - "requires": { + "dependencies": { "callsites": "^2.0.0" + }, + "engines": { + "node": ">=4" } }, - "caller-path": { + "node_modules/caller-path": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/caller-path/-/caller-path-2.0.0.tgz", "integrity": "sha1-Ro+DBE42mrIBD6xfBs7uFbsssfQ=", "dev": true, - "requires": { + "dependencies": { "caller-callsite": "^2.0.0" + }, + "engines": { + "node": ">=4" } }, - "callsites": { + "node_modules/callsites": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-2.0.0.tgz", "integrity": "sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "camelcase": { + "node_modules/camelcase": { "version": "5.3.1", "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true + "dev": true, + "engines": { + "node": ">=6" + } }, - "caniuse-api": { + "node_modules/caniuse-api": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/caniuse-api/-/caniuse-api-3.0.0.tgz", "integrity": "sha512-bsTwuIg/BZZK/vreVTYYbSWoe2F+71P7K5QGEX+pT250DZbfU1MQ5prOKpPR+LL6uWKK3KMwMCAS74QB3Um1uw==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "caniuse-lite": "^1.0.0", "lodash.memoize": "^4.1.2", "lodash.uniq": "^4.5.0" } }, - "caniuse-lite": { - "version": "1.0.30001039", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001039.tgz", - "integrity": "sha512-SezbWCTT34eyFoWHgx8UWso7YtvtM7oosmFoXbCkdC6qJzRfBTeTgE9REtKtiuKXuMwWTZEvdnFNGAyVMorv8Q==", - "dev": true + "node_modules/caniuse-lite": { + "version": "1.0.30001584", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001584.tgz", + "integrity": "sha512-LOz7CCQ9M1G7OjJOF9/mzmqmj3jE/7VOmrfw6Mgs0E8cjOsbRXQJHsPBfmBOXDskXKrHLyyW3n7kpDW/4BsfpQ==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + }, + { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/caniuse-lite" + }, + { + "type": "github", + "url": "https://github.com/sponsors/ai" + } + ] }, - "caseless": { + "node_modules/caseless": { "version": "0.12.0", "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", "dev": true }, - "chalk": { + "node_modules/chalk": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz", "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==", "dev": true, - "requires": { + "dependencies": { "ansi-styles": "^3.2.1", "escape-string-regexp": "^1.0.5", "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" } }, - "chokidar": { + "node_modules/chokidar": { "version": "2.1.8", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "deprecated": "Chokidar 2 does not receive security updates since 2019. Upgrade to chokidar 3 with 15x fewer dependencies", "dev": true, - "requires": { + "dependencies": { "anymatch": "^2.0.0", "async-each": "^1.0.1", "braces": "^2.3.2", - "fsevents": "^1.2.7", "glob-parent": "^3.1.0", "inherits": "^2.0.3", "is-binary-path": "^1.0.0", @@ -1760,276 +2107,321 @@ "path-is-absolute": "^1.0.0", "readdirp": "^2.2.1", "upath": "^1.1.1" + }, + "optionalDependencies": { + "fsevents": "^1.2.7" } }, - "cipher-base": { + "node_modules/cipher-base": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/cipher-base/-/cipher-base-1.0.4.tgz", "integrity": "sha512-Kkht5ye6ZGmwv40uUDZztayT2ThLQGfnj/T71N/XzeZeo3nf8foyW7zGTsPYkEya3m5f3cAypH+qe7YOrM1U2Q==", "dev": true, - "requires": { + "dependencies": { "inherits": "^2.0.1", "safe-buffer": "^5.0.1" } }, - "class-utils": { + "node_modules/class-utils": { "version": "0.3.6", "resolved": "https://registry.npmjs.org/class-utils/-/class-utils-0.3.6.tgz", "integrity": "sha512-qOhPa/Fj7s6TY8H8esGu5QNpMMQxz79h+urzrNYN6mn+9BnxlDGf5QZ+XeCDsxSjPqsSR56XOZOJmpeurnLMeg==", "dev": true, - "requires": { + "dependencies": { "arr-union": "^3.1.0", "define-property": "^0.2.5", "isobject": "^3.0.0", "static-extend": "^0.1.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "cli-cursor": { + "node_modules/cli-cursor": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-2.1.0.tgz", "integrity": "sha1-s12sN2R5+sw+lHR9QdDQ9SOP/LU=", "dev": true, - "requires": { + "dependencies": { "restore-cursor": "^2.0.0" + }, + "engines": { + "node": ">=4" } }, - "cli-spinners": { + "node_modules/cli-spinners": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-1.3.1.tgz", "integrity": "sha512-1QL4544moEsDVH9T/l6Cemov/37iv1RtoKf7NJ04A60+4MREXNfx/QvavbH6QoGdsD4N4Mwy49cmaINR/o2mdg==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "cliui": { + "node_modules/cliui": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", "dev": true, - "requires": { + "dependencies": { "string-width": "^3.1.0", "strip-ansi": "^5.2.0", "wrap-ansi": "^5.1.0" - }, + } + }, + "node_modules/cliui/node_modules/ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/cliui/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - } + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" } }, - "clone": { + "node_modules/clone": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.8" + } }, - "coa": { + "node_modules/coa": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/coa/-/coa-2.0.2.tgz", "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", "dev": true, - "requires": { + "dependencies": { "@types/q": "^1.5.1", "chalk": "^2.4.1", "q": "^1.1.2" + }, + "engines": { + "node": ">= 4.0" } }, - "collection-visit": { + "node_modules/collection-visit": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/collection-visit/-/collection-visit-1.0.0.tgz", "integrity": "sha1-S8A3PBZLwykbTTaMgpzxqApZ3KA=", "dev": true, - "requires": { + "dependencies": { "map-visit": "^1.0.0", "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "color": { + "node_modules/color": { "version": "3.1.2", "resolved": "https://registry.npmjs.org/color/-/color-3.1.2.tgz", "integrity": "sha512-vXTJhHebByxZn3lDvDJYw4lR5+uB3vuoHsuYA5AKuxRVn5wzzIfQKGLBmgdVRHKTJYeK5rvJcHnrd0Li49CFpg==", "dev": true, - "requires": { + "dependencies": { "color-convert": "^1.9.1", "color-string": "^1.5.2" } }, - "color-convert": { + "node_modules/color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==", "dev": true, - "requires": { + "dependencies": { "color-name": "1.1.3" } }, - "color-name": { + "node_modules/color-name": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=", "dev": true }, - "color-string": { + "node_modules/color-string": { "version": "1.5.3", "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.3.tgz", "integrity": "sha512-dC2C5qeWoYkxki5UAXapdjqO672AM4vZuPGRQfO8b5HKuKGBbKWpITyDYN7TOFKvRW7kOgAn3746clDBMDJyQw==", "dev": true, - "requires": { + "dependencies": { "color-name": "^1.0.0", "simple-swizzle": "^0.2.2" } }, - "combined-stream": { + "node_modules/combined-stream": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", "dev": true, - "requires": { + "dependencies": { "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" } }, - "command-exists": { + "node_modules/command-exists": { "version": "1.2.8", "resolved": "https://registry.npmjs.org/command-exists/-/command-exists-1.2.8.tgz", "integrity": "sha512-PM54PkseWbiiD/mMsbvW351/u+dafwTJ0ye2qB60G1aGQP9j3xK2gmMDc+R34L3nDtx4qMCitXT75mkbkGJDLw==", "dev": true }, - "commander": { + "node_modules/commander": { "version": "2.20.3", "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz", "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==", "dev": true }, - "component-emitter": { + "node_modules/component-emitter": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.3.0.tgz", "integrity": "sha512-Rd3se6QB+sO1TwqZjscQrurpEPIfO0/yYnSin6Q/rD3mOutHvUrCAhJub3r90uNb+SESBuE0QYoB90YdfatsRg==", "dev": true }, - "concat-map": { + "node_modules/concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", "dev": true }, - "concat-stream": { + "node_modules/concat-stream": { "version": "1.6.2", "resolved": "https://registry.npmjs.org/concat-stream/-/concat-stream-1.6.2.tgz", "integrity": "sha512-27HBghJxjiZtIk3Ycvn/4kbJk/1uZuJFfuPEns6LaEvpvG1f0hTea8lilrouyo9mVc2GWdcEZ8OLoGmSADlrCw==", "dev": true, - "requires": { + "engines": [ + "node >= 0.8" + ], + "dependencies": { "buffer-from": "^1.0.0", "inherits": "^2.0.3", "readable-stream": "^2.2.2", "typedarray": "^0.0.6" } }, - "console-browserify": { + "node_modules/console-browserify": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/console-browserify/-/console-browserify-1.2.0.tgz", "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==", "dev": true }, - "constants-browserify": { + "node_modules/constants-browserify": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", "integrity": "sha1-wguW2MYXdIqvHBYCF2DNJ/y4y3U=", "dev": true }, - "convert-source-map": { + "node_modules/convert-source-map": { "version": "1.7.0", "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-1.7.0.tgz", "integrity": "sha512-4FJkXzKXEDB1snCFZlLP4gpC3JILicCpGbzG9f9G7tGqGCzETQ2hWPrcinA9oU4wtf2biUaEH5065UnMeR33oA==", "dev": true, - "requires": { + "dependencies": { "safe-buffer": "~5.1.1" } }, - "copy-descriptor": { + "node_modules/copy-descriptor": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "core-js": { + "node_modules/core-js": { "version": "2.6.11", "resolved": "https://registry.npmjs.org/core-js/-/core-js-2.6.11.tgz", "integrity": "sha512-5wjnpaT/3dV+XB4borEsnAYQchn00XSgTAWKDkEqv+K8KevjbzmofK6hfJ9TZIlpj2N0xQpazy7PiRQiWHqzWg==", - "dev": true + "deprecated": "core-js@<3.23.3 is no longer maintained and not recommended for usage due to the number of issues. Because of the V8 engine whims, feature detection in old core-js versions could cause a slowdown up to 100x even if nothing is polyfilled. Some versions have web compatibility issues. Please, upgrade your dependencies to the actual version of core-js.", + "dev": true, + "hasInstallScript": true }, - "core-js-compat": { + "node_modules/core-js-compat": { "version": "3.6.4", "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.6.4.tgz", "integrity": "sha512-zAa3IZPvsJ0slViBQ2z+vgyyTuhd3MFn1rBQjZSKVEgB0UMYhUkCj9jJUVPgGTGqWvsBVmfnruXgTcNyTlEiSA==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.8.3", "semver": "7.0.0" }, - "dependencies": { - "semver": { - "version": "7.0.0", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", - "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", - "dev": true - } + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" } }, - "core-util-is": { + "node_modules/core-js-compat/node_modules/semver": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.0.0.tgz", + "integrity": "sha512-+GB6zVA9LWh6zovYQLALHwv5rb2PHGlJi3lfiqIHxR0uuwCgefcOJc59v9fv1w8GbStwxuuqqAjI9NMAOOgq1A==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/core-util-is": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz", "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", "dev": true }, - "cosmiconfig": { + "node_modules/cosmiconfig": { "version": "5.2.1", "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", "dev": true, - "requires": { + "dependencies": { "import-fresh": "^2.0.0", "is-directory": "^0.3.1", "js-yaml": "^3.13.1", "parse-json": "^4.0.0" + }, + "engines": { + "node": ">=4" } }, - "create-ecdh": { + "node_modules/create-ecdh": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.3.tgz", "integrity": "sha512-GbEHQPMOswGpKXM9kCWVrremUcBmjteUaQ01T9rkKCPDXfUHX0IoP9LpHYo2NPFampa4e+/pFDc3jQdxrxQLaw==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.1.0", "elliptic": "^6.0.0" } }, - "create-hash": { + "node_modules/create-hash": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/create-hash/-/create-hash-1.2.0.tgz", "integrity": "sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==", "dev": true, - "requires": { + "dependencies": { "cipher-base": "^1.0.1", "inherits": "^2.0.1", "md5.js": "^1.3.4", @@ -2037,12 +2429,12 @@ "sha.js": "^2.4.0" } }, - "create-hmac": { + "node_modules/create-hmac": { "version": "1.1.7", "resolved": "https://registry.npmjs.org/create-hmac/-/create-hmac-1.1.7.tgz", "integrity": "sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==", "dev": true, - "requires": { + "dependencies": { "cipher-base": "^1.0.3", "create-hash": "^1.1.0", "inherits": "^2.0.1", @@ -2051,25 +2443,28 @@ "sha.js": "^2.4.8" } }, - "cross-spawn": { + "node_modules/cross-spawn": { "version": "6.0.5", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", "integrity": "sha512-eTVLrBSt7fjbDygz805pMnstIs2VTBNkRm0qxZd+M7A5XDdxVRWO5MxGBXZhjY4cqLYLdtrGqRf8mBPmzwSpWQ==", "dev": true, - "requires": { + "dependencies": { "nice-try": "^1.0.4", "path-key": "^2.0.1", "semver": "^5.5.0", "shebang-command": "^1.2.0", "which": "^1.2.9" + }, + "engines": { + "node": ">=4.8" } }, - "crypto-browserify": { + "node_modules/crypto-browserify": { "version": "3.12.0", "resolved": "https://registry.npmjs.org/crypto-browserify/-/crypto-browserify-3.12.0.tgz", "integrity": "sha512-fz4spIh+znjO2VjL+IdhEpRJ3YN6sMzITSBijk6FK2UvTqruSQW+/cCZTSNsMiZNvUeq0CqurF+dAbyiGOY6Wg==", "dev": true, - "requires": { + "dependencies": { "browserify-cipher": "^1.0.0", "browserify-sign": "^4.0.0", "create-ecdh": "^4.0.0", @@ -2081,183 +2476,230 @@ "public-encrypt": "^4.0.0", "randombytes": "^2.0.0", "randomfill": "^1.0.3" + }, + "engines": { + "node": "*" } }, - "css-color-names": { + "node_modules/css-color-names": { "version": "0.0.4", "resolved": "https://registry.npmjs.org/css-color-names/-/css-color-names-0.0.4.tgz", "integrity": "sha1-gIrcLnnPhHOAabZGyyDsJ762KeA=", - "dev": true + "dev": true, + "engines": { + "node": "*" + } }, - "css-declaration-sorter": { + "node_modules/css-declaration-sorter": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-4.0.1.tgz", "integrity": "sha512-BcxQSKTSEEQUftYpBVnsH4SF05NTuBokb19/sBt6asXGKZ/6VP7PLG1CBCkFDYOnhXhPh0jMhO6xZ71oYHXHBA==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.1", "timsort": "^0.3.0" + }, + "engines": { + "node": ">4" } }, - "css-modules-loader-core": { + "node_modules/css-modules-loader-core": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/css-modules-loader-core/-/css-modules-loader-core-1.1.0.tgz", "integrity": "sha1-WQhmgpShvs0mGuCkziGwtVHyHRY=", "dev": true, - "requires": { + "dependencies": { "icss-replace-symbols": "1.1.0", "postcss": "6.0.1", "postcss-modules-extract-imports": "1.1.0", "postcss-modules-local-by-default": "1.2.0", "postcss-modules-scope": "1.1.0", "postcss-modules-values": "1.3.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/ansi-styles": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", + "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/chalk": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", + "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", + "dev": true, + "dependencies": { + "ansi-styles": "^2.2.1", + "escape-string-regexp": "^1.0.2", + "has-ansi": "^2.0.0", + "strip-ansi": "^3.0.0", + "supports-color": "^2.0.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/chalk/node_modules/supports-color": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", + "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/has-flag": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz", + "integrity": "sha1-nZ55MWXOAXoA8AQYxD+UKnsdEfo=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/postcss": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.1.tgz", + "integrity": "sha1-AA29H47vIXqjaLmiEsX8QLKo8/I=", + "dev": true, "dependencies": { - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "dev": true - }, - "ansi-styles": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-2.2.1.tgz", - "integrity": "sha1-tDLdM1i2NM914eRmQ2gkBTPB3b4=", - "dev": true - }, - "chalk": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", - "integrity": "sha1-qBFcVeSnAv5NFQq9OHKCKn4J/Jg=", - "dev": true, - "requires": { - "ansi-styles": "^2.2.1", - "escape-string-regexp": "^1.0.2", - "has-ansi": "^2.0.0", - "strip-ansi": "^3.0.0", - "supports-color": "^2.0.0" - }, - "dependencies": { - "supports-color": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-2.0.0.tgz", - "integrity": "sha1-U10EXOa2Nj+kARcIRimZXp3zJMc=", - "dev": true - } - } - }, - "has-flag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-1.0.0.tgz", - "integrity": "sha1-nZ55MWXOAXoA8AQYxD+UKnsdEfo=", - "dev": true - }, - "postcss": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.1.tgz", - "integrity": "sha1-AA29H47vIXqjaLmiEsX8QLKo8/I=", - "dev": true, - "requires": { - "chalk": "^1.1.3", - "source-map": "^0.5.6", - "supports-color": "^3.2.3" - } - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - }, - "strip-ansi": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", - "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "dev": true, - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "supports-color": { - "version": "3.2.3", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.2.3.tgz", - "integrity": "sha1-ZawFBLOVQXHYpklGsq48u4pfVPY=", - "dev": true, - "requires": { - "has-flag": "^1.0.0" - } - } + "chalk": "^1.1.3", + "source-map": "^0.5.6", + "supports-color": "^3.2.3" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/strip-ansi": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", + "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", + "dev": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-modules-loader-core/node_modules/supports-color": { + "version": "3.2.3", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-3.2.3.tgz", + "integrity": "sha1-ZawFBLOVQXHYpklGsq48u4pfVPY=", + "dev": true, + "dependencies": { + "has-flag": "^1.0.0" + }, + "engines": { + "node": ">=0.8.0" } }, - "css-select": { + "node_modules/css-select": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/css-select/-/css-select-2.1.0.tgz", "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", "dev": true, - "requires": { + "dependencies": { "boolbase": "^1.0.0", "css-what": "^3.2.1", "domutils": "^1.7.0", "nth-check": "^1.0.2" } }, - "css-select-base-adapter": { + "node_modules/css-select-base-adapter": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/css-select-base-adapter/-/css-select-base-adapter-0.1.1.tgz", "integrity": "sha512-jQVeeRG70QI08vSTwf1jHxp74JoZsr2XSgETae8/xC8ovSnL2WF87GTLO86Sbwdt2lK4Umg4HnnwMO4YF3Ce7w==", "dev": true }, - "css-selector-tokenizer": { + "node_modules/css-selector-tokenizer": { "version": "0.7.2", "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.2.tgz", "integrity": "sha512-yj856NGuAymN6r8bn8/Jl46pR+OC3eEvAhfGYDUe7YPtTPAYrSSw4oAniZ9Y8T5B92hjhwTBLUen0/vKPxf6pw==", "dev": true, - "requires": { + "dependencies": { "cssesc": "^3.0.0", "fastparse": "^1.1.2", "regexpu-core": "^4.6.0" } }, - "css-tree": { + "node_modules/css-tree": { "version": "1.0.0-alpha.37", "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.37.tgz", "integrity": "sha512-DMxWJg0rnz7UgxKT0Q1HU/L9BeJI0M6ksor0OgqOnF+aRCDWg/N2641HmVyU9KVIu0OVVWOb2IpC9A+BJRnejg==", "dev": true, - "requires": { + "dependencies": { "mdn-data": "2.0.4", "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" } }, - "css-what": { + "node_modules/css-what": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/css-what/-/css-what-3.2.1.tgz", "integrity": "sha512-WwOrosiQTvyms+Ti5ZC5vGEK0Vod3FTt1ca+payZqvKuGJF+dq7bG63DstxtN0dpm6FxY27a/zS3Wten+gEtGw==", - "dev": true + "dev": true, + "engines": { + "node": ">= 6" + } }, - "cssesc": { + "node_modules/cssesc": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", "integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==", - "dev": true + "dev": true, + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } }, - "cssnano": { + "node_modules/cssnano": { "version": "4.1.10", "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-4.1.10.tgz", "integrity": "sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ==", "dev": true, - "requires": { + "dependencies": { "cosmiconfig": "^5.0.0", "cssnano-preset-default": "^4.0.7", "is-resolvable": "^1.0.0", "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "cssnano-preset-default": { + "node_modules/cssnano-preset-default": { "version": "4.0.7", "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-4.0.7.tgz", "integrity": "sha512-x0YHHx2h6p0fCl1zY9L9roD7rnlltugGu7zXSKQx6k2rYw0Hi3IqxcoAGF7u9Q5w1nt7vK0ulxV8Lo+EvllGsA==", "dev": true, - "requires": { + "dependencies": { "css-declaration-sorter": "^4.0.1", "cssnano-util-raw-cache": "^4.0.1", "postcss": "^7.0.0", @@ -2288,362 +2730,426 @@ "postcss-reduce-transforms": "^4.0.2", "postcss-svgo": "^4.0.2", "postcss-unique-selectors": "^4.0.1" + }, + "engines": { + "node": ">=6.9.0" } }, - "cssnano-util-get-arguments": { + "node_modules/cssnano-util-get-arguments": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/cssnano-util-get-arguments/-/cssnano-util-get-arguments-4.0.0.tgz", "integrity": "sha1-7ToIKZ8h11dBsg87gfGU7UnMFQ8=", - "dev": true + "dev": true, + "engines": { + "node": ">=6.9.0" + } }, - "cssnano-util-get-match": { + "node_modules/cssnano-util-get-match": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/cssnano-util-get-match/-/cssnano-util-get-match-4.0.0.tgz", "integrity": "sha1-wOTKB/U4a7F+xeUiULT1lhNlFW0=", - "dev": true + "dev": true, + "engines": { + "node": ">=6.9.0" + } }, - "cssnano-util-raw-cache": { + "node_modules/cssnano-util-raw-cache": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/cssnano-util-raw-cache/-/cssnano-util-raw-cache-4.0.1.tgz", "integrity": "sha512-qLuYtWK2b2Dy55I8ZX3ky1Z16WYsx544Q0UWViebptpwn/xDBmog2TLg4f+DBMg1rJ6JDWtn96WHbOKDWt1WQA==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "cssnano-util-same-parent": { + "node_modules/cssnano-util-same-parent": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/cssnano-util-same-parent/-/cssnano-util-same-parent-4.0.1.tgz", "integrity": "sha512-WcKx5OY+KoSIAxBW6UBBRay1U6vkYheCdjyVNDm85zt5K9mHoGOfsOsqIszfAqrQQFIIKgjh2+FDgIj/zsl21Q==", - "dev": true + "dev": true, + "engines": { + "node": ">=6.9.0" + } }, - "csso": { + "node_modules/csso": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/csso/-/csso-4.0.3.tgz", "integrity": "sha512-NL3spysxUkcrOgnpsT4Xdl2aiEiBG6bXswAABQVHcMrfjjBisFOKwLDOmf4wf32aPdcJws1zds2B0Rg+jqMyHQ==", "dev": true, - "requires": { + "dependencies": { "css-tree": "1.0.0-alpha.39" }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/csso/node_modules/css-tree": { + "version": "1.0.0-alpha.39", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.39.tgz", + "integrity": "sha512-7UvkEYgBAHRG9Nt980lYxjsTrCyHFN53ky3wVsDkiMdVqylqRt+Zc+jm5qw7/qyOvN2dHSYtX0e4MbCCExSvnA==", + "dev": true, "dependencies": { - "css-tree": { - "version": "1.0.0-alpha.39", - "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.0.0-alpha.39.tgz", - "integrity": "sha512-7UvkEYgBAHRG9Nt980lYxjsTrCyHFN53ky3wVsDkiMdVqylqRt+Zc+jm5qw7/qyOvN2dHSYtX0e4MbCCExSvnA==", - "dev": true, - "requires": { - "mdn-data": "2.0.6", - "source-map": "^0.6.1" - } - }, - "mdn-data": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.6.tgz", - "integrity": "sha512-rQvjv71olwNHgiTbfPZFkJtjNMciWgswYeciZhtvWLO8bmX3TnhyA62I6sTWOyZssWHJJjY6/KiWwqQsWWsqOA==", - "dev": true - } + "mdn-data": "2.0.6", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" } }, - "cssom": { + "node_modules/csso/node_modules/mdn-data": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.6.tgz", + "integrity": "sha512-rQvjv71olwNHgiTbfPZFkJtjNMciWgswYeciZhtvWLO8bmX3TnhyA62I6sTWOyZssWHJJjY6/KiWwqQsWWsqOA==", + "dev": true + }, + "node_modules/cssom": { "version": "0.3.8", "resolved": "https://registry.npmjs.org/cssom/-/cssom-0.3.8.tgz", "integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==", "dev": true }, - "cssstyle": { + "node_modules/cssstyle": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/cssstyle/-/cssstyle-1.4.0.tgz", "integrity": "sha512-GBrLZYZ4X4x6/QEoBnIrqb8B/f5l4+8me2dkom/j1Gtbxy0kBv6OGzKuAsGM75bkGwGAFkt56Iwg28S3XTZgSA==", "dev": true, - "requires": { + "dependencies": { "cssom": "0.3.x" } }, - "dashdash": { + "node_modules/dashdash": { "version": "1.14.1", "resolved": "https://registry.npmjs.org/dashdash/-/dashdash-1.14.1.tgz", "integrity": "sha1-hTz6D3y+L+1d4gMmuN1YEDX24vA=", "dev": true, - "requires": { + "dependencies": { "assert-plus": "^1.0.0" + }, + "engines": { + "node": ">=0.10" } }, - "data-urls": { + "node_modules/data-urls": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-1.1.0.tgz", "integrity": "sha512-YTWYI9se1P55u58gL5GkQHW4P6VJBJ5iBT+B5a7i2Tjadhv52paJG0qHX4A0OR6/t52odI64KP2YvFpkDOi3eQ==", "dev": true, - "requires": { + "dependencies": { "abab": "^2.0.0", "whatwg-mimetype": "^2.2.0", "whatwg-url": "^7.0.0" } }, - "deasync": { + "node_modules/deasync": { "version": "0.1.19", "resolved": "https://registry.npmjs.org/deasync/-/deasync-0.1.19.tgz", "integrity": "sha512-oh3MRktfnPlLysCPpBpKZZzb4cUC/p0aA3SyRGp15lN30juJBTo/CiD0d4fR+f1kBtUQoJj1NE9RPNWQ7BQ9Mg==", "dev": true, - "requires": { + "hasInstallScript": true, + "dependencies": { "bindings": "^1.5.0", "node-addon-api": "^1.7.1" + }, + "engines": { + "node": ">=0.11.0" } }, - "debug": { + "node_modules/debug": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", + "deprecated": "Debug versions >=3.2.0 <3.2.7 || >=4 <4.3.1 have a low-severity ReDos regression when used in a Node.js environment. It is recommended you upgrade to 3.2.7 or 4.3.1. (https://github.com/visionmedia/debug/issues/797)", "dev": true, - "requires": { + "dependencies": { "ms": "^2.1.1" } }, - "decamelize": { + "node_modules/decamelize": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "decode-uri-component": { + "node_modules/decode-uri-component": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/decode-uri-component/-/decode-uri-component-0.2.0.tgz", "integrity": "sha1-6zkTMzRYd1y4TNGh+uBiEGu4dUU=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10" + } }, - "deep-is": { + "node_modules/deep-is": { "version": "0.1.3", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.3.tgz", "integrity": "sha1-s2nW+128E+7PUk+RsHD+7cNXzzQ=", "dev": true }, - "defaults": { + "node_modules/defaults": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/defaults/-/defaults-1.0.3.tgz", "integrity": "sha1-xlYFHpgX2f8I7YgUd/P+QBnz730=", "dev": true, - "requires": { - "clone": "^1.0.2" - }, "dependencies": { - "clone": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", - "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", - "dev": true - } + "clone": "^1.0.2" + } + }, + "node_modules/defaults/node_modules/clone": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", + "dev": true, + "engines": { + "node": ">=0.8" } }, - "define-properties": { + "node_modules/define-properties": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.1.3.tgz", "integrity": "sha512-3MqfYKj2lLzdMSf8ZIZE/V+Zuy+BgD6f164e8K2w7dgnpKArBDerGYpM46IYYcjnkdPNMjPk9A6VFB8+3SKlXQ==", "dev": true, - "requires": { + "dependencies": { "object-keys": "^1.0.12" + }, + "engines": { + "node": ">= 0.4" } }, - "define-property": { + "node_modules/define-property": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/define-property/-/define-property-2.0.2.tgz", "integrity": "sha512-jwK2UV4cnPpbcG7+VRARKTZPUWowwXA8bzH5NP6ud0oeAxyYPuGZUAC7hMugpCdz4BeSZl2Dl9k66CHJ/46ZYQ==", "dev": true, - "requires": { + "dependencies": { "is-descriptor": "^1.0.2", "isobject": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, "dependencies": { - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/define-property/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "delayed-stream": { + "node_modules/delayed-stream": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4.0" + } }, - "depd": { + "node_modules/depd": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "des.js": { + "node_modules/des.js": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/des.js/-/des.js-1.0.1.tgz", "integrity": "sha512-Q0I4pfFrv2VPd34/vfLrFOoRmlYj3OV50i7fskps1jZWK1kApMWWT9G6RRUeYedLcBDIhnSDaUvJMb3AhUlaEA==", "dev": true, - "requires": { + "dependencies": { "inherits": "^2.0.1", "minimalistic-assert": "^1.0.0" } }, - "destroy": { + "node_modules/destroy": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=", "dev": true }, - "diffie-hellman": { + "node_modules/diffie-hellman": { "version": "5.0.3", "resolved": "https://registry.npmjs.org/diffie-hellman/-/diffie-hellman-5.0.3.tgz", "integrity": "sha512-kqag/Nl+f3GwyK25fhUMYj81BUOrZ9IuJsjIcDE5icNM9FJHAVm3VcUDxdLPoQtTuUylWm6ZIknYJwwaPxsUzg==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.1.0", "miller-rabin": "^4.0.0", "randombytes": "^2.0.0" } }, - "dom-serializer": { + "node_modules/dom-serializer": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", "dev": true, - "requires": { + "dependencies": { "domelementtype": "^2.0.1", "entities": "^2.0.0" - }, - "dependencies": { - "domelementtype": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.1.tgz", - "integrity": "sha512-5HOHUDsYZWV8FGWN0Njbr/Rn7f/eWSQi1v7+HsUVwXgn8nWWlL64zKDkS0n8ZmQ3mlWOMuXOnR+7Nx/5tMO5AQ==", - "dev": true - }, - "entities": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.0.0.tgz", - "integrity": "sha512-D9f7V0JSRwIxlRI2mjMqufDrRDnx8p+eEOz7aUM9SuvF8gsBzra0/6tbjl1m8eQHrZlYj6PxqE00hZ1SAIKPLw==", - "dev": true - } } }, - "domain-browser": { + "node_modules/dom-serializer/node_modules/domelementtype": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.1.tgz", + "integrity": "sha512-5HOHUDsYZWV8FGWN0Njbr/Rn7f/eWSQi1v7+HsUVwXgn8nWWlL64zKDkS0n8ZmQ3mlWOMuXOnR+7Nx/5tMO5AQ==", + "dev": true + }, + "node_modules/dom-serializer/node_modules/entities": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.0.0.tgz", + "integrity": "sha512-D9f7V0JSRwIxlRI2mjMqufDrRDnx8p+eEOz7aUM9SuvF8gsBzra0/6tbjl1m8eQHrZlYj6PxqE00hZ1SAIKPLw==", + "dev": true + }, + "node_modules/domain-browser": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/domain-browser/-/domain-browser-1.2.0.tgz", "integrity": "sha512-jnjyiM6eRyZl2H+W8Q/zLMA481hzi0eszAaBUzIVnmYVDBbnLxVNnfu1HgEBvCbL+71FrxMl3E6lpKH7Ge3OXA==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4", + "npm": ">=1.2" + } }, - "domelementtype": { + "node_modules/domelementtype": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", "dev": true }, - "domexception": { + "node_modules/domexception": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/domexception/-/domexception-1.0.1.tgz", "integrity": "sha512-raigMkn7CJNNo6Ihro1fzG7wr3fHuYVytzquZKX5n0yizGsTcYgzdIUwj1X9pK0VvjeihV+XiclP+DjwbsSKug==", + "deprecated": "Use your platform's native DOMException instead", "dev": true, - "requires": { + "dependencies": { "webidl-conversions": "^4.0.2" } }, - "domhandler": { + "node_modules/domhandler": { "version": "2.4.2", "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-2.4.2.tgz", "integrity": "sha512-JiK04h0Ht5u/80fdLMCEmV4zkNh2BcoMFBmZ/91WtYZ8qVXSKjiw7fXMgFPnHcSZgOo3XdinHvmnDUeMf5R4wA==", "dev": true, - "requires": { + "dependencies": { "domelementtype": "1" } }, - "domutils": { + "node_modules/domutils": { "version": "1.7.0", "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", "dev": true, - "requires": { + "dependencies": { "dom-serializer": "0", "domelementtype": "1" } }, - "dot-prop": { + "node_modules/dot-prop": { "version": "5.2.0", "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.2.0.tgz", "integrity": "sha512-uEUyaDKoSQ1M4Oq8l45hSE26SnTxL6snNnqvK/VWx5wJhmff5z0FUVJDKDanor/6w3kzE3i7XZOk+7wC0EXr1A==", "dev": true, - "requires": { + "dependencies": { "is-obj": "^2.0.0" + }, + "engines": { + "node": ">=8" } }, - "dotenv": { + "node_modules/dotenv": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/dotenv/-/dotenv-5.0.1.tgz", "integrity": "sha512-4As8uPrjfwb7VXC+WnLCbXK7y+Ueb2B3zgNCePYfhxS1PYeaO1YTeplffTEcbfLhvFNGLAz90VvJs9yomG7bow==", - "dev": true + "dev": true, + "engines": { + "node": ">=4.6.0" + } }, - "dotenv-expand": { + "node_modules/dotenv-expand": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/dotenv-expand/-/dotenv-expand-5.1.0.tgz", "integrity": "sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA==", "dev": true }, - "duplexer2": { + "node_modules/duplexer2": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/duplexer2/-/duplexer2-0.1.4.tgz", "integrity": "sha1-ixLauHjA1p4+eJEFFmKjL8a93ME=", "dev": true, - "requires": { + "dependencies": { "readable-stream": "^2.0.2" } }, - "ecc-jsbn": { + "node_modules/ecc-jsbn": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz", "integrity": "sha1-OoOpBOVDUyh4dMVkt1SThoSamMk=", "dev": true, - "requires": { + "dependencies": { "jsbn": "~0.1.0", "safer-buffer": "^2.1.0" } }, - "ee-first": { + "node_modules/ee-first": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", "dev": true }, - "electron-to-chromium": { + "node_modules/electron-to-chromium": { "version": "1.3.398", "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.398.tgz", "integrity": "sha512-BJjxuWLKFbM5axH3vES7HKMQgAknq9PZHBkMK/rEXUQG9i1Iw5R+6hGkm6GtsQSANjSUrh/a6m32nzCNDNo/+w==", "dev": true }, - "elliptic": { + "node_modules/elliptic": { "version": "6.5.2", "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.2.tgz", "integrity": "sha512-f4x70okzZbIQl/NSRLkI/+tteV/9WqL98zx+SQ69KbXxmVrmjwsNUPn/gYJJ0sHvEak24cZgHIPegRePAtA/xw==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.4.0", "brorand": "^1.0.1", "hash.js": "^1.0.0", @@ -2653,45 +3159,54 @@ "minimalistic-crypto-utils": "^1.0.0" } }, - "emoji-regex": { + "node_modules/emoji-regex": { "version": "7.0.3", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-7.0.3.tgz", "integrity": "sha512-CwBLREIQ7LvYFB0WyRvwhq5N5qPhc6PMjD6bYggFlI5YyDgl+0vxq5VHbMOFqLg7hfWzmu8T5Z1QofhmTIhItA==", "dev": true }, - "encodeurl": { + "node_modules/encodeurl": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.8" + } }, - "entities": { + "node_modules/entities": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/entities/-/entities-1.1.2.tgz", "integrity": "sha512-f2LZMYl1Fzu7YSBKg+RoROelpOaNrcGmE9AZubeDfrCEia483oW4MI4VyFd5VNHIgQ/7qm1I0wUHK1eJnn2y2w==", "dev": true }, - "envinfo": { + "node_modules/envinfo": { "version": "7.5.0", "resolved": "https://registry.npmjs.org/envinfo/-/envinfo-7.5.0.tgz", "integrity": "sha512-jDgnJaF/Btomk+m3PZDTTCb5XIIIX3zYItnCRfF73zVgvinLoRomuhi75Y4su0PtQxWz4v66XnLLckyvyJTOIQ==", - "dev": true + "dev": true, + "bin": { + "envinfo": "dist/cli.js" + }, + "engines": { + "node": ">=4" + } }, - "error-ex": { + "node_modules/error-ex": { "version": "1.3.2", "resolved": "https://registry.npmjs.org/error-ex/-/error-ex-1.3.2.tgz", "integrity": "sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g==", "dev": true, - "requires": { + "dependencies": { "is-arrayish": "^0.2.1" } }, - "es-abstract": { + "node_modules/es-abstract": { "version": "1.17.5", "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.5.tgz", "integrity": "sha512-BR9auzDbySxOcfog0tLECW8l28eRGpDpU3Dm3Hp4q/N+VtLTmyj4EUN088XZWQDW/hzj6sYRDXeOFsaAODKvpg==", "dev": true, - "requires": { + "dependencies": { "es-to-primitive": "^1.2.1", "function-bind": "^1.1.1", "has": "^1.0.3", @@ -2704,97 +3219,141 @@ "string.prototype.trimleft": "^2.1.1", "string.prototype.trimright": "^2.1.1" }, - "dependencies": { - "object-inspect": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", - "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", - "dev": true - } + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/es-abstract/node_modules/object-inspect": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.7.0.tgz", + "integrity": "sha512-a7pEHdh1xKIAgTySUGgLMx/xwDZskN1Ud6egYYN3EdRW4ZMPNEDUTF+hwy2LUC+Bl+SyLXANnwz/jyh/qutKUw==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "es-to-primitive": { + "node_modules/es-to-primitive": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/es-to-primitive/-/es-to-primitive-1.2.1.tgz", "integrity": "sha512-QCOllgZJtaUo9miYBcLChTUaHNjJF3PYs1VidD7AwiEj1kYxKeQTctLAezAOH5ZKRH0g2IgPn6KwB4IT8iRpvA==", "dev": true, - "requires": { + "dependencies": { "is-callable": "^1.1.4", "is-date-object": "^1.0.1", "is-symbol": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "escape-html": { + "node_modules/escape-html": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=", "dev": true }, - "escape-string-regexp": { + "node_modules/escape-string-regexp": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz", "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.8.0" + } }, - "escodegen": { + "node_modules/escodegen": { "version": "1.9.1", "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.9.1.tgz", "integrity": "sha512-6hTjO1NAWkHnDk3OqQ4YrCuwwmGHL9S3nPlzBOUG/R44rda3wLNrfvQ5fkSGjyhHFKM7ALPKcKGrwvCLe0lC7Q==", "dev": true, - "requires": { + "dependencies": { "esprima": "^3.1.3", "estraverse": "^4.2.0", "esutils": "^2.0.2", - "optionator": "^0.8.1", + "optionator": "^0.8.1" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=4.0" + }, + "optionalDependencies": { "source-map": "~0.6.1" } }, - "esprima": { + "node_modules/esprima": { "version": "3.1.3", "resolved": "https://registry.npmjs.org/esprima/-/esprima-3.1.3.tgz", "integrity": "sha1-/cpRzuYTOJXjyI1TXOSdv/YqRjM=", - "dev": true + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } }, - "estraverse": { + "node_modules/estraverse": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", - "dev": true + "dev": true, + "engines": { + "node": ">=4.0" + } }, - "esutils": { + "node_modules/esutils": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz", "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "etag": { + "node_modules/etag": { "version": "1.8.1", "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "events": { + "node_modules/events": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/events/-/events-3.1.0.tgz", "integrity": "sha512-Rv+u8MLHNOdMjTAFeT3nCjHn2aGlx435FP/sDHNaRhDEMwyI/aB22Kj2qIN8R0cw3z28psEQLYwxVKLsKrMgWg==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.8.x" + } }, - "evp_bytestokey": { + "node_modules/evp_bytestokey": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz", "integrity": "sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==", "dev": true, - "requires": { + "dependencies": { "md5.js": "^1.3.4", "safe-buffer": "^5.1.1" } }, - "expand-brackets": { + "node_modules/expand-brackets": { "version": "2.1.4", "resolved": "https://registry.npmjs.org/expand-brackets/-/expand-brackets-2.1.4.tgz", "integrity": "sha1-t3c14xXOMPa27/D4OwQVGiJEliI=", "dev": true, - "requires": { + "dependencies": { "debug": "^2.3.3", "define-property": "^0.2.5", "extend-shallow": "^2.0.1", @@ -2803,75 +3362,86 @@ "snapdragon": "^0.8.1", "to-regex": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } + "ms": "2.0.0" + } + }, + "node_modules/expand-brackets/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "extend": { + "node_modules/expand-brackets/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "node_modules/extend": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", "integrity": "sha512-fjquC59cD7CyW6urNXK0FBufkZcoiGG80wTuPujX590cB5Ttln20E2UB4S/WARVqhXffZl2LNgS+gQdPIIim/g==", "dev": true }, - "extend-shallow": { + "node_modules/extend-shallow": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-3.0.2.tgz", "integrity": "sha1-Jqcarwc7OfshJxcnRhMcJwQCjbg=", "dev": true, - "requires": { + "dependencies": { "assign-symbols": "^1.0.0", "is-extendable": "^1.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extend-shallow/node_modules/is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } + "is-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" } }, - "extglob": { + "node_modules/extglob": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/extglob/-/extglob-2.0.4.tgz", "integrity": "sha512-Nmb6QXkELsuBr24CJSkilo6UHHgbekK5UiZgfE6UHD3Eb27YC6oD+bhcT+tJ6cl8dmsgdQxnWlcry8ksBIBLpw==", "dev": true, - "requires": { + "dependencies": { "array-unique": "^0.3.2", "define-property": "^1.0.0", "expand-brackets": "^2.1.4", @@ -2881,1029 +3451,1297 @@ "snapdragon": "^0.8.1", "to-regex": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "extsprintf": { + "node_modules/extsprintf": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", - "dev": true + "dev": true, + "engines": [ + "node >=0.6.0" + ] }, - "falafel": { + "node_modules/falafel": { "version": "2.2.4", "resolved": "https://registry.npmjs.org/falafel/-/falafel-2.2.4.tgz", "integrity": "sha512-0HXjo8XASWRmsS0X1EkhwEMZaD3Qvp7FfURwjLKjG1ghfRm/MGZl2r4cWUTv41KdNghTw4OUMmVtdGQp3+H+uQ==", "dev": true, - "requires": { + "dependencies": { "acorn": "^7.1.1", "foreach": "^2.0.5", "isarray": "^2.0.1", "object-keys": "^1.0.6" }, - "dependencies": { - "isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", - "dev": true - } + "engines": { + "node": ">=0.4.0" } }, - "fast-deep-equal": { + "node_modules/falafel/node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true + }, + "node_modules/fast-deep-equal": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.1.tgz", "integrity": "sha512-8UEa58QDLauDNfpbrX55Q9jrGHThw2ZMdOky5Gl1CDtVeJDPVrG4Jxx1N8jw2gkWaff5UUuX1KJd+9zGe2B+ZA==", "dev": true }, - "fast-glob": { + "node_modules/fast-glob": { "version": "2.2.7", "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-2.2.7.tgz", "integrity": "sha512-g1KuQwHOZAmOZMuBtHdxDtju+T2RT8jgCC9aANsbpdiDDTSnjgfuVsIBNKbUeJI3oKMRExcfNDtJl4OhbffMsw==", "dev": true, - "requires": { + "dependencies": { "@mrmlnc/readdir-enhanced": "^2.2.1", "@nodelib/fs.stat": "^1.1.2", "glob-parent": "^3.1.0", "is-glob": "^4.0.0", "merge2": "^1.2.3", "micromatch": "^3.1.10" + }, + "engines": { + "node": ">=4.0.0" } }, - "fast-json-stable-stringify": { + "node_modules/fast-json-stable-stringify": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz", "integrity": "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw==", "dev": true }, - "fast-levenshtein": { + "node_modules/fast-levenshtein": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz", "integrity": "sha1-PYpcZog6FqMMqGQ+hR8Zuqd5eRc=", "dev": true }, - "fastparse": { + "node_modules/fastparse": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/fastparse/-/fastparse-1.1.2.tgz", "integrity": "sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ==", "dev": true }, - "file-uri-to-path": { + "node_modules/file-uri-to-path": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz", "integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==", "dev": true }, - "filesize": { + "node_modules/filesize": { "version": "3.6.1", "resolved": "https://registry.npmjs.org/filesize/-/filesize-3.6.1.tgz", "integrity": "sha512-7KjR1vv6qnicaPMi1iiTcI85CyYwRO/PSFCu6SvqL8jN2Wjt/NIYQTFtFs7fSDCYOstUkEWIQGFUg5YZQfjlcg==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.4.0" + } }, - "fill-range": { + "node_modules/fill-range": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", "integrity": "sha1-1USBHUKPmOsGpj3EAtJAPDKMOPc=", "dev": true, - "requires": { + "dependencies": { "extend-shallow": "^2.0.1", "is-number": "^3.0.0", "repeat-string": "^1.6.1", "to-regex-range": "^2.1.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fill-range/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "find-up": { + "node_modules/find-up": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/find-up/-/find-up-2.1.0.tgz", "integrity": "sha1-RdG35QbHF93UgndaK3eSCjwMV6c=", "dev": true, - "requires": { + "dependencies": { "locate-path": "^2.0.0" + }, + "engines": { + "node": ">=4" } }, - "for-in": { + "node_modules/for-in": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/for-in/-/for-in-1.0.2.tgz", "integrity": "sha1-gQaNKVqBQuwKxybG4iAMMPttXoA=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "foreach": { + "node_modules/foreach": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/foreach/-/foreach-2.0.5.tgz", "integrity": "sha1-C+4AUBiusmDQo6865ljdATbsG5k=", "dev": true }, - "forever-agent": { + "node_modules/forever-agent": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/forever-agent/-/forever-agent-0.6.1.tgz", "integrity": "sha1-+8cfDEGt6zf5bFd60e1C2P2sypE=", - "dev": true + "dev": true, + "engines": { + "node": "*" + } }, - "form-data": { + "node_modules/form-data": { "version": "2.3.3", "resolved": "https://registry.npmjs.org/form-data/-/form-data-2.3.3.tgz", "integrity": "sha512-1lLKB2Mu3aGP1Q/2eCOx0fNbRMe7XdwktwOruhfqqd0rIJWwN4Dh+E3hrPSlDCXnSR7UtZ1N38rVXm+6+MEhJQ==", "dev": true, - "requires": { + "dependencies": { "asynckit": "^0.4.0", "combined-stream": "^1.0.6", "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 0.12" } }, - "fragment-cache": { + "node_modules/fragment-cache": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/fragment-cache/-/fragment-cache-0.2.1.tgz", "integrity": "sha1-QpD60n8T6Jvn8zeZxrxaCr//DRk=", "dev": true, - "requires": { + "dependencies": { "map-cache": "^0.2.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "fresh": { + "node_modules/fresh": { "version": "0.5.2", "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "fs.realpath": { + "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=", "dev": true }, - "fsevents": { + "node_modules/fsevents": { "version": "1.2.12", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.12.tgz", "integrity": "sha512-Ggd/Ktt7E7I8pxZRbGIs7vwqAPscSESMrCSkx2FtWeqmheJgCo2R74fTsZFCifr0VTPwqRpPv17+6b8Zp7th0Q==", + "bundleDependencies": [ + "node-pre-gyp" + ], + "deprecated": "The v1 package contains DANGEROUS / INSECURE binaries. Upgrade to safe fsevents v2", "dev": true, + "hasInstallScript": true, "optional": true, - "requires": { + "os": [ + "darwin" + ], + "dependencies": { "bindings": "^1.5.0", "nan": "^2.12.1", "node-pre-gyp": "*" }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/fsevents/node_modules/abbrev": { + "version": "1.1.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/ansi-regex": { + "version": "2.1.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/aproba": { + "version": "1.2.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/are-we-there-yet": { + "version": "1.1.5", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, "dependencies": { - "abbrev": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true - }, - "ansi-regex": { - "version": "2.1.1", - "bundled": true, - "dev": true, - "optional": true - }, - "aproba": { - "version": "1.2.0", - "bundled": true, - "dev": true, - "optional": true - }, - "are-we-there-yet": { - "version": "1.1.5", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "delegates": "^1.0.0", - "readable-stream": "^2.0.6" - } - }, - "balanced-match": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "brace-expansion": { - "version": "1.1.11", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "balanced-match": "^1.0.0", - "concat-map": "0.0.1" - } - }, - "chownr": { - "version": "1.1.4", - "bundled": true, - "dev": true, - "optional": true - }, - "code-point-at": { - "version": "1.1.0", - "bundled": true, - "dev": true, - "optional": true - }, - "concat-map": { - "version": "0.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "console-control-strings": { - "version": "1.1.0", - "bundled": true, - "dev": true, - "optional": true - }, - "core-util-is": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "debug": { - "version": "3.2.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "ms": "^2.1.1" - } - }, - "deep-extend": { - "version": "0.6.0", - "bundled": true, - "dev": true, - "optional": true - }, - "delegates": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "detect-libc": { - "version": "1.0.3", - "bundled": true, - "dev": true, - "optional": true - }, - "fs-minipass": { - "version": "1.2.7", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minipass": "^2.6.0" - } - }, - "fs.realpath": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "gauge": { - "version": "2.7.4", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "aproba": "^1.0.3", - "console-control-strings": "^1.0.0", - "has-unicode": "^2.0.0", - "object-assign": "^4.1.0", - "signal-exit": "^3.0.0", - "string-width": "^1.0.1", - "strip-ansi": "^3.0.1", - "wide-align": "^1.1.0" - } - }, - "glob": { - "version": "7.1.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "fs.realpath": "^1.0.0", - "inflight": "^1.0.4", - "inherits": "2", - "minimatch": "^3.0.4", - "once": "^1.3.0", - "path-is-absolute": "^1.0.0" - } - }, - "has-unicode": { - "version": "2.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "iconv-lite": { - "version": "0.4.24", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "safer-buffer": ">= 2.1.2 < 3" - } - }, - "ignore-walk": { - "version": "3.0.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minimatch": "^3.0.4" - } - }, - "inflight": { - "version": "1.0.6", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "once": "^1.3.0", - "wrappy": "1" - } - }, - "inherits": { - "version": "2.0.4", - "bundled": true, - "dev": true, - "optional": true - }, - "ini": { - "version": "1.3.5", - "bundled": true, - "dev": true, - "optional": true - }, - "is-fullwidth-code-point": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "number-is-nan": "^1.0.0" - } - }, - "isarray": { - "version": "1.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "minimatch": { - "version": "3.0.4", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "brace-expansion": "^1.1.7" - } - }, - "minimist": { - "version": "1.2.5", - "bundled": true, - "dev": true, - "optional": true - }, - "minipass": { - "version": "2.9.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - } - }, - "minizlib": { - "version": "1.3.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minipass": "^2.9.0" - } - }, - "mkdirp": { - "version": "0.5.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "minimist": "^1.2.5" - } - }, - "ms": { - "version": "2.1.2", - "bundled": true, - "dev": true, - "optional": true - }, - "needle": { - "version": "2.3.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "debug": "^3.2.6", - "iconv-lite": "^0.4.4", - "sax": "^1.2.4" - } - }, - "node-pre-gyp": { - "version": "0.14.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "detect-libc": "^1.0.2", - "mkdirp": "^0.5.1", - "needle": "^2.2.1", - "nopt": "^4.0.1", - "npm-packlist": "^1.1.6", - "npmlog": "^4.0.2", - "rc": "^1.2.7", - "rimraf": "^2.6.1", - "semver": "^5.3.0", - "tar": "^4.4.2" - } - }, - "nopt": { - "version": "4.0.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "abbrev": "1", - "osenv": "^0.1.4" - } - }, - "npm-bundled": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npm-normalize-package-bin": { - "version": "1.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "npm-packlist": { - "version": "1.4.8", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "ignore-walk": "^3.0.1", - "npm-bundled": "^1.0.1", - "npm-normalize-package-bin": "^1.0.1" - } - }, - "npmlog": { - "version": "4.1.2", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "are-we-there-yet": "~1.1.2", - "console-control-strings": "~1.1.0", - "gauge": "~2.7.3", - "set-blocking": "~2.0.0" - } - }, - "number-is-nan": { - "version": "1.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "object-assign": { - "version": "4.1.1", - "bundled": true, - "dev": true, - "optional": true - }, - "once": { - "version": "1.4.0", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "wrappy": "1" - } - }, - "os-homedir": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "os-tmpdir": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "osenv": { - "version": "0.1.5", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "os-homedir": "^1.0.0", - "os-tmpdir": "^1.0.0" - } - }, - "path-is-absolute": { - "version": "1.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "process-nextick-args": { - "version": "2.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "rc": { - "version": "1.2.8", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "deep-extend": "^0.6.0", - "ini": "~1.3.0", - "minimist": "^1.2.0", - "strip-json-comments": "~2.0.1" - } - }, - "readable-stream": { - "version": "2.3.7", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "core-util-is": "~1.0.0", - "inherits": "~2.0.3", - "isarray": "~1.0.0", - "process-nextick-args": "~2.0.0", - "safe-buffer": "~5.1.1", - "string_decoder": "~1.1.1", - "util-deprecate": "~1.0.1" - } - }, - "rimraf": { - "version": "2.7.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "glob": "^7.1.3" - } - }, - "safe-buffer": { - "version": "5.1.2", - "bundled": true, - "dev": true, - "optional": true - }, - "safer-buffer": { - "version": "2.1.2", - "bundled": true, - "dev": true, - "optional": true - }, - "sax": { - "version": "1.2.4", - "bundled": true, - "dev": true, - "optional": true - }, - "semver": { - "version": "5.7.1", - "bundled": true, - "dev": true, - "optional": true - }, - "set-blocking": { - "version": "2.0.0", - "bundled": true, - "dev": true, - "optional": true - }, - "signal-exit": { - "version": "3.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "string-width": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "code-point-at": "^1.0.0", - "is-fullwidth-code-point": "^1.0.0", - "strip-ansi": "^3.0.0" - } - }, - "string_decoder": { - "version": "1.1.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "safe-buffer": "~5.1.0" - } - }, - "strip-ansi": { - "version": "3.0.1", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "ansi-regex": "^2.0.0" - } - }, - "strip-json-comments": { - "version": "2.0.1", - "bundled": true, - "dev": true, - "optional": true - }, - "tar": { - "version": "4.4.13", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "chownr": "^1.1.1", - "fs-minipass": "^1.2.5", - "minipass": "^2.8.6", - "minizlib": "^1.2.1", - "mkdirp": "^0.5.0", - "safe-buffer": "^5.1.2", - "yallist": "^3.0.3" - } - }, - "util-deprecate": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "wide-align": { - "version": "1.1.3", - "bundled": true, - "dev": true, - "optional": true, - "requires": { - "string-width": "^1.0.2 || 2" - } - }, - "wrappy": { - "version": "1.0.2", - "bundled": true, - "dev": true, - "optional": true - }, - "yallist": { - "version": "3.1.1", - "bundled": true, - "dev": true, - "optional": true - } + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "node_modules/fsevents/node_modules/balanced-match": { + "version": "1.0.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/brace-expansion": { + "version": "1.1.11", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/fsevents/node_modules/chownr": { + "version": "1.1.4", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/code-point-at": { + "version": "1.1.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/concat-map": { + "version": "0.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/console-control-strings": { + "version": "1.1.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/core-util-is": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/debug": { + "version": "3.2.6", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/fsevents/node_modules/deep-extend": { + "version": "0.6.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/fsevents/node_modules/delegates": { + "version": "1.0.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/detect-libc": { + "version": "1.0.3", + "dev": true, + "inBundle": true, + "license": "Apache-2.0", + "optional": true, + "bin": { + "detect-libc": "bin/detect-libc.js" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/fsevents/node_modules/fs-minipass": { + "version": "1.2.7", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "minipass": "^2.6.0" + } + }, + "node_modules/fsevents/node_modules/fs.realpath": { + "version": "1.0.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/gauge": { + "version": "2.7.4", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "aproba": "^1.0.3", + "console-control-strings": "^1.0.0", + "has-unicode": "^2.0.0", + "object-assign": "^4.1.0", + "signal-exit": "^3.0.0", + "string-width": "^1.0.1", + "strip-ansi": "^3.0.1", + "wide-align": "^1.1.0" + } + }, + "node_modules/fsevents/node_modules/glob": { + "version": "7.1.6", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" + } + }, + "node_modules/fsevents/node_modules/has-unicode": { + "version": "2.0.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/iconv-lite": { + "version": "0.4.24", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/ignore-walk": { + "version": "3.0.3", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "minimatch": "^3.0.4" + } + }, + "node_modules/fsevents/node_modules/inflight": { + "version": "1.0.6", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/fsevents/node_modules/inherits": { + "version": "2.0.4", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/ini": { + "version": "1.3.5", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "engines": { + "node": "*" + } + }, + "node_modules/fsevents/node_modules/is-fullwidth-code-point": { + "version": "1.0.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "number-is-nan": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/isarray": { + "version": "1.0.0", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/minimatch": { + "version": "3.0.4", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/fsevents/node_modules/minimist": { + "version": "1.2.5", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/minipass": { + "version": "2.9.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "safe-buffer": "^5.1.2", + "yallist": "^3.0.0" + } + }, + "node_modules/fsevents/node_modules/minizlib": { + "version": "1.3.3", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "minipass": "^2.9.0" + } + }, + "node_modules/fsevents/node_modules/mkdirp": { + "version": "0.5.3", + "deprecated": "Legacy versions of mkdirp are no longer supported. Please update to mkdirp 1.x. (Note that the API surface has changed to use Promises in 1.x.)", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "minimist": "^1.2.5" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/fsevents/node_modules/ms": { + "version": "2.1.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/needle": { + "version": "2.3.3", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "debug": "^3.2.6", + "iconv-lite": "^0.4.4", + "sax": "^1.2.4" + }, + "bin": { + "needle": "bin/needle" + }, + "engines": { + "node": ">= 4.4.x" + } + }, + "node_modules/fsevents/node_modules/node-pre-gyp": { + "version": "0.14.0", + "dev": true, + "inBundle": true, + "license": "BSD-3-Clause", + "optional": true, + "dependencies": { + "detect-libc": "^1.0.2", + "mkdirp": "^0.5.1", + "needle": "^2.2.1", + "nopt": "^4.0.1", + "npm-packlist": "^1.1.6", + "npmlog": "^4.0.2", + "rc": "^1.2.7", + "rimraf": "^2.6.1", + "semver": "^5.3.0", + "tar": "^4.4.2" + }, + "bin": { + "node-pre-gyp": "bin/node-pre-gyp" + } + }, + "node_modules/fsevents/node_modules/nopt": { + "version": "4.0.3", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "abbrev": "1", + "osenv": "^0.1.4" + }, + "bin": { + "nopt": "bin/nopt.js" + } + }, + "node_modules/fsevents/node_modules/npm-bundled": { + "version": "1.1.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "npm-normalize-package-bin": "^1.0.1" + } + }, + "node_modules/fsevents/node_modules/npm-normalize-package-bin": { + "version": "1.0.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/npm-packlist": { + "version": "1.4.8", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "ignore-walk": "^3.0.1", + "npm-bundled": "^1.0.1", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "node_modules/fsevents/node_modules/npmlog": { + "version": "4.1.2", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" + } + }, + "node_modules/fsevents/node_modules/number-is-nan": { + "version": "1.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/object-assign": { + "version": "4.1.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/once": { + "version": "1.4.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/fsevents/node_modules/os-homedir": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/os-tmpdir": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/osenv": { + "version": "0.1.5", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "os-homedir": "^1.0.0", + "os-tmpdir": "^1.0.0" + } + }, + "node_modules/fsevents/node_modules/path-is-absolute": { + "version": "1.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/process-nextick-args": { + "version": "2.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/rc": { + "version": "1.2.8", + "dev": true, + "inBundle": true, + "license": "(BSD-2-Clause OR MIT OR Apache-2.0)", + "optional": true, + "dependencies": { + "deep-extend": "^0.6.0", + "ini": "~1.3.0", + "minimist": "^1.2.0", + "strip-json-comments": "~2.0.1" + }, + "bin": { + "rc": "cli.js" + } + }, + "node_modules/fsevents/node_modules/readable-stream": { + "version": "2.3.7", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "core-util-is": "~1.0.0", + "inherits": "~2.0.3", + "isarray": "~1.0.0", + "process-nextick-args": "~2.0.0", + "safe-buffer": "~5.1.1", + "string_decoder": "~1.1.1", + "util-deprecate": "~1.0.1" + } + }, + "node_modules/fsevents/node_modules/rimraf": { + "version": "2.7.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/fsevents/node_modules/safe-buffer": { + "version": "5.1.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/safer-buffer": { + "version": "2.1.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/sax": { + "version": "1.2.4", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/semver": { + "version": "5.7.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/fsevents/node_modules/set-blocking": { + "version": "2.0.0", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/signal-exit": { + "version": "3.0.2", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/string_decoder": { + "version": "1.1.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/fsevents/node_modules/string-width": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/strip-ansi": { + "version": "3.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/strip-json-comments": { + "version": "2.0.1", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fsevents/node_modules/tar": { + "version": "4.4.13", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "chownr": "^1.1.1", + "fs-minipass": "^1.2.5", + "minipass": "^2.8.6", + "minizlib": "^1.2.1", + "mkdirp": "^0.5.0", + "safe-buffer": "^5.1.2", + "yallist": "^3.0.3" + }, + "engines": { + "node": ">=4.5" + } + }, + "node_modules/fsevents/node_modules/util-deprecate": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "MIT", + "optional": true + }, + "node_modules/fsevents/node_modules/wide-align": { + "version": "1.1.3", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true, + "dependencies": { + "string-width": "^1.0.2 || 2" } }, - "function-bind": { + "node_modules/fsevents/node_modules/wrappy": { + "version": "1.0.2", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/fsevents/node_modules/yallist": { + "version": "3.1.1", + "dev": true, + "inBundle": true, + "license": "ISC", + "optional": true + }, + "node_modules/function-bind": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", "dev": true }, - "gensync": { + "node_modules/gensync": { "version": "1.0.0-beta.1", "resolved": "https://registry.npmjs.org/gensync/-/gensync-1.0.0-beta.1.tgz", "integrity": "sha512-r8EC6NO1sngH/zdD9fiRDLdcgnbayXah+mLgManTaIZJqEC1MZstmnox8KpnI2/fxQwrp5OpCOYWLp4rBl4Jcg==", - "dev": true + "dev": true, + "engines": { + "node": ">=6.9.0" + } }, - "get-caller-file": { + "node_modules/get-caller-file": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==", - "dev": true + "dev": true, + "engines": { + "node": "6.* || 8.* || >= 10.*" + } }, - "get-port": { + "node_modules/get-port": { "version": "3.2.0", "resolved": "https://registry.npmjs.org/get-port/-/get-port-3.2.0.tgz", "integrity": "sha1-3Xzn3hh8Bsi/NTeWrHHgmfCYDrw=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "get-value": { + "node_modules/get-value": { "version": "2.0.6", "resolved": "https://registry.npmjs.org/get-value/-/get-value-2.0.6.tgz", "integrity": "sha1-3BXKHGcjh8p2vTesCjlbogQqLCg=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "getpass": { + "node_modules/getpass": { "version": "0.1.7", "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", "dev": true, - "requires": { + "dependencies": { "assert-plus": "^1.0.0" } }, - "glob": { + "node_modules/glob": { "version": "7.1.6", "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", "dev": true, - "requires": { + "dependencies": { "fs.realpath": "^1.0.0", "inflight": "^1.0.4", "inherits": "2", "minimatch": "^3.0.4", "once": "^1.3.0", "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + }, + "funding": { + "url": "https://github.com/sponsors/isaacs" } }, - "glob-parent": { + "node_modules/glob-parent": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-3.1.0.tgz", "integrity": "sha1-nmr2KZ2NO9K9QEMIMr0RPfkGxa4=", "dev": true, - "requires": { + "dependencies": { "is-glob": "^3.1.0", "path-dirname": "^1.0.0" - }, + } + }, + "node_modules/glob-parent/node_modules/is-glob": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", + "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", + "dev": true, "dependencies": { - "is-glob": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-3.1.0.tgz", - "integrity": "sha1-e6WuJCF4BKxwcHuWkiVnSGzD6Eo=", - "dev": true, - "requires": { - "is-extglob": "^2.1.0" - } - } + "is-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "glob-to-regexp": { + "node_modules/glob-to-regexp": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=", "dev": true }, - "globals": { + "node_modules/globals": { "version": "11.12.0", "resolved": "https://registry.npmjs.org/globals/-/globals-11.12.0.tgz", "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "graceful-fs": { + "node_modules/graceful-fs": { "version": "4.2.3", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.3.tgz", "integrity": "sha512-a30VEBm4PEdx1dRB7MFK7BejejvCvBronbLjht+sHuGYj8PHs7M/5Z+rt5lw551vZ7yfTCj4Vuyy3mSJytDWRQ==", "dev": true }, - "grapheme-breaker": { + "node_modules/grapheme-breaker": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/grapheme-breaker/-/grapheme-breaker-0.3.2.tgz", "integrity": "sha1-W55reMODJFLSuiuxy4MPlidkEKw=", "dev": true, - "requires": { + "dependencies": { "brfs": "^1.2.0", "unicode-trie": "^0.3.1" } }, - "har-schema": { + "node_modules/har-schema": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/har-schema/-/har-schema-2.0.0.tgz", "integrity": "sha1-qUwiJOvKwEeCoNkDVSHyRzW37JI=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "har-validator": { + "node_modules/har-validator": { "version": "5.1.3", "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.3.tgz", "integrity": "sha512-sNvOCzEQNr/qrvJgc3UG/kD4QtlHycrzwS+6mfTrrSq97BvaYcPZZI1ZSqGSPR73Cxn4LKTD4PttRwfU7jWq5g==", + "deprecated": "this library is no longer supported", "dev": true, - "requires": { + "dependencies": { "ajv": "^6.5.5", "har-schema": "^2.0.0" + }, + "engines": { + "node": ">=6" } }, - "has": { + "node_modules/has": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", "dev": true, - "requires": { + "dependencies": { "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" } }, - "has-ansi": { + "node_modules/has-ansi": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/has-ansi/-/has-ansi-2.0.0.tgz", "integrity": "sha1-NPUEnOHs3ysGSa8+8k5F7TVBbZE=", "dev": true, - "requires": { + "dependencies": { "ansi-regex": "^2.0.0" }, - "dependencies": { - "ansi-regex": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "dev": true - } + "engines": { + "node": ">=0.10.0" } }, - "has-flag": { + "node_modules/has-ansi/node_modules/ansi-regex": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "has-symbols": { + "node_modules/has-symbols": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.1.tgz", "integrity": "sha512-PLcsoqu++dmEIZB+6totNFKq/7Do+Z0u4oT0zKOJNl3lYK6vGwwu2hjHs+68OEZbTjiUE9bgOABXbP/GvrS0Kg==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, - "has-value": { + "node_modules/has-value": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-value/-/has-value-1.0.0.tgz", "integrity": "sha1-GLKB2lhbHFxR3vJMkw7SmgvmsXc=", "dev": true, - "requires": { + "dependencies": { "get-value": "^2.0.6", "has-values": "^1.0.0", "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "has-values": { + "node_modules/has-values": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/has-values/-/has-values-1.0.0.tgz", "integrity": "sha1-lbC2P+whRmGab+V/51Yo1aOe/k8=", "dev": true, - "requires": { + "dependencies": { "is-number": "^3.0.0", "kind-of": "^4.0.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-values/node_modules/kind-of": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", + "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", + "dev": true, "dependencies": { - "kind-of": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-4.0.0.tgz", - "integrity": "sha1-IIE989cSkosgc3hpGkUGb65y3Vc=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "hash-base": { + "node_modules/hash-base": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.0.4.tgz", "integrity": "sha1-X8hoaEfs1zSZQDMZprCj8/auSRg=", "dev": true, - "requires": { + "dependencies": { "inherits": "^2.0.1", "safe-buffer": "^5.0.1" + }, + "engines": { + "node": ">=4" } }, - "hash.js": { + "node_modules/hash.js": { "version": "1.1.7", "resolved": "https://registry.npmjs.org/hash.js/-/hash.js-1.1.7.tgz", "integrity": "sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==", "dev": true, - "requires": { + "dependencies": { "inherits": "^2.0.3", "minimalistic-assert": "^1.0.1" } }, - "hex-color-regex": { + "node_modules/hex-color-regex": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/hex-color-regex/-/hex-color-regex-1.1.0.tgz", "integrity": "sha512-l9sfDFsuqtOqKDsQdqrMRk0U85RZc0RtOR9yPI7mRVOa4FsR/BVnZ0shmQRM96Ji99kYZP/7hn1cedc1+ApsTQ==", "dev": true }, - "hmac-drbg": { + "node_modules/hmac-drbg": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/hmac-drbg/-/hmac-drbg-1.0.1.tgz", "integrity": "sha1-0nRXAQJabHdabFRXk+1QL8DGSaE=", "dev": true, - "requires": { + "dependencies": { "hash.js": "^1.0.3", "minimalistic-assert": "^1.0.0", "minimalistic-crypto-utils": "^1.0.1" } }, - "hsl-regex": { + "node_modules/hsl-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/hsl-regex/-/hsl-regex-1.0.0.tgz", "integrity": "sha1-1JMwx4ntgZ4nakwNJy3/owsY/m4=", "dev": true }, - "hsla-regex": { + "node_modules/hsla-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/hsla-regex/-/hsla-regex-1.0.0.tgz", "integrity": "sha1-wc56MWjIxmFAM6S194d/OyJfnDg=", "dev": true }, - "html-comment-regex": { + "node_modules/html-comment-regex": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/html-comment-regex/-/html-comment-regex-1.1.2.tgz", "integrity": "sha512-P+M65QY2JQ5Y0G9KKdlDpo0zK+/OHptU5AaBwUfAIDJZk1MYf32Frm84EcOytfJE0t5JvkAnKlmjsXDnWzCJmQ==", "dev": true }, - "html-encoding-sniffer": { + "node_modules/html-encoding-sniffer": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/html-encoding-sniffer/-/html-encoding-sniffer-1.0.2.tgz", "integrity": "sha512-71lZziiDnsuabfdYiUeWdCVyKuqwWi23L8YeIgV9jSSZHCtb6wB1BKWooH7L3tn4/FuZJMVWyNaIDr4RGmaSYw==", "dev": true, - "requires": { + "dependencies": { "whatwg-encoding": "^1.0.1" } }, - "html-tags": { + "node_modules/html-tags": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/html-tags/-/html-tags-1.2.0.tgz", "integrity": "sha1-x43mW1Zjqll5id0rerSSANfk25g=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "htmlnano": { + "node_modules/htmlnano": { "version": "0.2.5", "resolved": "https://registry.npmjs.org/htmlnano/-/htmlnano-0.2.5.tgz", "integrity": "sha512-X1iPSwXG/iF9bVs+/obt2n6F64uH0ETkA8zp7qFDmLW9/+A6ueHGeb/+qD67T21qUY22owZPMdawljN50ajkqA==", "dev": true, - "requires": { + "dependencies": { "cssnano": "^4.1.10", "normalize-html-whitespace": "^1.0.0", "posthtml": "^0.12.0", @@ -3912,209 +4750,247 @@ "svgo": "^1.3.2", "terser": "^4.3.9", "uncss": "^0.17.2" + } + }, + "node_modules/htmlnano/node_modules/posthtml": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/posthtml/-/posthtml-0.12.0.tgz", + "integrity": "sha512-aNUEP/SfKUXAt+ghG51LC5MmafChBZeslVe/SSdfKIgLGUVRE68mrMF4V8XbH07ZifM91tCSuxY3eHIFLlecQw==", + "dev": true, + "dependencies": { + "posthtml-parser": "^0.4.1", + "posthtml-render": "^1.1.5" }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/htmlnano/node_modules/terser": { + "version": "4.6.10", + "resolved": "https://registry.npmjs.org/terser/-/terser-4.6.10.tgz", + "integrity": "sha512-qbF/3UOo11Hggsbsqm2hPa6+L4w7bkr+09FNseEe8xrcVD3APGLFqE+Oz1ZKAxjYnFsj80rLOfgAtJ0LNJjtTA==", + "dev": true, "dependencies": { - "posthtml": { - "version": "0.12.0", - "resolved": "https://registry.npmjs.org/posthtml/-/posthtml-0.12.0.tgz", - "integrity": "sha512-aNUEP/SfKUXAt+ghG51LC5MmafChBZeslVe/SSdfKIgLGUVRE68mrMF4V8XbH07ZifM91tCSuxY3eHIFLlecQw==", - "dev": true, - "requires": { - "posthtml-parser": "^0.4.1", - "posthtml-render": "^1.1.5" - } - }, - "terser": { - "version": "4.6.10", - "resolved": "https://registry.npmjs.org/terser/-/terser-4.6.10.tgz", - "integrity": "sha512-qbF/3UOo11Hggsbsqm2hPa6+L4w7bkr+09FNseEe8xrcVD3APGLFqE+Oz1ZKAxjYnFsj80rLOfgAtJ0LNJjtTA==", - "dev": true, - "requires": { - "commander": "^2.20.0", - "source-map": "~0.6.1", - "source-map-support": "~0.5.12" - } - } + "commander": "^2.20.0", + "source-map": "~0.6.1", + "source-map-support": "~0.5.12" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=6.0.0" } }, - "htmlparser2": { + "node_modules/htmlparser2": { "version": "3.10.1", "resolved": "https://registry.npmjs.org/htmlparser2/-/htmlparser2-3.10.1.tgz", "integrity": "sha512-IgieNijUMbkDovyoKObU1DUhm1iwNYE/fuifEoEHfd1oZKZDaONBSkal7Y01shxsM49R4XaMdGez3WnF9UfiCQ==", "dev": true, - "requires": { + "dependencies": { "domelementtype": "^1.3.1", "domhandler": "^2.3.0", "domutils": "^1.5.1", "entities": "^1.1.1", "inherits": "^2.0.1", "readable-stream": "^3.1.1" - }, + } + }, + "node_modules/htmlparser2/node_modules/readable-stream": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", + "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", + "dev": true, "dependencies": { - "readable-stream": { - "version": "3.6.0", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.0.tgz", - "integrity": "sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA==", - "dev": true, - "requires": { - "inherits": "^2.0.3", - "string_decoder": "^1.1.1", - "util-deprecate": "^1.0.1" - } - } + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" } }, - "http-errors": { + "node_modules/http-errors": { "version": "1.7.3", "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.3.tgz", "integrity": "sha512-ZTTX0MWrsQ2ZAhA1cejAwDLycFsd7I7nVtnkT3Ol0aqodaKW+0CTZDQ1uBv5whptCnc8e8HeRRJxRs0kmm/Qfw==", "dev": true, - "requires": { + "dependencies": { "depd": "~1.1.2", "inherits": "2.0.4", "setprototypeof": "1.1.1", "statuses": ">= 1.5.0 < 2", "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" } }, - "http-signature": { + "node_modules/http-signature": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/http-signature/-/http-signature-1.2.0.tgz", "integrity": "sha1-muzZJRFHcvPZW2WmCruPfBj7rOE=", "dev": true, - "requires": { + "dependencies": { "assert-plus": "^1.0.0", "jsprim": "^1.2.2", "sshpk": "^1.7.0" + }, + "engines": { + "node": ">=0.8", + "npm": ">=1.3.7" } }, - "https-browserify": { + "node_modules/https-browserify": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/https-browserify/-/https-browserify-1.0.0.tgz", "integrity": "sha1-7AbBDgo0wPL68Zn3/X/Hj//QPHM=", "dev": true }, - "iconv-lite": { + "node_modules/iconv-lite": { "version": "0.4.24", "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", "dev": true, - "requires": { + "dependencies": { "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" } }, - "icss-replace-symbols": { + "node_modules/icss-replace-symbols": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/icss-replace-symbols/-/icss-replace-symbols-1.1.0.tgz", "integrity": "sha1-Bupvg2ead0njhs/h/oEq5dsiPe0=", "dev": true }, - "ieee754": { + "node_modules/ieee754": { "version": "1.1.13", "resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.1.13.tgz", "integrity": "sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==", "dev": true }, - "import-fresh": { + "node_modules/import-fresh": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", "dev": true, - "requires": { + "dependencies": { "caller-path": "^2.0.0", "resolve-from": "^3.0.0" + }, + "engines": { + "node": ">=4" } }, - "indexes-of": { + "node_modules/indexes-of": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/indexes-of/-/indexes-of-1.0.1.tgz", "integrity": "sha1-8w9xbI4r00bHtn0985FVZqfAVgc=", "dev": true }, - "inflight": { + "node_modules/inflight": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", "dev": true, - "requires": { + "dependencies": { "once": "^1.3.0", "wrappy": "1" } }, - "inherits": { + "node_modules/inherits": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==", "dev": true }, - "invariant": { + "node_modules/invariant": { "version": "2.2.4", "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", "dev": true, - "requires": { + "dependencies": { "loose-envify": "^1.0.0" } }, - "is-absolute-url": { + "node_modules/is-absolute-url": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-2.1.0.tgz", "integrity": "sha1-UFMN+4T8yap9vnhS6Do3uTufKqY=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "is-accessor-descriptor": { + "node_modules/is-accessor-descriptor": { "version": "0.1.6", "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-0.1.6.tgz", "integrity": "sha1-qeEss66Nh2cn7u84Q/igiXtcmNY=", + "deprecated": "Please upgrade to v0.1.7", "dev": true, - "requires": { + "dependencies": { "kind-of": "^3.0.2" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-accessor-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-arrayish": { + "node_modules/is-arrayish": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.2.1.tgz", "integrity": "sha1-d8mYQFJ6qOyxqLppe4BkWnqSap0=", "dev": true }, - "is-binary-path": { + "node_modules/is-binary-path": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", "integrity": "sha1-dfFmQrSA8YenEcgUFh/TpKdlWJg=", "dev": true, - "requires": { + "dependencies": { "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-buffer": { + "node_modules/is-buffer": { "version": "1.1.6", "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", "dev": true }, - "is-callable": { + "node_modules/is-callable": { "version": "1.1.5", "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.1.5.tgz", "integrity": "sha512-ESKv5sMCJB2jnHTWZ3O5itG+O128Hsus4K4Qh1h2/cgn2vbgnLSVqfV46AeJA9D5EeeLa9w81KUXMtn34zhX+Q==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, - "is-color-stop": { + "node_modules/is-color-stop": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-color-stop/-/is-color-stop-1.1.0.tgz", "integrity": "sha1-z/9HGu5N1cnhWFmPvhKWe1za00U=", "dev": true, - "requires": { + "dependencies": { "css-color-names": "^0.0.4", "hex-color-regex": "^1.1.0", "hsl-regex": "^1.0.0", @@ -4123,245 +4999,320 @@ "rgba-regex": "^1.0.0" } }, - "is-data-descriptor": { + "node_modules/is-data-descriptor": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "deprecated": "Please upgrade to v0.1.5", "dev": true, - "requires": { + "dependencies": { "kind-of": "^3.0.2" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-data-descriptor/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-date-object": { + "node_modules/is-date-object": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.2.tgz", "integrity": "sha512-USlDT524woQ08aoZFzh3/Z6ch9Y/EWXEHQ/AaRN0SkKq4t2Jw2R2339tSXmwuVoY7LLlBCbOIlx2myP/L5zk0g==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, - "is-descriptor": { + "node_modules/is-descriptor": { "version": "0.1.6", "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-0.1.6.tgz", "integrity": "sha512-avDYr0SB3DwO9zsMov0gKCESFYqCnE4hq/4z3TdUlukEy5t9C0YRq7HLrsN52NAcqXKaepeCD0n+B0arnVG3Hg==", "dev": true, - "requires": { + "dependencies": { "is-accessor-descriptor": "^0.1.6", "is-data-descriptor": "^0.1.4", "kind-of": "^5.0.0" }, - "dependencies": { - "kind-of": { - "version": "5.1.0", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", - "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", - "dev": true - } + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-descriptor/node_modules/kind-of": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-5.1.0.tgz", + "integrity": "sha512-NGEErnH6F2vUuXDh+OlbcKW7/wOcfdRHaZ7VWtqCztfHri/++YKmP51OdWeGPuqCOba6kk2OTe5d02VmTB80Pw==", + "dev": true, + "engines": { + "node": ">=0.10.0" } }, - "is-directory": { + "node_modules/is-directory": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/is-directory/-/is-directory-0.3.1.tgz", "integrity": "sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "is-extendable": { + "node_modules/is-extendable": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "is-extglob": { + "node_modules/is-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "is-fullwidth-code-point": { + "node_modules/is-fullwidth-code-point": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-2.0.0.tgz", "integrity": "sha1-o7MKXE8ZkYMWeqq5O+764937ZU8=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "is-glob": { + "node_modules/is-glob": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz", "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==", "dev": true, - "requires": { + "dependencies": { "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-html": { + "node_modules/is-html": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-html/-/is-html-1.1.0.tgz", "integrity": "sha1-4E8cGNOUhRETlvmgJz6rUa8hhGQ=", "dev": true, - "requires": { + "dependencies": { "html-tags": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-number": { + "node_modules/is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", "integrity": "sha1-JP1iAaR4LPUFYcgQJ2r8fRLXEZU=", "dev": true, - "requires": { + "dependencies": { "kind-of": "^3.0.2" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-number/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-obj": { + "node_modules/is-obj": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/is-obj/-/is-obj-2.0.0.tgz", "integrity": "sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w==", - "dev": true + "dev": true, + "engines": { + "node": ">=8" + } }, - "is-plain-object": { + "node_modules/is-plain-object": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", "integrity": "sha512-h5PpgXkWitc38BBMYawTYMWJHFZJVnBquFE57xFpjB8pJFiF6gZ+bU+WyI/yqXiFR5mdLsgYNaPe8uao6Uv9Og==", "dev": true, - "requires": { + "dependencies": { "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "is-regex": { + "node_modules/is-regex": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.0.5.tgz", "integrity": "sha512-vlKW17SNq44owv5AQR3Cq0bQPEb8+kF3UKZ2fiZNOWtztYE5i0CzCZxFDwO58qAOWtxdBRVO/V5Qin1wjCqFYQ==", "dev": true, - "requires": { + "dependencies": { "has": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "is-resolvable": { + "node_modules/is-resolvable": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-resolvable/-/is-resolvable-1.1.0.tgz", "integrity": "sha512-qgDYXFSR5WvEfuS5dMj6oTMEbrrSaM0CrFk2Yiq/gXnBvD9pMa2jGXxyhGLfvhZpuMZe18CJpFxAt3CRs42NMg==", "dev": true }, - "is-svg": { + "node_modules/is-svg": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-svg/-/is-svg-3.0.0.tgz", "integrity": "sha512-gi4iHK53LR2ujhLVVj+37Ykh9GLqYHX6JOVXbLAucaG/Cqw9xwdFOjDM2qeifLs1sF1npXXFvDu0r5HNgCMrzQ==", "dev": true, - "requires": { + "dependencies": { "html-comment-regex": "^1.1.0" + }, + "engines": { + "node": ">=4" } }, - "is-symbol": { + "node_modules/is-symbol": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/is-symbol/-/is-symbol-1.0.3.tgz", "integrity": "sha512-OwijhaRSgqvhm/0ZdAcXNZt9lYdKFpcRDT5ULUuYXPoT794UNOdU+gpT6Rzo7b4V2HUl/op6GqY894AZwv9faQ==", "dev": true, - "requires": { + "dependencies": { "has-symbols": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "is-typedarray": { + "node_modules/is-typedarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-typedarray/-/is-typedarray-1.0.0.tgz", "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", "dev": true }, - "is-url": { + "node_modules/is-url": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/is-url/-/is-url-1.2.4.tgz", "integrity": "sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww==", "dev": true }, - "is-windows": { + "node_modules/is-windows": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "is-wsl": { + "node_modules/is-wsl": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-1.1.0.tgz", "integrity": "sha1-HxbkqiKwTRM2tmGIpmrzxgDDpm0=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "isarray": { + "node_modules/isarray": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", "dev": true }, - "isexe": { + "node_modules/isexe": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz", "integrity": "sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=", "dev": true }, - "isobject": { + "node_modules/isobject": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "isstream": { + "node_modules/isstream": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/isstream/-/isstream-0.1.2.tgz", "integrity": "sha1-R+Y/evVa+m+S4VAOaQ64uFKcCZo=", "dev": true }, - "js-tokens": { + "node_modules/js-tokens": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==", "dev": true }, - "js-yaml": { + "node_modules/js-yaml": { "version": "3.13.1", "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.13.1.tgz", "integrity": "sha512-YfbcO7jXDdyj0DGxYVSlSeQNHbD7XPWvrVWeVUujrQEoZzWJIRrCPoyk6kL6IAjAG2IolMK4T0hNUe0HOUs5Jw==", "dev": true, - "requires": { + "dependencies": { "argparse": "^1.0.7", "esprima": "^4.0.0" }, - "dependencies": { - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - } + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/js-yaml/node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" } }, - "jsbn": { + "node_modules/jsbn": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", "dev": true }, - "jsdom": { + "node_modules/jsdom": { "version": "14.1.0", "resolved": "https://registry.npmjs.org/jsdom/-/jsdom-14.1.0.tgz", "integrity": "sha512-O901mfJSuTdwU2w3Sn+74T+RnDVP+FuV5fH8tcPWyqrseRAb0s5xOtPgCFiPOtLcyK7CLIJwPyD83ZqQWvA5ng==", "dev": true, - "requires": { + "dependencies": { "abab": "^2.0.0", "acorn": "^6.0.4", "acorn-globals": "^4.3.0", @@ -4389,253 +5340,319 @@ "ws": "^6.1.2", "xml-name-validator": "^3.0.0" }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jsdom/node_modules/acorn": { + "version": "6.4.1", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.1.tgz", + "integrity": "sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/jsdom/node_modules/escodegen": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.1.tgz", + "integrity": "sha512-Bmt7NcRySdIfNPfU2ZoXDrrXsG9ZjvDxcAlMfDUgRBjLOWTuIACXPBFJH7Z+cLb40JeQco5toikyc9t9P8E9SQ==", + "dev": true, + "dependencies": { + "esprima": "^4.0.1", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=4.0" + }, + "optionalDependencies": { + "source-map": "~0.6.1" + } + }, + "node_modules/jsdom/node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/jsdom/node_modules/ws": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", + "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", + "dev": true, "dependencies": { - "acorn": { - "version": "6.4.1", - "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.1.tgz", - "integrity": "sha512-ZVA9k326Nwrj3Cj9jlh3wGFutC2ZornPNARZwsNYqQYgN0EsV2d53w5RN/co65Ohn4sUAUtb1rSUAOD6XN9idA==", - "dev": true - }, - "escodegen": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.1.tgz", - "integrity": "sha512-Bmt7NcRySdIfNPfU2ZoXDrrXsG9ZjvDxcAlMfDUgRBjLOWTuIACXPBFJH7Z+cLb40JeQco5toikyc9t9P8E9SQ==", - "dev": true, - "requires": { - "esprima": "^4.0.1", - "estraverse": "^4.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - } - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - }, - "ws": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", - "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", - "dev": true, - "requires": { - "async-limiter": "~1.0.0" - } - } + "async-limiter": "~1.0.0" } }, - "jsesc": { + "node_modules/jsesc": { "version": "2.5.2", "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", - "dev": true + "dev": true, + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } }, - "json-parse-better-errors": { + "node_modules/json-parse-better-errors": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/json-parse-better-errors/-/json-parse-better-errors-1.0.2.tgz", "integrity": "sha512-mrqyZKfX5EhL7hvqcV6WG1yYjnjeuYDzDhhcAAUrq8Po85NBQBJP+ZDUT75qZQ98IkUoBqdkExkukOU7Ts2wrw==", "dev": true }, - "json-schema": { + "node_modules/json-schema": { "version": "0.2.3", "resolved": "https://registry.npmjs.org/json-schema/-/json-schema-0.2.3.tgz", "integrity": "sha1-tIDIkuWaLwWVTOcnvT8qTogvnhM=", "dev": true }, - "json-schema-traverse": { + "node_modules/json-schema-traverse": { "version": "0.4.1", "resolved": "https://registry.npmjs.org/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz", "integrity": "sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg==", "dev": true }, - "json-stringify-safe": { + "node_modules/json-stringify-safe": { "version": "5.0.1", "resolved": "https://registry.npmjs.org/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz", "integrity": "sha1-Epai1Y/UXxmg9s4B1lcB4sc1tus=", "dev": true }, - "json5": { + "node_modules/json5": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", "dev": true, - "requires": { + "dependencies": { "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" } }, - "jsprim": { + "node_modules/jsprim": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", "dev": true, - "requires": { + "engines": [ + "node >=0.6.0" + ], + "dependencies": { "assert-plus": "1.0.0", "extsprintf": "1.3.0", "json-schema": "0.2.3", "verror": "1.10.0" } }, - "kind-of": { + "node_modules/kind-of": { "version": "6.0.3", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", "integrity": "sha512-dcS1ul+9tmeD95T+x28/ehLgd9mENa3LsvDTtzm3vyBEO7RPptvAD+t44WVXaUjTBRcrpFeFlC8WCruUR456hw==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "leven": { + "node_modules/leven": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/leven/-/leven-3.1.0.tgz", "integrity": "sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A==", - "dev": true + "dev": true, + "engines": { + "node": ">=6" + } }, - "levenary": { + "node_modules/levenary": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/levenary/-/levenary-1.1.1.tgz", "integrity": "sha512-mkAdOIt79FD6irqjYSs4rdbnlT5vRonMEvBVPVb3XmevfS8kgRXwfes0dhPdEtzTWD/1eNE/Bm/G1iRt6DcnQQ==", "dev": true, - "requires": { + "dependencies": { "leven": "^3.1.0" + }, + "engines": { + "node": ">= 6" } }, - "levn": { + "node_modules/levn": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/levn/-/levn-0.3.0.tgz", "integrity": "sha1-OwmSTt+fCDwEkP3UwLxEIeBHZO4=", "dev": true, - "requires": { + "dependencies": { "prelude-ls": "~1.1.2", "type-check": "~0.3.2" + }, + "engines": { + "node": ">= 0.8.0" } }, - "locate-path": { + "node_modules/locate-path": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-2.0.0.tgz", "integrity": "sha1-K1aLJl7slExtnA3pw9u7ygNUzY4=", "dev": true, - "requires": { + "dependencies": { "p-locate": "^2.0.0", "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=4" } }, - "lodash": { + "node_modules/lodash": { "version": "4.17.15", "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.15.tgz", "integrity": "sha512-8xOcRHvCjnocdS5cpwXQXVzmmh5e5+saE2QGoeQmbKmRS6J3VQppPOIt0MnmE+4xlZoumy0GPG0D0MVIQbNA1A==", "dev": true }, - "lodash.clone": { + "node_modules/lodash.clone": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/lodash.clone/-/lodash.clone-4.5.0.tgz", "integrity": "sha1-GVhwRQ9aExkkeN9Lw9I9LeoZB7Y=", "dev": true }, - "lodash.memoize": { + "node_modules/lodash.memoize": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/lodash.memoize/-/lodash.memoize-4.1.2.tgz", "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", "dev": true }, - "lodash.sortby": { + "node_modules/lodash.sortby": { "version": "4.7.0", "resolved": "https://registry.npmjs.org/lodash.sortby/-/lodash.sortby-4.7.0.tgz", "integrity": "sha1-7dFMgk4sycHgsKG0K7UhBRakJDg=", "dev": true }, - "lodash.uniq": { + "node_modules/lodash.uniq": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", "integrity": "sha1-0CJTc662Uq3BvILklFM5qEJ1R3M=", "dev": true }, - "log-symbols": { + "node_modules/log-symbols": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-2.2.0.tgz", "integrity": "sha512-VeIAFslyIerEJLXHziedo2basKbMKtTw3vfn5IzG0XTjhAVEJyNHnL2p7vc+wBDSdQuUpNw3M2u6xb9QsAY5Eg==", "dev": true, - "requires": { + "dependencies": { "chalk": "^2.0.1" + }, + "engines": { + "node": ">=4" } }, - "loose-envify": { + "node_modules/loose-envify": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz", "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==", "dev": true, - "requires": { + "dependencies": { "js-tokens": "^3.0.0 || ^4.0.0" + }, + "bin": { + "loose-envify": "cli.js" } }, - "magic-string": { + "node_modules/magic-string": { "version": "0.22.5", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.22.5.tgz", "integrity": "sha512-oreip9rJZkzvA8Qzk9HFs8fZGF/u7H/gtrE8EN6RjKJ9kh2HlC+yQ2QezifqTZfGyiuAV0dRv5a+y/8gBb1m9w==", "dev": true, - "requires": { + "dependencies": { "vlq": "^0.2.2" } }, - "map-cache": { + "node_modules/map-cache": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/map-cache/-/map-cache-0.2.2.tgz", "integrity": "sha1-wyq9C9ZSXZsFFkW7TyasXcmKDb8=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "map-visit": { + "node_modules/map-visit": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/map-visit/-/map-visit-1.0.0.tgz", "integrity": "sha1-7Nyo8TFE5mDxtb1B8S80edmN+48=", "dev": true, - "requires": { + "dependencies": { "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "md5.js": { + "node_modules/md5.js": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", "integrity": "sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==", "dev": true, - "requires": { + "dependencies": { "hash-base": "^3.0.0", "inherits": "^2.0.1", "safe-buffer": "^5.1.2" } }, - "mdn-data": { + "node_modules/mdn-data": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.4.tgz", "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==", "dev": true }, - "merge-source-map": { + "node_modules/merge-source-map": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/merge-source-map/-/merge-source-map-1.0.4.tgz", "integrity": "sha1-pd5GU42uhNQRTMXqArR3KmNGcB8=", "dev": true, - "requires": { - "source-map": "^0.5.6" - }, "dependencies": { - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } + "source-map": "^0.5.6" + } + }, + "node_modules/merge-source-map/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true, + "engines": { + "node": ">=0.10.0" } }, - "merge2": { + "node_modules/merge2": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.3.0.tgz", "integrity": "sha512-2j4DAdlBOkiSZIsaXk4mTE3sRS02yBHAtfy127xRV3bQUFqXkjHCHLW6Scv7DwNRbIWNHH8zpnz9zMaKXIdvYw==", - "dev": true + "dev": true, + "engines": { + "node": ">= 6" + } }, - "micromatch": { + "node_modules/micromatch": { "version": "3.1.10", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", "integrity": "sha512-MWikgl9n9M3w+bpsY3He8L+w9eF9338xRl8IAO5viDizwSzziFEyUzo2xrrloB64ADbTf8uA8vRqqttDTOmccg==", "dev": true, - "requires": { + "dependencies": { "arr-diff": "^4.0.0", "array-unique": "^0.3.2", "braces": "^2.3.1", @@ -4649,121 +5666,152 @@ "regex-not": "^1.0.0", "snapdragon": "^0.8.1", "to-regex": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "miller-rabin": { + "node_modules/miller-rabin": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/miller-rabin/-/miller-rabin-4.0.1.tgz", "integrity": "sha512-115fLhvZVqWwHPbClyntxEVfVDfl9DLLTuJvq3g2O/Oxi8AiNouAHvDSzHS0viUJc+V5vm3eq91Xwqn9dp4jRA==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.0.0", "brorand": "^1.0.1" + }, + "bin": { + "miller-rabin": "bin/miller-rabin" } }, - "mime": { + "node_modules/mime": { "version": "1.6.0", "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==", - "dev": true + "dev": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } }, - "mime-db": { + "node_modules/mime-db": { "version": "1.43.0", "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.43.0.tgz", "integrity": "sha512-+5dsGEEovYbT8UY9yD7eE4XTc4UwJ1jBYlgaQQF38ENsKR3wj/8q8RFZrF9WIZpB2V1ArTVFUva8sAul1NzRzQ==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "mime-types": { + "node_modules/mime-types": { "version": "2.1.26", "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.26.tgz", "integrity": "sha512-01paPWYgLrkqAyrlDorC1uDwl2p3qZT7yl806vW7DvDoxwXi46jsjFbg+WdwotBIk6/MbEhO/dh5aZ5sNj/dWQ==", "dev": true, - "requires": { + "dependencies": { "mime-db": "1.43.0" + }, + "engines": { + "node": ">= 0.6" } }, - "mimic-fn": { + "node_modules/mimic-fn": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-1.2.0.tgz", "integrity": "sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "minimalistic-assert": { + "node_modules/minimalistic-assert": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz", "integrity": "sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==", "dev": true }, - "minimalistic-crypto-utils": { + "node_modules/minimalistic-crypto-utils": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz", "integrity": "sha1-9sAMHAsIIkblxNmd+4x8CDsrWCo=", "dev": true }, - "minimatch": { + "node_modules/minimatch": { "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", "dev": true, - "requires": { + "dependencies": { "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" } }, - "minimist": { + "node_modules/minimist": { "version": "1.2.5", "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==", "dev": true }, - "mixin-deep": { + "node_modules/mixin-deep": { "version": "1.3.2", "resolved": "https://registry.npmjs.org/mixin-deep/-/mixin-deep-1.3.2.tgz", "integrity": "sha512-WRoDn//mXBiJ1H40rqa3vH0toePwSsGb45iInWlTySa+Uu4k3tYUSxa2v1KqAiLtvlrSzaExqS1gtk96A9zvEA==", "dev": true, - "requires": { + "dependencies": { "for-in": "^1.0.2", "is-extendable": "^1.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/mixin-deep/node_modules/is-extendable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", + "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", + "dev": true, "dependencies": { - "is-extendable": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-1.0.1.tgz", - "integrity": "sha512-arnXMxT1hhoKo9k1LZdmlNyJdDDfy2v0fXjFlmok4+i8ul/6WlbVge9bhM74OpNPQPMGUToDtz+KXa1PneJxOA==", - "dev": true, - "requires": { - "is-plain-object": "^2.0.4" - } - } + "is-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" } }, - "mkdirp": { + "node_modules/mkdirp": { "version": "0.5.5", "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.5.tgz", "integrity": "sha512-NKmAlESf6jMGym1++R0Ra7wvhV+wFW63FaSOFPwRahvea0gMUcGUhVeAg/0BC0wiv9ih5NYPB1Wn1UEI1/L+xQ==", "dev": true, - "requires": { + "dependencies": { "minimist": "^1.2.5" + }, + "bin": { + "mkdirp": "bin/cmd.js" } }, - "ms": { + "node_modules/ms": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==", "dev": true }, - "nan": { + "node_modules/nan": { "version": "2.14.0", "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.0.tgz", "integrity": "sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg==", "dev": true, "optional": true }, - "nanomatch": { + "node_modules/nanomatch": { "version": "1.2.13", "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", "integrity": "sha512-fpoe2T0RbHwBTBUOftAfBPaDEi06ufaUai0mE6Yn1kacc3SnTErfb/h+X94VXzI64rKFHYImXSvdwGGCmwOqCA==", "dev": true, - "requires": { + "dependencies": { "arr-diff": "^4.0.0", "array-unique": "^0.3.2", "define-property": "^2.0.2", @@ -4775,32 +5823,38 @@ "regex-not": "^1.0.0", "snapdragon": "^0.8.1", "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "nice-try": { + "node_modules/nice-try": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/nice-try/-/nice-try-1.0.5.tgz", "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", "dev": true }, - "node-addon-api": { + "node_modules/node-addon-api": { "version": "1.7.1", "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-1.7.1.tgz", "integrity": "sha512-2+DuKodWvwRTrCfKOeR24KIc5unKjOh8mz17NCzVnHWfjAdDqbfbjqh7gUT+BkXBRQM52+xCHciKWonJ3CbJMQ==", "dev": true }, - "node-forge": { + "node_modules/node-forge": { "version": "0.7.6", "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.7.6.tgz", "integrity": "sha512-sol30LUpz1jQFBjOKwbjxijiE3b6pjd74YwfD0fJOKPjF+fONKb2Yg8rYgS6+bK6VDl+/wfr4IYpC7jDzLUIfw==", - "dev": true + "dev": true, + "engines": { + "node": "*" + } }, - "node-libs-browser": { + "node_modules/node-libs-browser": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/node-libs-browser/-/node-libs-browser-2.2.1.tgz", "integrity": "sha512-h/zcD8H9kaDZ9ALUWwlBUDo6TKF8a7qBSCSEGfjTVIYeqsioSKaAX+BN7NgiMGp6iSIXZ3PxgCu8KS3b71YK5Q==", "dev": true, - "requires": { + "dependencies": { "assert": "^1.1.1", "browserify-zlib": "^0.2.0", "buffer": "^4.3.0", @@ -4824,268 +5878,338 @@ "url": "^0.11.0", "util": "^0.11.0", "vm-browserify": "^1.0.1" - }, - "dependencies": { - "punycode": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", - "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", - "dev": true - } } }, - "node-releases": { + "node_modules/node-libs-browser/node_modules/punycode": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.4.1.tgz", + "integrity": "sha1-wNWmOycYgArY4esPpSachN1BhF4=", + "dev": true + }, + "node_modules/node-releases": { "version": "1.1.53", "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.53.tgz", "integrity": "sha512-wp8zyQVwef2hpZ/dJH7SfSrIPD6YoJz6BDQDpGEkcA0s3LpAQoxBIYmfIq6QAhC1DhwsyCgTaTTcONwX8qzCuQ==", "dev": true }, - "normalize-html-whitespace": { + "node_modules/normalize-html-whitespace": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/normalize-html-whitespace/-/normalize-html-whitespace-1.0.0.tgz", "integrity": "sha512-9ui7CGtOOlehQu0t/OhhlmDyc71mKVlv+4vF+me4iZLPrNtRL2xoquEdfZxasC/bdQi/Hr3iTrpyRKIG+ocabA==", - "dev": true + "dev": true, + "engines": { + "node": ">= 8" + } }, - "normalize-path": { + "node_modules/normalize-path": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz", "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "normalize-url": { + "node_modules/normalize-url": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-3.3.0.tgz", "integrity": "sha512-U+JJi7duF1o+u2pynbp2zXDW2/PADgC30f0GsHZtRh+HOcXHnw137TrNlyxxRvWW5fjKd3bcLHPxofWuCjaeZg==", - "dev": true + "dev": true, + "engines": { + "node": ">=6" + } }, - "nth-check": { + "node_modules/nth-check": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-1.0.2.tgz", "integrity": "sha512-WeBOdju8SnzPN5vTUJYxYUxLeXpCaVP5i5e0LF8fg7WORF2Wd7wFX/pk0tYZk7s8T+J7VLy0Da6J1+wCT0AtHg==", "dev": true, - "requires": { + "dependencies": { "boolbase": "~1.0.0" } }, - "nwsapi": { + "node_modules/nwsapi": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/nwsapi/-/nwsapi-2.2.0.tgz", "integrity": "sha512-h2AatdwYH+JHiZpv7pt/gSX1XoRGb7L/qSIeuqA6GwYoF9w1vP1cw42TO0aI2pNyshRK5893hNSl+1//vHK7hQ==", "dev": true }, - "oauth-sign": { + "node_modules/oauth-sign": { "version": "0.9.0", "resolved": "https://registry.npmjs.org/oauth-sign/-/oauth-sign-0.9.0.tgz", "integrity": "sha512-fexhUFFPTGV8ybAtSIGbV6gOkSv8UtRbDBnAyLQw4QPKkgNlsH2ByPGtMUqdWkos6YCRmAqViwgZrJc/mRDzZQ==", - "dev": true + "dev": true, + "engines": { + "node": "*" + } }, - "object-assign": { + "node_modules/object-assign": { "version": "4.1.1", "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz", "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "object-copy": { + "node_modules/object-copy": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", "integrity": "sha1-fn2Fi3gb18mRpBupde04EnVOmYw=", "dev": true, - "requires": { + "dependencies": { "copy-descriptor": "^0.1.0", "define-property": "^0.2.5", "kind-of": "^3.0.3" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "object-inspect": { + "node_modules/object-copy/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-inspect": { "version": "1.4.1", "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.4.1.tgz", "integrity": "sha512-wqdhLpfCUbEsoEwl3FXwGyv8ief1k/1aUdIPCqVnupM6e8l63BEJdiF/0swtn04/8p05tG/T0FrpTlfwvljOdw==", "dev": true }, - "object-keys": { + "node_modules/object-keys": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", "integrity": "sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.4" + } }, - "object-visit": { + "node_modules/object-visit": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/object-visit/-/object-visit-1.0.1.tgz", "integrity": "sha1-95xEk68MU3e1n+OdOV5BBC3QRbs=", "dev": true, - "requires": { + "dependencies": { "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "object.assign": { + "node_modules/object.assign": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.0.tgz", "integrity": "sha512-exHJeq6kBKj58mqGyTQ9DFvrZC/eR6OwxzoM9YRoGBqrXYonaFyGiFMuc9VZrXf7DarreEwMpurG3dd+CNyW5w==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.2", "function-bind": "^1.1.1", "has-symbols": "^1.0.0", "object-keys": "^1.0.11" + }, + "engines": { + "node": ">= 0.4" } }, - "object.getownpropertydescriptors": { + "node_modules/object.getownpropertydescriptors": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/object.getownpropertydescriptors/-/object.getownpropertydescriptors-2.1.0.tgz", "integrity": "sha512-Z53Oah9A3TdLoblT7VKJaTDdXdT+lQO+cNpKVnya5JDe9uLvzu1YyY1yFDFrcxrlRgWrEFH0jJtD/IbuwjcEVg==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.0-next.1" + }, + "engines": { + "node": ">= 0.8" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "object.pick": { + "node_modules/object.pick": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", "integrity": "sha1-h6EKxMFpS9Lhy/U1kaZhQftd10c=", "dev": true, - "requires": { + "dependencies": { "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "object.values": { + "node_modules/object.values": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/object.values/-/object.values-1.1.1.tgz", "integrity": "sha512-WTa54g2K8iu0kmS/us18jEmdv1a4Wi//BZ/DTVYEcH0XhLM5NYdpDHja3gt57VrZLcNAO2WGA+KpWsDBaHt6eA==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.0-next.1", "function-bind": "^1.1.1", "has": "^1.0.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "on-finished": { + "node_modules/on-finished": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", "dev": true, - "requires": { + "dependencies": { "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" } }, - "once": { + "node_modules/once": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", "dev": true, - "requires": { + "dependencies": { "wrappy": "1" } }, - "onetime": { + "node_modules/onetime": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/onetime/-/onetime-2.0.1.tgz", "integrity": "sha1-BnQoIw/WdEOyeUsiu6UotoZ5YtQ=", "dev": true, - "requires": { + "dependencies": { "mimic-fn": "^1.0.0" + }, + "engines": { + "node": ">=4" } }, - "opn": { + "node_modules/opn": { "version": "5.5.0", "resolved": "https://registry.npmjs.org/opn/-/opn-5.5.0.tgz", "integrity": "sha512-PqHpggC9bLV0VeWcdKhkpxY+3JTzetLSqTCWL/z/tFIbI6G8JCjondXklT1JinczLz2Xib62sSp0T/gKT4KksA==", "dev": true, - "requires": { + "dependencies": { "is-wsl": "^1.1.0" + }, + "engines": { + "node": ">=4" } }, - "optionator": { + "node_modules/optionator": { "version": "0.8.3", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.8.3.tgz", "integrity": "sha512-+IW9pACdk3XWmmTXG8m3upGUJst5XRGzxMRjXzAuJ1XnIFNvfhjjIuYkDvysnPQ7qzqVzLt78BCruntqRhWQbA==", "dev": true, - "requires": { + "dependencies": { "deep-is": "~0.1.3", "fast-levenshtein": "~2.0.6", "levn": "~0.3.0", "prelude-ls": "~1.1.2", "type-check": "~0.3.2", "word-wrap": "~1.2.3" + }, + "engines": { + "node": ">= 0.8.0" } }, - "ora": { + "node_modules/ora": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/ora/-/ora-2.1.0.tgz", "integrity": "sha512-hNNlAd3gfv/iPmsNxYoAPLvxg7HuPozww7fFonMZvL84tP6Ox5igfk5j/+a9rtJJwqMgKK+JgWsAQik5o0HTLA==", "dev": true, - "requires": { + "dependencies": { "chalk": "^2.3.1", "cli-cursor": "^2.1.0", "cli-spinners": "^1.1.0", "log-symbols": "^2.2.0", "strip-ansi": "^4.0.0", "wcwidth": "^1.0.1" + }, + "engines": { + "node": ">=4" } }, - "os-browserify": { + "node_modules/os-browserify": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/os-browserify/-/os-browserify-0.3.0.tgz", "integrity": "sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc=", "dev": true }, - "p-limit": { + "node_modules/p-limit": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-1.3.0.tgz", "integrity": "sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q==", "dev": true, - "requires": { + "dependencies": { "p-try": "^1.0.0" + }, + "engines": { + "node": ">=4" } }, - "p-locate": { + "node_modules/p-locate": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-2.0.0.tgz", "integrity": "sha1-IKAQOyIqcMj9OcwuWAaA893l7EM=", "dev": true, - "requires": { + "dependencies": { "p-limit": "^1.1.0" + }, + "engines": { + "node": ">=4" } }, - "p-try": { + "node_modules/p-try": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/p-try/-/p-try-1.0.0.tgz", "integrity": "sha1-y8ec26+P1CKOE/Yh8rGiN8GyB7M=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "pako": { + "node_modules/pako": { "version": "0.2.9", "resolved": "https://registry.npmjs.org/pako/-/pako-0.2.9.tgz", "integrity": "sha1-8/dSL073gjSNqBYbrZ7P1Rv4OnU=", "dev": true }, - "parcel-bundler": { + "node_modules/parcel-bundler": { "version": "1.12.4", "resolved": "https://registry.npmjs.org/parcel-bundler/-/parcel-bundler-1.12.4.tgz", "integrity": "sha512-G+iZGGiPEXcRzw0fiRxWYCKxdt/F7l9a0xkiU4XbcVRJCSlBnioWEwJMutOCCpoQmaQtjB4RBHDGIHN85AIhLQ==", + "deprecated": "Parcel v1 is no longer maintained. Please migrate to v2, which is published under the 'parcel' package. See https://v2.parceljs.org/getting-started/migration for details.", "dev": true, - "requires": { + "hasInstallScript": true, + "dependencies": { "@babel/code-frame": "^7.0.0", "@babel/core": "^7.4.4", "@babel/generator": "^7.4.4", @@ -5145,14 +6269,20 @@ "terser": "^3.7.3", "v8-compile-cache": "^2.0.0", "ws": "^5.1.1" + }, + "bin": { + "parcel": "bin/cli.js" + }, + "engines": { + "node": ">= 6.0.0" } }, - "parse-asn1": { + "node_modules/parse-asn1": { "version": "5.1.5", "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.5.tgz", "integrity": "sha512-jkMYn1dcJqF6d5CpU689bq7w/b5ALS9ROVSpQDPrZsqqesUJii9qutvoT5ltGedNXMO2e16YUWIghG9KxaViTQ==", "dev": true, - "requires": { + "dependencies": { "asn1.js": "^4.0.0", "browserify-aes": "^1.0.0", "create-hash": "^1.1.0", @@ -5161,234 +6291,276 @@ "safe-buffer": "^5.1.1" } }, - "parse-json": { + "node_modules/parse-json": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", "dev": true, - "requires": { + "dependencies": { "error-ex": "^1.3.1", "json-parse-better-errors": "^1.0.1" + }, + "engines": { + "node": ">=4" } }, - "parse5": { + "node_modules/parse5": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/parse5/-/parse5-5.1.0.tgz", "integrity": "sha512-fxNG2sQjHvlVAYmzBZS9YlDp6PTSSDwa98vkD4QgVDDCAo84z5X1t5XyJQ62ImdLXx5NdIIfihey6xpum9/gRQ==", "dev": true }, - "parseurl": { + "node_modules/parseurl": { "version": "1.3.3", "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.8" + } }, - "pascalcase": { + "node_modules/pascalcase": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "path-browserify": { + "node_modules/path-browserify": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/path-browserify/-/path-browserify-0.0.1.tgz", "integrity": "sha512-BapA40NHICOS+USX9SN4tyhq+A2RrN/Ws5F0Z5aMHDp98Fl86lX8Oti8B7uN93L4Ifv4fHOEA+pQw87gmMO/lQ==", "dev": true }, - "path-dirname": { + "node_modules/path-dirname": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/path-dirname/-/path-dirname-1.0.2.tgz", "integrity": "sha1-zDPSTVJeCZpTiMAzbG4yuRYGCeA=", "dev": true }, - "path-exists": { + "node_modules/path-exists": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "path-is-absolute": { + "node_modules/path-is-absolute": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "path-key": { + "node_modules/path-key": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/path-key/-/path-key-2.0.1.tgz", "integrity": "sha1-QRyttXTFoUDTpLGRDUDYDMn0C0A=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "path-parse": { + "node_modules/path-parse": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz", "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==", "dev": true }, - "pbkdf2": { + "node_modules/pbkdf2": { "version": "3.0.17", "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.0.17.tgz", "integrity": "sha512-U/il5MsrZp7mGg3mSQfn742na2T+1/vHDCG5/iTI3X9MKUuYUZVLQhyRsg06mCgDBTd57TxzgZt7P+fYfjRLtA==", "dev": true, - "requires": { + "dependencies": { "create-hash": "^1.1.2", "create-hmac": "^1.1.4", "ripemd160": "^2.0.1", "safe-buffer": "^5.0.1", "sha.js": "^2.4.8" + }, + "engines": { + "node": ">=0.12" } }, - "performance-now": { + "node_modules/performance-now": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/performance-now/-/performance-now-2.1.0.tgz", "integrity": "sha1-Ywn04OX6kT7BxpMHrjZLSzd8nns=", "dev": true }, - "physical-cpu-count": { + "node_modules/physical-cpu-count": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/physical-cpu-count/-/physical-cpu-count-2.0.0.tgz", "integrity": "sha1-GN4vl+S/epVRrXURlCtUlverpmA=", "dev": true }, - "pkg-up": { + "node_modules/pkg-up": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/pkg-up/-/pkg-up-2.0.0.tgz", "integrity": "sha1-yBmscoBZpGHKscOImivjxJoATX8=", "dev": true, - "requires": { + "dependencies": { "find-up": "^2.1.0" + }, + "engines": { + "node": ">=4" } }, - "pn": { + "node_modules/pn": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/pn/-/pn-1.1.0.tgz", "integrity": "sha512-2qHaIQr2VLRFoxe2nASzsV6ef4yOOH+Fi9FBOVH6cqeSgUnoyySPZkxzLuzd+RYOQTRpROA0ztTMqxROKSb/nA==", "dev": true }, - "posix-character-classes": { + "node_modules/posix-character-classes": { "version": "0.1.1", "resolved": "https://registry.npmjs.org/posix-character-classes/-/posix-character-classes-0.1.1.tgz", "integrity": "sha1-AerA/jta9xoqbAL+q7jB/vfgDqs=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "postcss": { + "node_modules/postcss": { "version": "7.0.27", "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.27.tgz", "integrity": "sha512-WuQETPMcW9Uf1/22HWUWP9lgsIC+KEHg2kozMflKjbeUtw9ujvFX6QmIfozaErDkmLWS9WEnEdEe6Uo9/BNTdQ==", "dev": true, - "requires": { + "dependencies": { "chalk": "^2.4.2", "source-map": "^0.6.1", "supports-color": "^6.1.0" }, - "dependencies": { - "supports-color": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", - "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", - "dev": true, - "requires": { - "has-flag": "^3.0.0" - } - } + "engines": { + "node": ">=6.0.0" + }, + "funding": { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/postcss" } }, - "postcss-calc": { + "node_modules/postcss-calc": { "version": "7.0.2", "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-7.0.2.tgz", "integrity": "sha512-rofZFHUg6ZIrvRwPeFktv06GdbDYLcGqh9EwiMutZg+a0oePCCw1zHOEiji6LCpyRcjTREtPASuUqeAvYlEVvQ==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.27", "postcss-selector-parser": "^6.0.2", "postcss-value-parser": "^4.0.2" - }, - "dependencies": { - "postcss-value-parser": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.0.3.tgz", - "integrity": "sha512-N7h4pG+Nnu5BEIzyeaaIYWs0LI5XC40OrRh5L60z0QjFsqGWcHcbkBvpe1WYpcIS9yQ8sOi/vIPt1ejQCrMVrg==", - "dev": true - } } }, - "postcss-colormin": { + "node_modules/postcss-calc/node_modules/postcss-value-parser": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.0.3.tgz", + "integrity": "sha512-N7h4pG+Nnu5BEIzyeaaIYWs0LI5XC40OrRh5L60z0QjFsqGWcHcbkBvpe1WYpcIS9yQ8sOi/vIPt1ejQCrMVrg==", + "dev": true + }, + "node_modules/postcss-colormin": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-4.0.3.tgz", "integrity": "sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "color": "^3.0.0", "has": "^1.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-convert-values": { + "node_modules/postcss-convert-values": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-4.0.1.tgz", "integrity": "sha512-Kisdo1y77KUC0Jmn0OXU/COOJbzM8cImvw1ZFsBgBgMgb1iL23Zs/LXRe3r+EZqM3vGYKdQ2YJVQ5VkJI+zEJQ==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-discard-comments": { + "node_modules/postcss-discard-comments": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-4.0.2.tgz", "integrity": "sha512-RJutN259iuRf3IW7GZyLM5Sw4GLTOH8FmsXBnv8Ab/Tc2k4SR4qbV4DNbyyY4+Sjo362SyDmW2DQ7lBSChrpkg==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-discard-duplicates": { + "node_modules/postcss-discard-duplicates": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-4.0.2.tgz", "integrity": "sha512-ZNQfR1gPNAiXZhgENFfEglF93pciw0WxMkJeVmw8eF+JZBbMD7jp6C67GqJAXVZP2BWbOztKfbsdmMp/k8c6oQ==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-discard-empty": { + "node_modules/postcss-discard-empty": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-4.0.1.tgz", "integrity": "sha512-B9miTzbznhDjTfjvipfHoqbWKwd0Mj+/fL5s1QOz06wufguil+Xheo4XpOnc4NqKYBCNqqEzgPv2aPBIJLox0w==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-discard-overridden": { + "node_modules/postcss-discard-overridden": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-4.0.1.tgz", "integrity": "sha512-IYY2bEDD7g1XM1IDEsUT4//iEYCxAmP5oDSFMVU/JVvT7gh+l4fmjciLqGgwjdWpQIdb0Che2VX00QObS5+cTg==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-merge-longhand": { + "node_modules/postcss-merge-longhand": { "version": "4.0.11", "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-4.0.11.tgz", "integrity": "sha512-alx/zmoeXvJjp7L4mxEMjh8lxVlDFX1gqWHzaaQewwMZiVhLo42TEClKaeHbRf6J7j82ZOdTJ808RtN0ZOZwvw==", "dev": true, - "requires": { + "dependencies": { "css-color-names": "0.0.4", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0", "stylehacks": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-merge-rules": { + "node_modules/postcss-merge-rules": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-4.0.3.tgz", "integrity": "sha512-U7e3r1SbvYzO0Jr3UT/zKBVgYYyhAz0aitvGIYOYK5CPmkNih+WDSsS5tvPrJ8YMQYlEMvsZIiqmn7HdFUaeEQ==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "caniuse-api": "^3.0.0", "cssnano-util-same-parent": "^4.0.0", @@ -5396,407 +6568,500 @@ "postcss-selector-parser": "^3.0.0", "vendors": "^1.0.0" }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-merge-rules/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" } }, - "postcss-minify-font-values": { + "node_modules/postcss-minify-font-values": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-4.0.2.tgz", "integrity": "sha512-j85oO6OnRU9zPf04+PZv1LYIYOprWm6IA6zkXkrJXyRveDEuQggG6tvoy8ir8ZwjLxLuGfNkCZEQG7zan+Hbtg==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-minify-gradients": { + "node_modules/postcss-minify-gradients": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-4.0.2.tgz", "integrity": "sha512-qKPfwlONdcf/AndP1U8SJ/uzIJtowHlMaSioKzebAXSG4iJthlWC9iSWznQcX4f66gIWX44RSA841HTHj3wK+Q==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-arguments": "^4.0.0", "is-color-stop": "^1.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-minify-params": { + "node_modules/postcss-minify-params": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-4.0.2.tgz", "integrity": "sha512-G7eWyzEx0xL4/wiBBJxJOz48zAKV2WG3iZOqVhPet/9geefm/Px5uo1fzlHu+DOjT+m0Mmiz3jkQzVHe6wxAWg==", "dev": true, - "requires": { + "dependencies": { "alphanum-sort": "^1.0.0", "browserslist": "^4.0.0", "cssnano-util-get-arguments": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0", "uniqs": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-minify-selectors": { + "node_modules/postcss-minify-selectors": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-4.0.2.tgz", "integrity": "sha512-D5S1iViljXBj9kflQo4YutWnJmwm8VvIsU1GeXJGiG9j8CIg9zs4voPMdQDUmIxetUOh60VilsNzCiAFTOqu3g==", "dev": true, - "requires": { + "dependencies": { "alphanum-sort": "^1.0.0", "has": "^1.0.0", "postcss": "^7.0.0", "postcss-selector-parser": "^3.0.0" }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/postcss-minify-selectors/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" } }, - "postcss-modules-extract-imports": { + "node_modules/postcss-modules-extract-imports": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-1.1.0.tgz", "integrity": "sha1-thTJcgvmgW6u41+zpfqh26agXds=", "dev": true, - "requires": { + "dependencies": { "postcss": "^6.0.1" - }, + } + }, + "node_modules/postcss-modules-extract-imports/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, "dependencies": { - "postcss": { - "version": "6.0.23", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", - "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", - "dev": true, - "requires": { - "chalk": "^2.4.1", - "source-map": "^0.6.1", - "supports-color": "^5.4.0" - } - } + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" } }, - "postcss-modules-local-by-default": { + "node_modules/postcss-modules-local-by-default": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-1.2.0.tgz", "integrity": "sha1-99gMOYxaOT+nlkRmvRlQCn1hwGk=", "dev": true, - "requires": { + "dependencies": { "css-selector-tokenizer": "^0.7.0", "postcss": "^6.0.1" - }, + } + }, + "node_modules/postcss-modules-local-by-default/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, "dependencies": { - "postcss": { - "version": "6.0.23", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", - "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", - "dev": true, - "requires": { - "chalk": "^2.4.1", - "source-map": "^0.6.1", - "supports-color": "^5.4.0" - } - } + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" } }, - "postcss-modules-scope": { + "node_modules/postcss-modules-scope": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-1.1.0.tgz", "integrity": "sha1-1upkmUx5+XtipytCb75gVqGUu5A=", "dev": true, - "requires": { + "dependencies": { "css-selector-tokenizer": "^0.7.0", "postcss": "^6.0.1" - }, + } + }, + "node_modules/postcss-modules-scope/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, "dependencies": { - "postcss": { - "version": "6.0.23", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", - "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", - "dev": true, - "requires": { - "chalk": "^2.4.1", - "source-map": "^0.6.1", - "supports-color": "^5.4.0" - } - } + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" } }, - "postcss-modules-values": { + "node_modules/postcss-modules-values": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-1.3.0.tgz", "integrity": "sha1-7P+p1+GSUYOJ9CrQ6D9yrsRW6iA=", "dev": true, - "requires": { + "dependencies": { "icss-replace-symbols": "^1.1.0", "postcss": "^6.0.1" - }, + } + }, + "node_modules/postcss-modules-values/node_modules/postcss": { + "version": "6.0.23", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", + "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", + "dev": true, "dependencies": { - "postcss": { - "version": "6.0.23", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-6.0.23.tgz", - "integrity": "sha512-soOk1h6J3VMTZtVeVpv15/Hpdl2cBLX3CAw4TAbkpTJiNPk9YP/zWcD1ND+xEtvyuuvKzbxliTOIyvkSeSJ6ag==", - "dev": true, - "requires": { - "chalk": "^2.4.1", - "source-map": "^0.6.1", - "supports-color": "^5.4.0" - } - } + "chalk": "^2.4.1", + "source-map": "^0.6.1", + "supports-color": "^5.4.0" + }, + "engines": { + "node": ">=4.0.0" } }, - "postcss-normalize-charset": { + "node_modules/postcss-normalize-charset": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-4.0.1.tgz", "integrity": "sha512-gMXCrrlWh6G27U0hF3vNvR3w8I1s2wOBILvA87iNXaPvSNo5uZAMYsZG7XjCUf1eVxuPfyL4TJ7++SGZLc9A3g==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-display-values": { + "node_modules/postcss-normalize-display-values": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-4.0.2.tgz", "integrity": "sha512-3F2jcsaMW7+VtRMAqf/3m4cPFhPD3EFRgNs18u+k3lTJJlVe7d0YPO+bnwqo2xg8YiRpDXJI2u8A0wqJxMsQuQ==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-match": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-positions": { + "node_modules/postcss-normalize-positions": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-4.0.2.tgz", "integrity": "sha512-Dlf3/9AxpxE+NF1fJxYDeggi5WwV35MXGFnnoccP/9qDtFrTArZ0D0R+iKcg5WsUd8nUYMIl8yXDCtcrT8JrdA==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-arguments": "^4.0.0", "has": "^1.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-repeat-style": { + "node_modules/postcss-normalize-repeat-style": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-4.0.2.tgz", "integrity": "sha512-qvigdYYMpSuoFs3Is/f5nHdRLJN/ITA7huIoCyqqENJe9PvPmLhNLMu7QTjPdtnVf6OcYYO5SHonx4+fbJE1+Q==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-arguments": "^4.0.0", "cssnano-util-get-match": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-string": { + "node_modules/postcss-normalize-string": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-4.0.2.tgz", "integrity": "sha512-RrERod97Dnwqq49WNz8qo66ps0swYZDSb6rM57kN2J+aoyEAJfZ6bMx0sx/F9TIEX0xthPGCmeyiam/jXif0eA==", "dev": true, - "requires": { + "dependencies": { "has": "^1.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-timing-functions": { + "node_modules/postcss-normalize-timing-functions": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-4.0.2.tgz", "integrity": "sha512-acwJY95edP762e++00Ehq9L4sZCEcOPyaHwoaFOhIwWCDfik6YvqsYNxckee65JHLKzuNSSmAdxwD2Cud1Z54A==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-match": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-unicode": { + "node_modules/postcss-normalize-unicode": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-4.0.1.tgz", "integrity": "sha512-od18Uq2wCYn+vZ/qCOeutvHjB5jm57ToxRaMeNuf0nWVHaP9Hua56QyMF6fs/4FSUnVIw0CBPsU0K4LnBPwYwg==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-url": { + "node_modules/postcss-normalize-url": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-4.0.1.tgz", "integrity": "sha512-p5oVaF4+IHwu7VpMan/SSpmpYxcJMtkGppYf0VbdH5B6hN8YNmVyJLuY9FmLQTzY3fag5ESUUHDqM+heid0UVA==", "dev": true, - "requires": { + "dependencies": { "is-absolute-url": "^2.0.0", "normalize-url": "^3.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-normalize-whitespace": { + "node_modules/postcss-normalize-whitespace": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-4.0.2.tgz", "integrity": "sha512-tO8QIgrsI3p95r8fyqKV+ufKlSHh9hMJqACqbv2XknufqEDhDvbguXGBBqxw9nsQoXWf0qOqppziKJKHMD4GtA==", "dev": true, - "requires": { + "dependencies": { "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-ordered-values": { + "node_modules/postcss-ordered-values": { "version": "4.1.2", "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-4.1.2.tgz", "integrity": "sha512-2fCObh5UanxvSxeXrtLtlwVThBvHn6MQcu4ksNT2tsaV2Fg76R2CV98W7wNSlX+5/pFwEyaDwKLLoEV7uRybAw==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-arguments": "^4.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-reduce-initial": { + "node_modules/postcss-reduce-initial": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-4.0.3.tgz", "integrity": "sha512-gKWmR5aUulSjbzOfD9AlJiHCGH6AEVLaM0AV+aSioxUDd16qXP1PCh8d1/BGVvpdWn8k/HiK7n6TjeoXN1F7DA==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "caniuse-api": "^3.0.0", "has": "^1.0.0", "postcss": "^7.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-reduce-transforms": { + "node_modules/postcss-reduce-transforms": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-4.0.2.tgz", "integrity": "sha512-EEVig1Q2QJ4ELpJXMZR8Vt5DQx8/mo+dGWSR7vWXqcob2gQLyQGsionYcGKATXvQzMPn6DSN1vTN7yFximdIAg==", "dev": true, - "requires": { + "dependencies": { "cssnano-util-get-match": "^4.0.0", "has": "^1.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-selector-parser": { + "node_modules/postcss-selector-parser": { "version": "6.0.2", "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.2.tgz", "integrity": "sha512-36P2QR59jDTOAiIkqEprfJDsoNrvwFei3eCqKd1Y0tUsBimsq39BLp7RD+JWny3WgB1zGhJX8XVePwm9k4wdBg==", "dev": true, - "requires": { + "dependencies": { "cssesc": "^3.0.0", "indexes-of": "^1.0.1", "uniq": "^1.0.1" + }, + "engines": { + "node": ">=4" } }, - "postcss-svgo": { + "node_modules/postcss-svgo": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-4.0.2.tgz", "integrity": "sha512-C6wyjo3VwFm0QgBy+Fu7gCYOkCmgmClghO+pjcxvrcBKtiKt0uCF+hvbMO1fyv5BMImRK90SMb+dwUnfbGd+jw==", "dev": true, - "requires": { + "dependencies": { "is-svg": "^3.0.0", "postcss": "^7.0.0", "postcss-value-parser": "^3.0.0", "svgo": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-unique-selectors": { + "node_modules/postcss-unique-selectors": { "version": "4.0.1", "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-4.0.1.tgz", "integrity": "sha512-+JanVaryLo9QwZjKrmJgkI4Fn8SBgRO6WXQBJi7KiAVPlmxikB5Jzc4EvXMT2H0/m0RjrVVm9rGNhZddm/8Spg==", "dev": true, - "requires": { + "dependencies": { "alphanum-sort": "^1.0.0", "postcss": "^7.0.0", "uniqs": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" } }, - "postcss-value-parser": { + "node_modules/postcss-value-parser": { "version": "3.3.1", "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-3.3.1.tgz", "integrity": "sha512-pISE66AbVkp4fDQ7VHBwRNXzAAKJjw4Vw7nWI/+Q3vuly7SNfgYXvm6i5IgFylHGK5sP/xHAbB7N49OS4gWNyQ==", "dev": true }, - "posthtml": { + "node_modules/postcss/node_modules/supports-color": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-6.1.0.tgz", + "integrity": "sha512-qe1jfm1Mg7Nq/NSh6XE24gPXROEVsWHxC1LIx//XNlD9iw7YZQGjZNjYN7xGaEG6iKdA8EtNFW6R0gjnVXp+wQ==", + "dev": true, + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/posthtml": { "version": "0.11.6", "resolved": "https://registry.npmjs.org/posthtml/-/posthtml-0.11.6.tgz", "integrity": "sha512-C2hrAPzmRdpuL3iH0TDdQ6XCc9M7Dcc3zEW5BLerY65G4tWWszwv6nG/ksi6ul5i2mx22ubdljgktXCtNkydkw==", "dev": true, - "requires": { + "dependencies": { "posthtml-parser": "^0.4.1", "posthtml-render": "^1.1.5" + }, + "engines": { + "node": ">=6.0.0" } }, - "posthtml-parser": { + "node_modules/posthtml-parser": { "version": "0.4.2", "resolved": "https://registry.npmjs.org/posthtml-parser/-/posthtml-parser-0.4.2.tgz", "integrity": "sha512-BUIorsYJTvS9UhXxPTzupIztOMVNPa/HtAm9KHni9z6qEfiJ1bpOBL5DfUOL9XAc3XkLIEzBzpph+Zbm4AdRAg==", "dev": true, - "requires": { + "dependencies": { "htmlparser2": "^3.9.2" } }, - "posthtml-render": { + "node_modules/posthtml-render": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/posthtml-render/-/posthtml-render-1.2.0.tgz", "integrity": "sha512-dQB+hoAKDtnI94RZm/wxBUH9My8OJcXd0uhWmGh2c7tVtQ85A+OS3yCN3LNbFtPz3bViwBJXAeoi+CBGMXM0DA==", - "dev": true + "dev": true, + "engines": { + "node": ">=6" + } }, - "prelude-ls": { + "node_modules/prelude-ls": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.1.2.tgz", "integrity": "sha1-IZMqVJ9eUv/ZqCf1cOBL5iqX2lQ=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.8.0" + } }, - "private": { + "node_modules/private": { "version": "0.1.8", "resolved": "https://registry.npmjs.org/private/-/private-0.1.8.tgz", "integrity": "sha512-VvivMrbvd2nKkiG38qjULzlc+4Vx4wm/whI9pQD35YrARNnhxeiRktSOhSukRLFNlzg6Br/cJPet5J/u19r/mg==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "process": { + "node_modules/process": { "version": "0.11.10", "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6.0" + } }, - "process-nextick-args": { + "node_modules/process-nextick-args": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/process-nextick-args/-/process-nextick-args-2.0.1.tgz", "integrity": "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==", "dev": true }, - "psl": { + "node_modules/psl": { "version": "1.8.0", "resolved": "https://registry.npmjs.org/psl/-/psl-1.8.0.tgz", "integrity": "sha512-RIdOzyoavK+hA18OGGWDqUTsCLhtA7IcZ/6NCs4fFJaHBDab+pDDmDIByWFRQJq2Cd7r1OoQxBGKOaztq+hjIQ==", "dev": true }, - "public-encrypt": { + "node_modules/public-encrypt": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/public-encrypt/-/public-encrypt-4.0.3.tgz", "integrity": "sha512-zVpa8oKZSz5bTMTFClc1fQOnyyEzpl5ozpi1B5YcvBrdohMjH2rfsBtyXcuNuwjsDIXmBYlF2N5FlJYhR29t8Q==", "dev": true, - "requires": { + "dependencies": { "bn.js": "^4.1.0", "browserify-rsa": "^4.0.0", "create-hash": "^1.1.0", @@ -5805,90 +7070,123 @@ "safe-buffer": "^5.1.2" } }, - "punycode": { + "node_modules/punycode": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/punycode/-/punycode-2.1.1.tgz", "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", - "dev": true + "dev": true, + "engines": { + "node": ">=6" + } }, - "purgecss": { + "node_modules/purgecss": { "version": "1.4.2", "resolved": "https://registry.npmjs.org/purgecss/-/purgecss-1.4.2.tgz", "integrity": "sha512-hkOreFTgiyMHMmC2BxzdIw5DuC6kxAbP/gGOGd3MEsF3+5m69rIvUEPaxrnoUtfODTFKe9hcXjGwC6jcjoyhOw==", "dev": true, - "requires": { + "dependencies": { "glob": "^7.1.3", "postcss": "^7.0.14", "postcss-selector-parser": "^6.0.0", "yargs": "^14.0.0" + }, + "bin": { + "purgecss": "bin/purgecss" + }, + "engines": { + "node": ">=4.4.0", + "npm": ">=5.2.0" + }, + "funding": { + "url": "https://github.com/sponsors/Ffloriel" } }, - "q": { + "node_modules/q": { "version": "1.5.1", "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.6.0", + "teleport": ">=0.2.0" + } }, - "qs": { + "node_modules/qs": { "version": "6.5.2", "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz", "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.6" + } }, - "querystring": { + "node_modules/querystring": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", - "dev": true + "deprecated": "The querystring API is considered Legacy. new code should use the URLSearchParams API instead.", + "dev": true, + "engines": { + "node": ">=0.4.x" + } }, - "querystring-es3": { + "node_modules/querystring-es3": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/querystring-es3/-/querystring-es3-0.2.1.tgz", "integrity": "sha1-nsYfeQSYdXB9aUFFlv2Qek1xHnM=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4.x" + } }, - "quote-stream": { + "node_modules/quote-stream": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/quote-stream/-/quote-stream-1.0.2.tgz", "integrity": "sha1-hJY/jJwmuULhU/7rU6rnRlK34LI=", "dev": true, - "requires": { + "dependencies": { "buffer-equal": "0.0.1", "minimist": "^1.1.3", "through2": "^2.0.0" + }, + "bin": { + "quote-stream": "bin/cmd.js" } }, - "randombytes": { + "node_modules/randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", "dev": true, - "requires": { + "dependencies": { "safe-buffer": "^5.1.0" } }, - "randomfill": { + "node_modules/randomfill": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/randomfill/-/randomfill-1.0.4.tgz", "integrity": "sha512-87lcbR8+MhcWcUiQ+9e+Rwx8MyR2P7qnt15ynUlbm3TU/fjbgz4GsvfSUDTemtCCtVCqb4ZcEFlyPNTh9bBTLw==", "dev": true, - "requires": { + "dependencies": { "randombytes": "^2.0.5", "safe-buffer": "^5.1.0" } }, - "range-parser": { + "node_modules/range-parser": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "readable-stream": { + "node_modules/readable-stream": { "version": "2.3.7", "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-2.3.7.tgz", "integrity": "sha512-Ebho8K4jIbHAxnuxi7o42OrZgF/ZTNcsZj6nRKyUmkhLFq8CHItp/fy6hQZuZmP/n3yZ9VBUbp4zz/mX8hmYPw==", "dev": true, - "requires": { + "dependencies": { "core-util-is": "~1.0.0", "inherits": "~2.0.3", "isarray": "~1.0.0", @@ -5898,119 +7196,142 @@ "util-deprecate": "~1.0.1" } }, - "readdirp": { + "node_modules/readdirp": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-2.2.1.tgz", "integrity": "sha512-1JU/8q+VgFZyxwrJ+SVIOsh+KywWGpds3NTqikiKpDMZWScmAYyKIgqkO+ARvNWJfXeXR1zxz7aHF4u4CyH6vQ==", "dev": true, - "requires": { + "dependencies": { "graceful-fs": "^4.1.11", "micromatch": "^3.1.10", "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" } }, - "regenerate": { + "node_modules/regenerate": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.0.tgz", "integrity": "sha512-1G6jJVDWrt0rK99kBjvEtziZNCICAuvIPkSiUFIQxVP06RCVpq3dmDo2oi6ABpYaDYaTRr67BEhL8r1wgEZZKg==", "dev": true }, - "regenerate-unicode-properties": { + "node_modules/regenerate-unicode-properties": { "version": "8.2.0", "resolved": "https://registry.npmjs.org/regenerate-unicode-properties/-/regenerate-unicode-properties-8.2.0.tgz", "integrity": "sha512-F9DjY1vKLo/tPePDycuH3dn9H1OTPIkVD9Kz4LODu+F2C75mgjAJ7x/gwy6ZcSNRAAkhNlJSOHRe8k3p+K9WhA==", "dev": true, - "requires": { + "dependencies": { "regenerate": "^1.4.0" + }, + "engines": { + "node": ">=4" } }, - "regenerator-runtime": { + "node_modules/regenerator-runtime": { "version": "0.13.5", "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.5.tgz", "integrity": "sha512-ZS5w8CpKFinUzOwW3c83oPeVXoNsrLsaCoLtJvAClH135j/R77RuymhiSErhm2lKcwSCIpmvIWSbDkIfAqKQlA==", "dev": true }, - "regenerator-transform": { + "node_modules/regenerator-transform": { "version": "0.14.4", "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.4.tgz", "integrity": "sha512-EaJaKPBI9GvKpvUz2mz4fhx7WPgvwRLY9v3hlNHWmAuJHI13T4nwKnNvm5RWJzEdnI5g5UwtOww+S8IdoUC2bw==", "dev": true, - "requires": { + "dependencies": { "@babel/runtime": "^7.8.4", "private": "^0.1.8" } }, - "regex-not": { + "node_modules/regex-not": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/regex-not/-/regex-not-1.0.2.tgz", "integrity": "sha512-J6SDjUgDxQj5NusnOtdFxDwN/+HWykR8GELwctJ7mdqhcyy1xEc4SRFHUXvxTp661YaVKAjfRLZ9cCqS6tn32A==", "dev": true, - "requires": { + "dependencies": { "extend-shallow": "^3.0.2", "safe-regex": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "regexpu-core": { + "node_modules/regexpu-core": { "version": "4.7.0", "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.7.0.tgz", "integrity": "sha512-TQ4KXRnIn6tz6tjnrXEkD/sshygKH/j5KzK86X8MkeHyZ8qst/LZ89j3X4/8HEIfHANTFIP/AbXakeRhWIl5YQ==", "dev": true, - "requires": { + "dependencies": { "regenerate": "^1.4.0", "regenerate-unicode-properties": "^8.2.0", "regjsgen": "^0.5.1", "regjsparser": "^0.6.4", "unicode-match-property-ecmascript": "^1.0.4", "unicode-match-property-value-ecmascript": "^1.2.0" + }, + "engines": { + "node": ">=4" } }, - "regjsgen": { + "node_modules/regjsgen": { "version": "0.5.1", "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.5.1.tgz", "integrity": "sha512-5qxzGZjDs9w4tzT3TPhCJqWdCc3RLYwy9J2NB0nm5Lz+S273lvWcpjaTGHsT1dc6Hhfq41uSEOw8wBmxrKOuyg==", "dev": true }, - "regjsparser": { + "node_modules/regjsparser": { "version": "0.6.4", "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.4.tgz", "integrity": "sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==", "dev": true, - "requires": { + "dependencies": { "jsesc": "~0.5.0" }, - "dependencies": { - "jsesc": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", - "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", - "dev": true - } + "bin": { + "regjsparser": "bin/parser" + } + }, + "node_modules/regjsparser/node_modules/jsesc": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-0.5.0.tgz", + "integrity": "sha1-597mbjXW/Bb3EP6R1c9p9w8IkR0=", + "dev": true, + "bin": { + "jsesc": "bin/jsesc" } }, - "remove-trailing-separator": { + "node_modules/remove-trailing-separator": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz", "integrity": "sha1-wkvOKig62tW8P1jg1IJJuSN52O8=", "dev": true }, - "repeat-element": { + "node_modules/repeat-element": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "repeat-string": { + "node_modules/repeat-string": { "version": "1.6.1", "resolved": "https://registry.npmjs.org/repeat-string/-/repeat-string-1.6.1.tgz", "integrity": "sha1-jcrkcOHIirwtYA//Sndihtp15jc=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10" + } }, - "request": { + "node_modules/request": { "version": "2.88.2", "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "deprecated": "request has been deprecated, see https://github.com/request/request/issues/3142", "dev": true, - "requires": { + "dependencies": { "aws-sign2": "~0.7.0", "aws4": "^1.8.0", "caseless": "~0.12.0", @@ -6031,156 +7352,197 @@ "tough-cookie": "~2.5.0", "tunnel-agent": "^0.6.0", "uuid": "^3.3.2" + }, + "engines": { + "node": ">= 6" } }, - "request-promise-core": { + "node_modules/request-promise-core": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/request-promise-core/-/request-promise-core-1.1.3.tgz", "integrity": "sha512-QIs2+ArIGQVp5ZYbWD5ZLCY29D5CfWizP8eWnm8FoGD1TX61veauETVQbrV60662V0oFBkrDOuaBI8XgtuyYAQ==", "dev": true, - "requires": { + "dependencies": { "lodash": "^4.17.15" + }, + "engines": { + "node": ">=0.10.0" + }, + "peerDependencies": { + "request": "^2.34" } }, - "request-promise-native": { + "node_modules/request-promise-native": { "version": "1.0.8", "resolved": "https://registry.npmjs.org/request-promise-native/-/request-promise-native-1.0.8.tgz", "integrity": "sha512-dapwLGqkHtwL5AEbfenuzjTYg35Jd6KPytsC2/TLkVMz8rm+tNt72MGUWT1RP/aYawMpN6HqbNGBQaRcBtjQMQ==", + "deprecated": "request-promise-native has been deprecated because it extends the now deprecated request package, see https://github.com/request/request/issues/3142", "dev": true, - "requires": { + "dependencies": { "request-promise-core": "1.1.3", "stealthy-require": "^1.1.1", "tough-cookie": "^2.3.3" + }, + "engines": { + "node": ">=0.12.0" + }, + "peerDependencies": { + "request": "^2.34" } }, - "require-directory": { + "node_modules/require-directory": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/require-directory/-/require-directory-2.1.1.tgz", "integrity": "sha1-jGStX9MNqxyXbiNE/+f3kqam30I=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "require-main-filename": { + "node_modules/require-main-filename": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/require-main-filename/-/require-main-filename-2.0.0.tgz", "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", "dev": true }, - "resolve": { + "node_modules/resolve": { "version": "1.15.1", "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.15.1.tgz", "integrity": "sha512-84oo6ZTtoTUpjgNEr5SJyzQhzL72gaRodsSfyxC/AXRvwu0Yse9H8eF9IpGo7b8YetZhlI6v7ZQ6bKBFV/6S7w==", "dev": true, - "requires": { + "dependencies": { "path-parse": "^1.0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "resolve-from": { + "node_modules/resolve-from": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "resolve-url": { + "node_modules/resolve-url": { "version": "0.2.1", "resolved": "https://registry.npmjs.org/resolve-url/-/resolve-url-0.2.1.tgz", "integrity": "sha1-LGN/53yJOv0qZj/iGqkIAGjiBSo=", + "deprecated": "https://github.com/lydell/resolve-url#deprecated", "dev": true }, - "restore-cursor": { + "node_modules/restore-cursor": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-2.0.0.tgz", "integrity": "sha1-n37ih/gv0ybU/RYpI9YhKe7g368=", "dev": true, - "requires": { + "dependencies": { "onetime": "^2.0.0", "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=4" } }, - "ret": { + "node_modules/ret": { "version": "0.1.15", "resolved": "https://registry.npmjs.org/ret/-/ret-0.1.15.tgz", "integrity": "sha512-TTlYpa+OL+vMMNG24xSlQGEJ3B/RzEfUlLct7b5G/ytav+wPrplCpVMFuwzXbkecJrb6IYo1iFb0S9v37754mg==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.12" + } }, - "rgb-regex": { + "node_modules/rgb-regex": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/rgb-regex/-/rgb-regex-1.0.1.tgz", "integrity": "sha1-wODWiC3w4jviVKR16O3UGRX+rrE=", "dev": true }, - "rgba-regex": { + "node_modules/rgba-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/rgba-regex/-/rgba-regex-1.0.0.tgz", "integrity": "sha1-QzdOLiyglosO8VI0YLfXMP8i7rM=", "dev": true }, - "rimraf": { + "node_modules/rimraf": { "version": "2.7.1", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", "dev": true, - "requires": { + "dependencies": { "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" } }, - "ripemd160": { + "node_modules/ripemd160": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", "dev": true, - "requires": { + "dependencies": { "hash-base": "^3.0.0", "inherits": "^2.0.1" } }, - "safe-buffer": { + "node_modules/safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", "dev": true }, - "safe-regex": { + "node_modules/safe-regex": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", "dev": true, - "requires": { + "dependencies": { "ret": "~0.1.10" } }, - "safer-buffer": { + "node_modules/safer-buffer": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==", "dev": true }, - "sax": { + "node_modules/sax": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/sax/-/sax-1.2.4.tgz", "integrity": "sha512-NqVDv9TpANUjFm0N8uM5GxL36UgKi9/atZw+x7YFnQ8ckwFGKrl4xX4yWtrey3UJm5nP1kUbnYgLopqWNSRhWw==", "dev": true }, - "saxes": { + "node_modules/saxes": { "version": "3.1.11", "resolved": "https://registry.npmjs.org/saxes/-/saxes-3.1.11.tgz", "integrity": "sha512-Ydydq3zC+WYDJK1+gRxRapLIED9PWeSuuS41wqyoRmzvhhh9nc+QQrVMKJYzJFULazeGhzSV0QleN2wD3boh2g==", "dev": true, - "requires": { + "dependencies": { "xmlchars": "^2.1.1" + }, + "engines": { + "node": ">=8" } }, - "semver": { + "node_modules/semver": { "version": "5.7.1", "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true + "dev": true, + "bin": { + "semver": "bin/semver" + } }, - "send": { + "node_modules/send": { "version": "0.17.1", "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", "dev": true, - "requires": { + "dependencies": { "debug": "2.6.9", "depd": "~1.1.2", "destroy": "~1.0.4", @@ -6195,151 +7557,167 @@ "range-parser": "~1.2.1", "statuses": "~1.5.0" }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - }, - "dependencies": { - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - }, - "ms": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", - "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", - "dev": true - } + "ms": "2.0.0" } }, - "serialize-to-js": { + "node_modules/send/node_modules/debug/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "node_modules/send/node_modules/ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==", + "dev": true + }, + "node_modules/serialize-to-js": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/serialize-to-js/-/serialize-to-js-3.1.1.tgz", "integrity": "sha512-F+NGU0UHMBO4Q965tjw7rvieNVjlH6Lqi2emq/Lc9LUURYJbiCzmpi4Cy1OOjjVPtxu0c+NE85LU6968Wko5ZA==", - "dev": true + "dev": true, + "engines": { + "node": ">=4.0.0" + } }, - "serve-static": { + "node_modules/serve-static": { "version": "1.14.1", "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", "dev": true, - "requires": { + "dependencies": { "encodeurl": "~1.0.2", "escape-html": "~1.0.3", "parseurl": "~1.3.3", "send": "0.17.1" + }, + "engines": { + "node": ">= 0.8.0" } }, - "set-blocking": { + "node_modules/set-blocking": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", "dev": true }, - "set-value": { + "node_modules/set-value": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/set-value/-/set-value-2.0.1.tgz", "integrity": "sha512-JxHc1weCN68wRY0fhCoXpyK55m/XPHafOmK4UWD7m2CI14GMcFypt4w/0+NV5f/ZMby2F6S2wwA7fgynh9gWSw==", "dev": true, - "requires": { + "dependencies": { "extend-shallow": "^2.0.1", "is-extendable": "^0.1.1", "is-plain-object": "^2.0.3", "split-string": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/set-value/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, "dependencies": { - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - } + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "setimmediate": { + "node_modules/setimmediate": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", "dev": true }, - "setprototypeof": { + "node_modules/setprototypeof": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==", "dev": true }, - "sha.js": { + "node_modules/sha.js": { "version": "2.4.11", "resolved": "https://registry.npmjs.org/sha.js/-/sha.js-2.4.11.tgz", "integrity": "sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==", "dev": true, - "requires": { + "dependencies": { "inherits": "^2.0.1", "safe-buffer": "^5.0.1" + }, + "bin": { + "sha.js": "bin.js" } }, - "shallow-copy": { + "node_modules/shallow-copy": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/shallow-copy/-/shallow-copy-0.0.1.tgz", "integrity": "sha1-QV9CcC1z2BAzApLMXuhurhoRoXA=", "dev": true }, - "shebang-command": { + "node_modules/shebang-command": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", "integrity": "sha1-RKrGW2lbAzmJaMOfNj/uXer98eo=", "dev": true, - "requires": { + "dependencies": { "shebang-regex": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "shebang-regex": { + "node_modules/shebang-regex": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/shebang-regex/-/shebang-regex-1.0.0.tgz", "integrity": "sha1-2kL0l0DAtC2yypcoVxyxkMmO/qM=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "signal-exit": { + "node_modules/signal-exit": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.3.tgz", "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", "dev": true }, - "simple-swizzle": { + "node_modules/simple-swizzle": { "version": "0.2.2", "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz", "integrity": "sha1-pNprY1/8zMoz9w0Xy5JZLeleVXo=", "dev": true, - "requires": { - "is-arrayish": "^0.3.1" - }, "dependencies": { - "is-arrayish": { - "version": "0.3.2", - "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", - "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", - "dev": true - } + "is-arrayish": "^0.3.1" } }, - "snapdragon": { + "node_modules/simple-swizzle/node_modules/is-arrayish": { + "version": "0.3.2", + "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz", + "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==", + "dev": true + }, + "node_modules/snapdragon": { "version": "0.8.2", "resolved": "https://registry.npmjs.org/snapdragon/-/snapdragon-0.8.2.tgz", "integrity": "sha512-FtyOnWN/wCHTVXOMwvSv26d+ko5vWlIDD6zoUJ7LW8vh+ZBC8QdljveRP+crNrtBwioEUWy/4dMtbBjA4ioNlg==", "dev": true, - "requires": { + "dependencies": { "base": "^0.11.1", "debug": "^2.2.0", "define-property": "^0.2.5", @@ -6349,131 +7727,164 @@ "source-map-resolve": "^0.5.0", "use": "^3.1.0" }, - "dependencies": { - "debug": { - "version": "2.6.9", - "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", - "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - }, - "extend-shallow": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", - "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", - "dev": true, - "requires": { - "is-extendable": "^0.1.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "source-map": { - "version": "0.5.7", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true - } + "engines": { + "node": ">=0.10.0" } }, - "snapdragon-node": { + "node_modules/snapdragon-node": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/snapdragon-node/-/snapdragon-node-2.1.1.tgz", "integrity": "sha512-O27l4xaMYt/RSQ5TR3vpWCAB5Kb/czIcqUFOM/C4fYcLnbZUc1PkjTAMjof2pBWaSTwOUd6qUHcFGVGj7aIwnw==", "dev": true, - "requires": { + "dependencies": { "define-property": "^1.0.0", "isobject": "^3.0.0", "snapdragon-util": "^3.0.1" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", + "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", + "dev": true, + "dependencies": { + "is-descriptor": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-accessor-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", + "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, "dependencies": { - "define-property": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-1.0.0.tgz", - "integrity": "sha1-dp66rz9KY6rTr56NMEybvnm/sOY=", - "dev": true, - "requires": { - "is-descriptor": "^1.0.0" - } - }, - "is-accessor-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-accessor-descriptor/-/is-accessor-descriptor-1.0.0.tgz", - "integrity": "sha512-m5hnHTkcVsPfqx3AKlyttIPb7J+XykHvJP2B9bZDjlhLIoEq4XoK64Vg7boZlVWYK6LUY94dYPEE7Lh0ZkZKcQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-data-descriptor": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", - "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", - "dev": true, - "requires": { - "kind-of": "^6.0.0" - } - }, - "is-descriptor": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", - "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", - "dev": true, - "requires": { - "is-accessor-descriptor": "^1.0.0", - "is-data-descriptor": "^1.0.0", - "kind-of": "^6.0.2" - } - } + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-data-descriptor": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-1.0.0.tgz", + "integrity": "sha512-jbRXy1FmtAoCjQkVmIVYwuuqDFUbaOeDjmed1tOGPrsMhtJA4rD9tkgA0F1qJ3gRFRXcHYVkdeaP50Q5rE/jLQ==", + "deprecated": "Please upgrade to v1.0.1", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-node/node_modules/is-descriptor": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-descriptor/-/is-descriptor-1.0.2.tgz", + "integrity": "sha512-2eis5WqQGV7peooDyLmNEPUrps9+SXX5c9pL3xEB+4e9HnGuDa7mB7kHxHw4CbqS9k1T2hOH3miL8n8WtiYVtg==", + "dev": true, + "dependencies": { + "is-accessor-descriptor": "^1.0.0", + "is-data-descriptor": "^1.0.0", + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=0.10.0" } }, - "snapdragon-util": { + "node_modules/snapdragon-util": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/snapdragon-util/-/snapdragon-util-3.0.1.tgz", "integrity": "sha512-mbKkMdQKsjX4BAL4bRYTj21edOf8cN7XHdYUJEe+Zn99hVEYcMvKPct1IqNe7+AZPirn8BCDOQBHQZknqmKlZQ==", "dev": true, - "requires": { + "dependencies": { "kind-of": "^3.2.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon-util/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "source-map": { + "node_modules/snapdragon/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/snapdragon/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, + "dependencies": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/node_modules/extend-shallow": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/extend-shallow/-/extend-shallow-2.0.1.tgz", + "integrity": "sha1-Ua99YUrZqfYQ6huvu5idaxxWiQ8=", + "dev": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "node_modules/snapdragon/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "source-map-resolve": { + "node_modules/source-map-resolve": { "version": "0.5.3", "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.5.3.tgz", "integrity": "sha512-Htz+RnsXWk5+P2slx5Jh3Q66vhQj1Cllm0zvnaY98+NFx+Dv2CF/f5O/t8x+KaNdrdIAsruNzoh/KpialbqAnw==", + "deprecated": "See https://github.com/lydell/source-map-resolve#deprecated", "dev": true, - "requires": { + "dependencies": { "atob": "^2.1.2", "decode-uri-component": "^0.2.0", "resolve-url": "^0.2.1", @@ -6481,43 +7892,47 @@ "urix": "^0.1.0" } }, - "source-map-support": { + "node_modules/source-map-support": { "version": "0.5.16", "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.16.tgz", "integrity": "sha512-efyLRJDr68D9hBBNIPWFjhpFzURh+KJykQwvMyW5UiZzYwoF6l4YMMDIJJEyFWxWCqfyxLzz6tSfUFR+kXXsVQ==", "dev": true, - "requires": { + "dependencies": { "buffer-from": "^1.0.0", "source-map": "^0.6.0" } }, - "source-map-url": { + "node_modules/source-map-url": { "version": "0.4.0", "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.0.tgz", "integrity": "sha1-PpNdfd1zYxuXZZlW1VEo6HtQhKM=", + "deprecated": "See https://github.com/lydell/source-map-url#deprecated", "dev": true }, - "split-string": { + "node_modules/split-string": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/split-string/-/split-string-3.1.0.tgz", "integrity": "sha512-NzNVhJDYpwceVVii8/Hu6DKfD2G+NrQHlS/V/qgv763EYudVwEcMQNxd2lh+0VrUByXN/oJkl5grOhYWvQUYiw==", "dev": true, - "requires": { + "dependencies": { "extend-shallow": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "sprintf-js": { + "node_modules/sprintf-js": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.0.3.tgz", "integrity": "sha1-BOaSb2YolTVPPdAVIDYzuFcpfiw=", "dev": true }, - "sshpk": { + "node_modules/sshpk": { "version": "1.16.1", "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", "dev": true, - "requires": { + "dependencies": { "asn1": "~0.2.3", "assert-plus": "^1.0.0", "bcrypt-pbkdf": "^1.0.0", @@ -6527,71 +7942,98 @@ "jsbn": "~0.1.0", "safer-buffer": "^2.0.2", "tweetnacl": "~0.14.0" + }, + "bin": { + "sshpk-conv": "bin/sshpk-conv", + "sshpk-sign": "bin/sshpk-sign", + "sshpk-verify": "bin/sshpk-verify" + }, + "engines": { + "node": ">=0.10.0" } }, - "stable": { + "node_modules/stable": { "version": "0.1.8", "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", + "deprecated": "Modern JS already guarantees Array#sort() is a stable sort, so this library is deprecated. See the compatibility table on MDN: https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#browser_compatibility", "dev": true }, - "static-eval": { + "node_modules/static-eval": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/static-eval/-/static-eval-2.0.5.tgz", "integrity": "sha512-nNbV6LbGtMBgv7e9LFkt5JV8RVlRsyJrphfAt9tOtBBW/SfnzZDf2KnS72an8e434A+9e/BmJuTxeGPvrAK7KA==", "dev": true, - "requires": { + "dependencies": { "escodegen": "^1.11.1" - }, + } + }, + "node_modules/static-eval/node_modules/escodegen": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.1.tgz", + "integrity": "sha512-Bmt7NcRySdIfNPfU2ZoXDrrXsG9ZjvDxcAlMfDUgRBjLOWTuIACXPBFJH7Z+cLb40JeQco5toikyc9t9P8E9SQ==", + "dev": true, "dependencies": { - "escodegen": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/escodegen/-/escodegen-1.14.1.tgz", - "integrity": "sha512-Bmt7NcRySdIfNPfU2ZoXDrrXsG9ZjvDxcAlMfDUgRBjLOWTuIACXPBFJH7Z+cLb40JeQco5toikyc9t9P8E9SQ==", - "dev": true, - "requires": { - "esprima": "^4.0.1", - "estraverse": "^4.2.0", - "esutils": "^2.0.2", - "optionator": "^0.8.1", - "source-map": "~0.6.1" - } - }, - "esprima": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", - "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", - "dev": true - } + "esprima": "^4.0.1", + "estraverse": "^4.2.0", + "esutils": "^2.0.2", + "optionator": "^0.8.1" + }, + "bin": { + "escodegen": "bin/escodegen.js", + "esgenerate": "bin/esgenerate.js" + }, + "engines": { + "node": ">=4.0" + }, + "optionalDependencies": { + "source-map": "~0.6.1" + } + }, + "node_modules/static-eval/node_modules/esprima": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.1.tgz", + "integrity": "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A==", + "dev": true, + "bin": { + "esparse": "bin/esparse.js", + "esvalidate": "bin/esvalidate.js" + }, + "engines": { + "node": ">=4" } }, - "static-extend": { + "node_modules/static-extend": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/static-extend/-/static-extend-0.1.2.tgz", "integrity": "sha1-YICcOcv/VTNyJv1eC1IPNB8ftcY=", "dev": true, - "requires": { + "dependencies": { "define-property": "^0.2.5", "object-copy": "^0.1.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/node_modules/define-property": { + "version": "0.2.5", + "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", + "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", + "dev": true, "dependencies": { - "define-property": { - "version": "0.2.5", - "resolved": "https://registry.npmjs.org/define-property/-/define-property-0.2.5.tgz", - "integrity": "sha1-w1se+RjsPJkPmlvFe+BKrOxcgRY=", - "dev": true, - "requires": { - "is-descriptor": "^0.1.0" - } - } + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "static-module": { + "node_modules/static-module": { "version": "2.2.5", "resolved": "https://registry.npmjs.org/static-module/-/static-module-2.2.5.tgz", "integrity": "sha512-D8vv82E/Kpmz3TXHKG8PPsCPg+RAX6cbCOyvjM6x04qZtQ47EtJFVwRsdov3n5d6/6ynrOY9XB4JkaZwB2xoRQ==", "dev": true, - "requires": { + "dependencies": { "concat-stream": "~1.6.0", "convert-source-map": "^1.5.1", "duplexer2": "~0.1.4", @@ -6608,34 +8050,40 @@ "through2": "~2.0.3" } }, - "statuses": { + "node_modules/statuses": { "version": "1.5.0", "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", - "dev": true + "dev": true, + "engines": { + "node": ">= 0.6" + } }, - "stealthy-require": { + "node_modules/stealthy-require": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/stealthy-require/-/stealthy-require-1.1.1.tgz", "integrity": "sha1-NbCYdbT/SfJqd35QmzCQoyJr8ks=", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "stream-browserify": { + "node_modules/stream-browserify": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/stream-browserify/-/stream-browserify-2.0.2.tgz", "integrity": "sha512-nX6hmklHs/gr2FuxYDltq8fJA1GDlxKQCz8O/IM4atRqBH8OORmBNgfvW5gG10GT/qQ9u0CzIvr2X5Pkt6ntqg==", "dev": true, - "requires": { + "dependencies": { "inherits": "~2.0.1", "readable-stream": "^2.0.2" } }, - "stream-http": { + "node_modules/stream-http": { "version": "2.8.3", "resolved": "https://registry.npmjs.org/stream-http/-/stream-http-2.8.3.tgz", "integrity": "sha512-+TSkfINHDo4J+ZobQLWiMouQYB+UVYFttRA94FpEzzJ7ZdqcL4uUUQ7WkdkI4DSozGmgBUE/a47L+38PenXhUw==", "dev": true, - "requires": { + "dependencies": { "builtin-status-codes": "^3.0.0", "inherits": "^2.0.1", "readable-stream": "^2.3.6", @@ -6643,133 +8091,169 @@ "xtend": "^4.0.0" } }, - "string-width": { + "node_modules/string_decoder": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", + "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", + "dev": true, + "dependencies": { + "safe-buffer": "~5.1.0" + } + }, + "node_modules/string-width": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/string-width/-/string-width-3.1.0.tgz", "integrity": "sha512-vafcv6KjVZKSgz06oM/H6GDBrAtz8vdhQakGjFIvNrHA6y3HCF1CInLy+QLq8dTJPQ1b+KDUqDFctkdRW44e1w==", "dev": true, - "requires": { + "dependencies": { "emoji-regex": "^7.0.1", "is-fullwidth-code-point": "^2.0.0", "strip-ansi": "^5.1.0" }, + "engines": { + "node": ">=6" + } + }, + "node_modules/string-width/node_modules/ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/string-width/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - } + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" } }, - "string.prototype.trimend": { + "node_modules/string.prototype.trimend": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.0.tgz", "integrity": "sha512-EEJnGqa/xNfIg05SxiPSqRS7S9qwDhYts1TSLR1BQfYUfPe1stofgGKvwERK9+9yf+PpfBMlpBaCHucXGPQfUA==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.5" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "string.prototype.trimleft": { + "node_modules/string.prototype.trimleft": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/string.prototype.trimleft/-/string.prototype.trimleft-2.1.2.tgz", "integrity": "sha512-gCA0tza1JBvqr3bfAIFJGqfdRTyPae82+KTnm3coDXkZN9wnuW3HjGgN386D7hfv5CHQYCI022/rJPVlqXyHSw==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.5", "string.prototype.trimstart": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "string.prototype.trimright": { + "node_modules/string.prototype.trimright": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/string.prototype.trimright/-/string.prototype.trimright-2.1.2.tgz", "integrity": "sha512-ZNRQ7sY3KroTaYjRS6EbNiiHrOkjihL9aQE/8gfQ4DtAC/aEBRHFJa44OmoWxGGqXuJlfKkZW4WcXErGr+9ZFg==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.5", "string.prototype.trimend": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "string.prototype.trimstart": { + "node_modules/string.prototype.trimstart": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.0.tgz", "integrity": "sha512-iCP8g01NFYiiBOnwG1Xc3WZLyoo+RuBymwIlWncShXDDJYWN6DbnM3odslBJdgCdRlq94B5s63NWAZlcn2CS4w==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.5" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "string_decoder": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.1.1.tgz", - "integrity": "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==", - "dev": true, - "requires": { - "safe-buffer": "~5.1.0" - } - }, - "strip-ansi": { + "node_modules/strip-ansi": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-4.0.0.tgz", "integrity": "sha1-qEeQIusaw2iocTibY1JixQXuNo8=", "dev": true, - "requires": { + "dependencies": { "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" } }, - "stylehacks": { + "node_modules/stylehacks": { "version": "4.0.3", "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-4.0.3.tgz", "integrity": "sha512-7GlLk9JwlElY4Y6a/rmbH2MhVlTyVmiJd1PfTCqFaIBEGMYNsrO/v3SeGTdhBThLg4Z+NbOk/qFMwCa+J+3p/g==", "dev": true, - "requires": { + "dependencies": { "browserslist": "^4.0.0", "postcss": "^7.0.0", "postcss-selector-parser": "^3.0.0" }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/stylehacks/node_modules/postcss-selector-parser": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", + "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", + "dev": true, "dependencies": { - "postcss-selector-parser": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-3.1.2.tgz", - "integrity": "sha512-h7fJ/5uWuRVyOtkO45pnt1Ih40CEleeyCHzipqAZO2e5H20g25Y48uYnFUiShvY4rZWNJ/Bib/KVPmanaCtOhA==", - "dev": true, - "requires": { - "dot-prop": "^5.2.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1" - } - } + "dot-prop": "^5.2.0", + "indexes-of": "^1.0.1", + "uniq": "^1.0.1" + }, + "engines": { + "node": ">=8" } }, - "supports-color": { + "node_modules/supports-color": { "version": "5.5.0", "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz", "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==", "dev": true, - "requires": { + "dependencies": { "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" } }, - "svgo": { + "node_modules/svgo": { "version": "1.3.2", "resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz", "integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==", + "deprecated": "This SVGO version is no longer supported. Upgrade to v2.x.x.", "dev": true, - "requires": { + "dependencies": { "chalk": "^2.4.1", "coa": "^2.0.2", "css-select": "^2.0.0", @@ -6783,183 +8267,230 @@ "stable": "^0.1.8", "unquote": "~1.1.1", "util.promisify": "~1.0.0" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=4.0.0" } }, - "symbol-tree": { + "node_modules/symbol-tree": { "version": "3.2.4", "resolved": "https://registry.npmjs.org/symbol-tree/-/symbol-tree-3.2.4.tgz", "integrity": "sha512-9QNk5KwDF+Bvz+PyObkmSYjI5ksVUYtjW7AU22r2NKcfLJcXp96hkDWU3+XndOsUb+AQ9QhfzfCT2O+CNWT5Tw==", "dev": true }, - "terser": { + "node_modules/terser": { "version": "3.17.0", "resolved": "https://registry.npmjs.org/terser/-/terser-3.17.0.tgz", "integrity": "sha512-/FQzzPJmCpjAH9Xvk2paiWrFq+5M6aVOf+2KRbwhByISDX/EujxsK+BAvrhb6H+2rtrLCHK9N01wO014vrIwVQ==", "dev": true, - "requires": { + "dependencies": { "commander": "^2.19.0", "source-map": "~0.6.1", "source-map-support": "~0.5.10" + }, + "bin": { + "terser": "bin/uglifyjs" + }, + "engines": { + "node": ">=6.0.0" } }, - "through2": { + "node_modules/through2": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/through2/-/through2-2.0.5.tgz", "integrity": "sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==", "dev": true, - "requires": { + "dependencies": { "readable-stream": "~2.3.6", "xtend": "~4.0.1" } }, - "timers-browserify": { + "node_modules/timers-browserify": { "version": "2.0.11", "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.11.tgz", "integrity": "sha512-60aV6sgJ5YEbzUdn9c8kYGIqOubPoUdqQCul3SBAsRCZ40s6Y5cMcrW4dt3/k/EsbLVJNl9n6Vz3fTc+k2GeKQ==", "dev": true, - "requires": { + "dependencies": { "setimmediate": "^1.0.4" + }, + "engines": { + "node": ">=0.6.0" } }, - "timsort": { + "node_modules/timsort": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/timsort/-/timsort-0.3.0.tgz", "integrity": "sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=", "dev": true }, - "tiny-inflate": { + "node_modules/tiny-inflate": { "version": "1.0.3", "resolved": "https://registry.npmjs.org/tiny-inflate/-/tiny-inflate-1.0.3.tgz", "integrity": "sha512-pkY1fj1cKHb2seWDy0B16HeWyczlJA9/WW3u3c4z/NiWDsO3DOU5D7nhTLE9CF0yXv/QZFY7sEJmj24dK+Rrqw==", "dev": true }, - "to-arraybuffer": { + "node_modules/to-arraybuffer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", "integrity": "sha1-fSKbH8xjfkZsoIEYCDanqr/4P0M=", "dev": true }, - "to-fast-properties": { + "node_modules/to-fast-properties": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/to-fast-properties/-/to-fast-properties-2.0.0.tgz", "integrity": "sha1-3F5pjL0HkmW8c+A3doGk5Og/YW4=", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "to-object-path": { + "node_modules/to-object-path": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/to-object-path/-/to-object-path-0.3.0.tgz", "integrity": "sha1-KXWIt7Dn4KwI4E5nL4XB9JmeF68=", "dev": true, - "requires": { + "dependencies": { "kind-of": "^3.0.2" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/to-object-path/node_modules/kind-of": { + "version": "3.2.2", + "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", + "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", + "dev": true, "dependencies": { - "kind-of": { - "version": "3.2.2", - "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", - "integrity": "sha1-MeohpzS6ubuw8yRm2JOupR5KPGQ=", - "dev": true, - "requires": { - "is-buffer": "^1.1.5" - } - } + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" } }, - "to-regex": { + "node_modules/to-regex": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/to-regex/-/to-regex-3.0.2.tgz", "integrity": "sha512-FWtleNAtZ/Ki2qtqej2CXTOayOH9bHDQF+Q48VpWyDXjbYxA4Yz8iDB31zXOBUlOHHKidDbqGVrTUvQMPmBGBw==", "dev": true, - "requires": { + "dependencies": { "define-property": "^2.0.2", "extend-shallow": "^3.0.2", "regex-not": "^1.0.2", "safe-regex": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" } }, - "to-regex-range": { + "node_modules/to-regex-range": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-2.1.1.tgz", "integrity": "sha1-fIDBe53+vlmeJzZ+DU3VWQFB2zg=", "dev": true, - "requires": { + "dependencies": { "is-number": "^3.0.0", "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "toidentifier": { + "node_modules/toidentifier": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.6" + } }, - "tough-cookie": { + "node_modules/tough-cookie": { "version": "2.5.0", "resolved": "https://registry.npmjs.org/tough-cookie/-/tough-cookie-2.5.0.tgz", "integrity": "sha512-nlLsUzgm1kfLXSXfRZMc1KLAugd4hqJHDTvc2hDIwS3mZAfMEuMbc03SujMF+GEcpaX/qboeycw6iO8JwVv2+g==", "dev": true, - "requires": { + "dependencies": { "psl": "^1.1.28", "punycode": "^2.1.1" + }, + "engines": { + "node": ">=0.8" } }, - "tr46": { + "node_modules/tr46": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/tr46/-/tr46-1.0.1.tgz", "integrity": "sha1-qLE/1r/SSJUZZ0zN5VujaTtwbQk=", "dev": true, - "requires": { + "dependencies": { "punycode": "^2.1.0" } }, - "tty-browserify": { + "node_modules/tty-browserify": { "version": "0.0.0", "resolved": "https://registry.npmjs.org/tty-browserify/-/tty-browserify-0.0.0.tgz", "integrity": "sha1-oVe6QC2iTpv5V/mqadUk7tQpAaY=", "dev": true }, - "tunnel-agent": { + "node_modules/tunnel-agent": { "version": "0.6.0", "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz", "integrity": "sha1-J6XeoGs2sEoKmWZ3SykIaPD8QP0=", "dev": true, - "requires": { + "dependencies": { "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" } }, - "tweetnacl": { + "node_modules/tweetnacl": { "version": "0.14.5", "resolved": "https://registry.npmjs.org/tweetnacl/-/tweetnacl-0.14.5.tgz", "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", "dev": true }, - "type-check": { + "node_modules/type-check": { "version": "0.3.2", "resolved": "https://registry.npmjs.org/type-check/-/type-check-0.3.2.tgz", "integrity": "sha1-WITKtRLPHTVeP7eE8wgEsrUg23I=", "dev": true, - "requires": { + "dependencies": { "prelude-ls": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" } }, - "typedarray": { + "node_modules/typedarray": { "version": "0.0.6", "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", "dev": true }, - "typescript": { + "node_modules/typescript": { "version": "3.8.3", "resolved": "https://registry.npmjs.org/typescript/-/typescript-3.8.3.tgz", "integrity": "sha512-MYlEfn5VrLNsgudQTVJeNaQFUAI7DkhnOjdpAp4T+ku1TfQClewlbSuTVHiA+8skNBgaf02TL/kLOvig4y3G8w==", - "dev": true + "dev": true, + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } }, - "uncss": { + "node_modules/uncss": { "version": "0.17.3", "resolved": "https://registry.npmjs.org/uncss/-/uncss-0.17.3.tgz", "integrity": "sha512-ksdDWl81YWvF/X14fOSw4iu8tESDHFIeyKIeDrK6GEVTQvqJc1WlOEXqostNwOCi3qAj++4EaLsdAgPmUbEyog==", "dev": true, - "requires": { + "dependencies": { "commander": "^2.20.0", "glob": "^7.1.4", "is-absolute-url": "^3.0.1", @@ -6970,399 +8501,464 @@ "postcss-selector-parser": "6.0.2", "request": "^2.88.0" }, - "dependencies": { - "is-absolute-url": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-3.0.3.tgz", - "integrity": "sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==", - "dev": true - } + "bin": { + "uncss": "bin/uncss" + }, + "engines": { + "node": ">=6.0" + } + }, + "node_modules/uncss/node_modules/is-absolute-url": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/is-absolute-url/-/is-absolute-url-3.0.3.tgz", + "integrity": "sha512-opmNIX7uFnS96NtPmhWQgQx6/NYFgsUXYMllcfzwWKUMwfo8kku1TvE6hkNcH+Q1ts5cMVrsY7j0bxXQDciu9Q==", + "dev": true, + "engines": { + "node": ">=8" } }, - "unicode-canonical-property-names-ecmascript": { + "node_modules/unicode-canonical-property-names-ecmascript": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-1.0.4.tgz", "integrity": "sha512-jDrNnXWHd4oHiTZnx/ZG7gtUTVp+gCcTTKr8L0HjlwphROEW3+Him+IpvC+xcJEFegapiMZyZe02CyuOnRmbnQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "unicode-match-property-ecmascript": { + "node_modules/unicode-match-property-ecmascript": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-1.0.4.tgz", "integrity": "sha512-L4Qoh15vTfntsn4P1zqnHulG0LdXgjSO035fEpdtp6YxXhMT51Q6vgM5lYdG/5X3MjS+k/Y9Xw4SFCY9IkR0rg==", "dev": true, - "requires": { + "dependencies": { "unicode-canonical-property-names-ecmascript": "^1.0.4", "unicode-property-aliases-ecmascript": "^1.0.4" + }, + "engines": { + "node": ">=4" } }, - "unicode-match-property-value-ecmascript": { + "node_modules/unicode-match-property-value-ecmascript": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-1.2.0.tgz", "integrity": "sha512-wjuQHGQVofmSJv1uVISKLE5zO2rNGzM/KCYZch/QQvez7C1hUhBIuZ701fYXExuufJFMPhv2SyL8CyoIfMLbIQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "unicode-property-aliases-ecmascript": { + "node_modules/unicode-property-aliases-ecmascript": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-1.1.0.tgz", "integrity": "sha512-PqSoPh/pWetQ2phoj5RLiaqIk4kCNwoV3CI+LfGmWLKI3rE3kl1h59XpX2BjgDrmbxD9ARtQobPGU1SguCYuQg==", - "dev": true + "dev": true, + "engines": { + "node": ">=4" + } }, - "unicode-trie": { + "node_modules/unicode-trie": { "version": "0.3.1", "resolved": "https://registry.npmjs.org/unicode-trie/-/unicode-trie-0.3.1.tgz", "integrity": "sha1-1nHd3YkQGgi6w3tqUWEBBgIFIIU=", "dev": true, - "requires": { + "dependencies": { "pako": "^0.2.5", "tiny-inflate": "^1.0.0" } }, - "union-value": { + "node_modules/union-value": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/union-value/-/union-value-1.0.1.tgz", "integrity": "sha512-tJfXmxMeWYnczCVs7XAEvIV7ieppALdyepWMkHkwciRpZraG/xwT+s2JN8+pr1+8jCRf80FFzvr+MpQeeoF4Xg==", "dev": true, - "requires": { + "dependencies": { "arr-union": "^3.1.0", "get-value": "^2.0.6", "is-extendable": "^0.1.1", "set-value": "^2.0.1" + }, + "engines": { + "node": ">=0.10.0" } }, - "uniq": { + "node_modules/uniq": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", "dev": true }, - "uniqs": { + "node_modules/uniqs": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/uniqs/-/uniqs-2.0.0.tgz", "integrity": "sha1-/+3ks2slKQaW5uFl1KWe25mOawI=", "dev": true }, - "unquote": { + "node_modules/unquote": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/unquote/-/unquote-1.1.1.tgz", "integrity": "sha1-j97XMk7G6IoP+LkF58CYzcCG1UQ=", "dev": true }, - "unset-value": { + "node_modules/unset-value": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", "integrity": "sha1-g3aHP30jNRef+x5vw6jtDfyKtVk=", "dev": true, - "requires": { + "dependencies": { "has-value": "^0.3.1", "isobject": "^3.0.0" }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-value": { + "version": "0.3.1", + "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", + "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", + "dev": true, "dependencies": { - "has-value": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/has-value/-/has-value-0.3.1.tgz", - "integrity": "sha1-ex9YutpiyoJ+wKIHgCVlSEWZXh8=", - "dev": true, - "requires": { - "get-value": "^2.0.3", - "has-values": "^0.1.4", - "isobject": "^2.0.0" - }, - "dependencies": { - "isobject": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", - "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", - "dev": true, - "requires": { - "isarray": "1.0.0" - } - } - } - }, - "has-values": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", - "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", - "dev": true - } + "get-value": "^2.0.3", + "has-values": "^0.1.4", + "isobject": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-value/node_modules/isobject": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-2.1.0.tgz", + "integrity": "sha1-8GVWEJaj8dou9GJy+BXIQNh+DIk=", + "dev": true, + "dependencies": { + "isarray": "1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/unset-value/node_modules/has-values": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/has-values/-/has-values-0.1.4.tgz", + "integrity": "sha1-bWHeldkd/Km5oCCJrThL/49it3E=", + "dev": true, + "engines": { + "node": ">=0.10.0" } }, - "upath": { + "node_modules/upath": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/upath/-/upath-1.2.0.tgz", "integrity": "sha512-aZwGpamFO61g3OlfT7OQCHqhGnW43ieH9WZeP7QxN/G/jS4jfqUkZxoryvJgVPEcrl5NL/ggHsSmLMHuH64Lhg==", - "dev": true + "dev": true, + "engines": { + "node": ">=4", + "yarn": "*" + } }, - "uri-js": { + "node_modules/uri-js": { "version": "4.2.2", "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.2.2.tgz", "integrity": "sha512-KY9Frmirql91X2Qgjry0Wd4Y+YTdrdZheS8TFwvkbLWf/G5KNJDCh6pKL5OZctEW4+0Baa5idK2ZQuELRwPznQ==", "dev": true, - "requires": { + "dependencies": { "punycode": "^2.1.0" } }, - "urix": { + "node_modules/urix": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/urix/-/urix-0.1.0.tgz", "integrity": "sha1-2pN/emLiH+wf0Y1Js1wpNQZ6bHI=", + "deprecated": "Please see https://github.com/lydell/urix#deprecated", "dev": true }, - "url": { + "node_modules/url": { "version": "0.11.0", "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", "dev": true, - "requires": { + "dependencies": { "punycode": "1.3.2", "querystring": "0.2.0" - }, - "dependencies": { - "punycode": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", - "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", - "dev": true - } } }, - "use": { + "node_modules/url/node_modules/punycode": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/punycode/-/punycode-1.3.2.tgz", + "integrity": "sha1-llOgNvt8HuQjQvIyXM7v6jkmxI0=", + "dev": true + }, + "node_modules/use": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/use/-/use-3.1.1.tgz", "integrity": "sha512-cwESVXlO3url9YWlFW/TA9cshCEhtu7IKJ/p5soJ/gGpj7vbvFrAY/eIioQ6Dw23KjZhYgiIo8HOs1nQ2vr/oQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "util": { + "node_modules/util": { "version": "0.11.1", "resolved": "https://registry.npmjs.org/util/-/util-0.11.1.tgz", "integrity": "sha512-HShAsny+zS2TZfaXxD9tYj4HQGlBezXZMZuM/S5PKLLoZkShZiGk9o5CzukI1LVHZvjdvZ2Sj1aW/Ndn2NB/HQ==", "dev": true, - "requires": { - "inherits": "2.0.3" - }, "dependencies": { - "inherits": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "dev": true - } + "inherits": "2.0.3" } }, - "util-deprecate": { + "node_modules/util-deprecate": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz", "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", "dev": true }, - "util.promisify": { + "node_modules/util.promisify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/util.promisify/-/util.promisify-1.0.1.tgz", "integrity": "sha512-g9JpC/3He3bm38zsLupWryXHoEcS22YHthuPQSJdMy6KNrzIRzWqcsHzD/WUnqe45whVou4VIsPew37DoXWNrA==", "dev": true, - "requires": { + "dependencies": { "define-properties": "^1.1.3", "es-abstract": "^1.17.2", "has-symbols": "^1.0.1", "object.getownpropertydescriptors": "^2.1.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, - "uuid": { + "node_modules/util/node_modules/inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", + "dev": true + }, + "node_modules/uuid": { "version": "3.4.0", "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", - "dev": true + "deprecated": "Please upgrade to version 7 or higher. Older versions may use Math.random() in certain circumstances, which is known to be problematic. See https://v8.dev/blog/math-random for details.", + "dev": true, + "bin": { + "uuid": "bin/uuid" + } }, - "v8-compile-cache": { + "node_modules/v8-compile-cache": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/v8-compile-cache/-/v8-compile-cache-2.1.0.tgz", "integrity": "sha512-usZBT3PW+LOjM25wbqIlZwPeJV+3OSz3M1k1Ws8snlW39dZyYL9lOGC5FgPVHfk0jKmjiDV8Z0mIbVQPiwFs7g==", "dev": true }, - "vendors": { + "node_modules/vendors": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/vendors/-/vendors-1.0.4.tgz", "integrity": "sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w==", - "dev": true + "dev": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } }, - "verror": { + "node_modules/verror": { "version": "1.10.0", "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", "dev": true, - "requires": { + "engines": [ + "node >=0.6.0" + ], + "dependencies": { "assert-plus": "^1.0.0", "core-util-is": "1.0.2", "extsprintf": "^1.2.0" } }, - "vlq": { + "node_modules/vlq": { "version": "0.2.3", "resolved": "https://registry.npmjs.org/vlq/-/vlq-0.2.3.tgz", "integrity": "sha512-DRibZL6DsNhIgYQ+wNdWDL2SL3bKPlVrRiBqV5yuMm++op8W4kGFtaQfCs4KEJn0wBZcHVHJ3eoywX8983k1ow==", "dev": true }, - "vm-browserify": { + "node_modules/vm-browserify": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/vm-browserify/-/vm-browserify-1.1.2.tgz", "integrity": "sha512-2ham8XPWTONajOR0ohOKOHXkm3+gaBmGut3SRuu75xLd/RRaY6vqgh8NBYYk7+RW3u5AtzPQZG8F10LHkl0lAQ==", "dev": true }, - "w3c-hr-time": { + "node_modules/w3c-hr-time": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/w3c-hr-time/-/w3c-hr-time-1.0.2.tgz", "integrity": "sha512-z8P5DvDNjKDoFIHK7q8r8lackT6l+jo/Ye3HOle7l9nICP9lf1Ci25fy9vHd0JOWewkIFzXIEig3TdKT7JQ5fQ==", + "deprecated": "Use your platform's native performance.now() and performance.timeOrigin.", "dev": true, - "requires": { + "dependencies": { "browser-process-hrtime": "^1.0.0" } }, - "w3c-xmlserializer": { + "node_modules/w3c-xmlserializer": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/w3c-xmlserializer/-/w3c-xmlserializer-1.1.2.tgz", "integrity": "sha512-p10l/ayESzrBMYWRID6xbuCKh2Fp77+sA0doRuGn4tTIMrrZVeqfpKjXHY+oDh3K4nLdPgNwMTVP6Vp4pvqbNg==", "dev": true, - "requires": { + "dependencies": { "domexception": "^1.0.1", "webidl-conversions": "^4.0.2", "xml-name-validator": "^3.0.0" } }, - "wcwidth": { + "node_modules/wcwidth": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/wcwidth/-/wcwidth-1.0.1.tgz", "integrity": "sha1-8LDc+RW8X/FSivrbLA4XtTLaL+g=", "dev": true, - "requires": { + "dependencies": { "defaults": "^1.0.3" } }, - "webidl-conversions": { + "node_modules/webidl-conversions": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-4.0.2.tgz", "integrity": "sha512-YQ+BmxuTgd6UXZW3+ICGfyqRyHXVlD5GtQr5+qjiNW7bF0cqrzX500HVXPBOvgXb5YnzDd+h0zqyv61KUD7+Sg==", "dev": true }, - "whatwg-encoding": { + "node_modules/whatwg-encoding": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/whatwg-encoding/-/whatwg-encoding-1.0.5.tgz", "integrity": "sha512-b5lim54JOPN9HtzvK9HFXvBma/rnfFeqsic0hSpjtDbVxR3dJKLc+KB4V6GgiGOvl7CY/KNh8rxSo9DKQrnUEw==", "dev": true, - "requires": { + "dependencies": { "iconv-lite": "0.4.24" } }, - "whatwg-mimetype": { + "node_modules/whatwg-mimetype": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/whatwg-mimetype/-/whatwg-mimetype-2.3.0.tgz", "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==", "dev": true }, - "whatwg-url": { + "node_modules/whatwg-url": { "version": "7.1.0", "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-7.1.0.tgz", "integrity": "sha512-WUu7Rg1DroM7oQvGWfOiAK21n74Gg+T4elXEQYkOhtyLeWiJFoOGLXPKI/9gzIie9CtwVLm8wtw6YJdKyxSjeg==", "dev": true, - "requires": { + "dependencies": { "lodash.sortby": "^4.7.0", "tr46": "^1.0.1", "webidl-conversions": "^4.0.2" } }, - "which": { + "node_modules/which": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", "dev": true, - "requires": { + "dependencies": { "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" } }, - "which-module": { + "node_modules/which-module": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", "dev": true }, - "word-wrap": { + "node_modules/word-wrap": { "version": "1.2.3", "resolved": "https://registry.npmjs.org/word-wrap/-/word-wrap-1.2.3.tgz", "integrity": "sha512-Hz/mrNwitNRh/HUAtM/VT/5VH+ygD6DV7mYKZAtHOrbs8U7lvPS6xf7EJKMF0uW1KJCl0H701g3ZGus+muE5vQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.10.0" + } }, - "wrap-ansi": { + "node_modules/wrap-ansi": { "version": "5.1.0", "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-5.1.0.tgz", "integrity": "sha512-QC1/iN/2/RPVJ5jYK8BGttj5z83LmSKmvbvrXPNCLZSEb32KKVDJDl/MOt2N01qU2H/FkzEa9PKto1BqDjtd7Q==", "dev": true, - "requires": { + "dependencies": { "ansi-styles": "^3.2.0", "string-width": "^3.0.0", "strip-ansi": "^5.0.0" }, + "engines": { + "node": ">=6" + } + }, + "node_modules/wrap-ansi/node_modules/ansi-regex": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", + "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/wrap-ansi/node_modules/strip-ansi": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", + "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", + "dev": true, "dependencies": { - "ansi-regex": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-4.1.0.tgz", - "integrity": "sha512-1apePfXM1UOSqw0o9IiFAovVz9M5S1Dg+4TrDwfMewQ6p/rmMueb7tWZjQ1rx4Loy1ArBggoqGpfqqdI4rondg==", - "dev": true - }, - "strip-ansi": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-5.2.0.tgz", - "integrity": "sha512-DuRs1gKbBqsMKIZlrffwlug8MHkcnpjs5VPmL1PAh+mA30U0DTotfDZ0d2UUsXpPmPmMMJ6W773MaA3J+lbiWA==", - "dev": true, - "requires": { - "ansi-regex": "^4.1.0" - } - } + "ansi-regex": "^4.1.0" + }, + "engines": { + "node": ">=6" } }, - "wrappy": { + "node_modules/wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", "dev": true }, - "ws": { + "node_modules/ws": { "version": "5.2.2", "resolved": "https://registry.npmjs.org/ws/-/ws-5.2.2.tgz", "integrity": "sha512-jaHFD6PFv6UgoIVda6qZllptQsMlDEJkTQcybzzXDYM1XO9Y8em691FGMPmM46WGyLU4z9KMgQN+qrux/nhlHA==", "dev": true, - "requires": { + "dependencies": { "async-limiter": "~1.0.0" } }, - "xml-name-validator": { + "node_modules/xml-name-validator": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/xml-name-validator/-/xml-name-validator-3.0.0.tgz", "integrity": "sha512-A5CUptxDsvxKJEU3yO6DuWBSJz/qizqzJKOMIfUJHETbBw/sFaDxgd6fxm1ewUaM0jZ444Fc5vC5ROYurg/4Pw==", "dev": true }, - "xmlchars": { + "node_modules/xmlchars": { "version": "2.2.0", "resolved": "https://registry.npmjs.org/xmlchars/-/xmlchars-2.2.0.tgz", "integrity": "sha512-JZnDKK8B0RCDw84FNdDAIpZK+JuJw+s7Lz8nksI7SIuU3UXJJslUthsi+uWBUYOwPFwW7W7PRLRfUKpxjtjFCw==", "dev": true }, - "xtend": { + "node_modules/xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", - "dev": true + "dev": true, + "engines": { + "node": ">=0.4" + } }, - "y18n": { + "node_modules/y18n": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.0.tgz", "integrity": "sha512-r9S/ZyXu/Xu9q1tYlpsLIsa3EeLXXk0VwlxqTcFRfg9EhMW+17kbt9G0NrgCmhGb5vT2hyhJZLfDGx+7+5Uj/w==", "dev": true }, - "yargs": { + "node_modules/yargs": { "version": "14.2.3", "resolved": "https://registry.npmjs.org/yargs/-/yargs-14.2.3.tgz", "integrity": "sha512-ZbotRWhF+lkjijC/VhmOT9wSgyBQ7+zr13+YLkhfsSiTriYsMzkTUFP18pFhWwBeMa5gUc1MzbhrO6/VB7c9Xg==", "dev": true, - "requires": { + "dependencies": { "cliui": "^5.0.0", "decamelize": "^1.2.0", "find-up": "^3.0.0", @@ -7374,62 +8970,78 @@ "which-module": "^2.0.0", "y18n": "^4.0.0", "yargs-parser": "^15.0.1" - }, - "dependencies": { - "find-up": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", - "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", - "dev": true, - "requires": { - "locate-path": "^3.0.0" - } - }, - "locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", - "dev": true, - "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" - } - }, - "p-limit": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", - "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", - "dev": true, - "requires": { - "p-try": "^2.0.0" - } - }, - "p-locate": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", - "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", - "dev": true, - "requires": { - "p-limit": "^2.0.0" - } - }, - "p-try": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", - "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", - "dev": true - } } }, - "yargs-parser": { + "node_modules/yargs-parser": { "version": "15.0.1", "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-15.0.1.tgz", "integrity": "sha512-0OAMV2mAZQrs3FkNpDQcBk1x5HXb8X4twADss4S0Iuk+2dGnLOE/fRHrsYm542GduMveyA77OF4wrNJuanRCWw==", "dev": true, - "requires": { + "dependencies": { "camelcase": "^5.0.0", "decamelize": "^1.2.0" } + }, + "node_modules/yargs/node_modules/find-up": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", + "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "dev": true, + "dependencies": { + "locate-path": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/locate-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", + "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "dev": true, + "dependencies": { + "p-locate": "^3.0.0", + "path-exists": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/yargs/node_modules/p-locate": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-3.0.0.tgz", + "integrity": "sha512-x+12w/To+4GFfgJhBEpiDcLozRJGegY+Ei7/z0tSLkMmxGZNybVMSfWj9aJn8Z5Fc7dBUNJOOVgPv2H7IwulSQ==", + "dev": true, + "dependencies": { + "p-limit": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/yargs/node_modules/p-try": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/p-try/-/p-try-2.2.0.tgz", + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true, + "engines": { + "node": ">=6" + } } } } diff --git a/jipt/package.json b/jipt/package.json index 7d850119c..d02652c6a 100644 --- a/jipt/package.json +++ b/jipt/package.json @@ -7,6 +7,7 @@ "build": "parcel index.html", "build-production": "parcel build index.ts --out-dir=jipt-dist --target browser --experimental-scope-hoisting", "build-production-inline": "parcel build index.ts --out-dir=../priv/static/jipt --target browser --experimental-scope-hoisting", + "watch-production-inline": "parcel watch index.ts --out-dir=../priv/static/jipt --target browser", "test": "echo \"Error: no test specified\" && exit 1" }, "keywords": [], diff --git a/jipt/src/accent.ts b/jipt/src/accent.ts index 213a5eef4..21c2b0a57 100644 --- a/jipt/src/accent.ts +++ b/jipt/src/accent.ts @@ -14,7 +14,7 @@ export interface Config { const state = new State({ nodes: new WeakMap(), projectTranslations: new Map(), - refs: new Map(), + refs: new Map() }); const liveNode = new LiveNode(state); @@ -39,7 +39,7 @@ export const Accent = { const mutation = new Mutation(liveNode); mutation.bindEvents(); - }, + } }; export default Accent; diff --git a/jipt/src/frame-listener.ts b/jipt/src/frame-listener.ts index 5c9567735..7f486d21a 100644 --- a/jipt/src/frame-listener.ts +++ b/jipt/src/frame-listener.ts @@ -10,7 +10,7 @@ const enum ACTIONS { redirectIfEmbedded = 'redirectIfEmbedded', login = 'login', loggedIn = 'loggedIn', - changeText = 'changeText', + changeText = 'changeText' } interface Props { @@ -47,9 +47,9 @@ export default class FrameListener { ); } - private handleAccentMessage({data}) { - if (!data.jipt) return; - const action = data.action; + private handleAccentMessage(event: MessageEvent) { + if (!event.data.jipt) return; + const action = event.data.action; if (action === ACTIONS.listTranslations) { return this.handleListTranslations(event); @@ -76,7 +76,7 @@ export default class FrameListener { } } - private handleListTranslations(event) { + private handleListTranslations(event: MessageEvent) { const currentRevision = this.state.getCurrentRevision(); const newRevision = event.data.payload.revisionId; this.state.projectTranslations = event.data.payload.translations; @@ -102,7 +102,7 @@ export default class FrameListener { this.ui.collapse(); } - private handleChangeText(event) { + private handleChangeText(event: MessageEvent) { const ref = this.state.refs.get(event.data.payload.translationId); if (!ref) return; @@ -113,11 +113,11 @@ export default class FrameListener { }); } - private handleUpdateTranslation(event) { + private handleUpdateTranslation(event: MessageEvent) { const ref = this.state.refs.get(event.data.payload.translationId); if (!ref) return; - ref.elements.forEach((meta, node: HTMLElement) => { + ref.elements.forEach((_meta, node: HTMLElement) => { if (!this.liveNode.isLive(node)) return; Mutation.nodeStyleRefresh(node, event.data.payload); diff --git a/jipt/src/mutation/live-node.ts b/jipt/src/mutation/live-node.ts index 642cada6a..7a68589da 100644 --- a/jipt/src/mutation/live-node.ts +++ b/jipt/src/mutation/live-node.ts @@ -34,7 +34,7 @@ export default class LiveNode { attribute.value = newAttribute; this.state.addReference(node, translation, { - attributeName: attribute.name, + attributeName: attribute.name }); }); } diff --git a/jipt/src/mutation/mutation.ts b/jipt/src/mutation/mutation.ts index a47700b33..5f7d570e7 100644 --- a/jipt/src/mutation/mutation.ts +++ b/jipt/src/mutation/mutation.ts @@ -75,7 +75,7 @@ export default class Mutation { characterData: true, characterDataOldValue: true, childList: true, - subtree: true, + subtree: true }); } diff --git a/jipt/src/ui/pin.ts b/jipt/src/ui/pin.ts index 4645be292..3478a42bf 100644 --- a/jipt/src/ui/pin.ts +++ b/jipt/src/ui/pin.ts @@ -78,7 +78,7 @@ export default class Pin { ); } - private pinContent(id) { + private pinContent(id: string) { return `
diff --git a/jipt/src/ui/styles.ts b/jipt/src/ui/styles.ts index 490e569a7..29166f7bd 100644 --- a/jipt/src/ui/styles.ts +++ b/jipt/src/ui/styles.ts @@ -21,11 +21,11 @@ const translationNodeUpdated = // Frame const frameBase = - 'position: fixed; z-index: 10001; bottom: 0; right: 10px; background: #fff; box-shadow: 0 3px 20px rgba(0, 0, 0, 0.3); transition: transform 0.2s ease-in-out;'; + 'position: fixed; overflow: hidden; border-radius: 8px 0 0 0; z-index: 10001; bottom: 1px; right: 0; background: #fff; box-shadow: 0 3px 20px rgba(0, 0, 0, 0.3); transition: transform 0.2s ease-in-out;'; const frameCollapsed = `${frameBase} transform: translate3d(0, 558px, 0);`; const frameExpanded = `${frameBase} transform: translate3d(0, 0, 0);`; const frameCentered = - 'position: fixed; z-index: 10003; top: calc(50% - 300px); right: calc(50% - 300px); background: #fff; box-shadow: 0 3px 20px rgba(0, 0, 0, 0.3);'; + 'position: fixed; z-index: 10003; top: calc(50% - 300px); right: calc(50% - 300px); background: #fff; box-shadow: 0 3px 20px rgba(0, 0, 0, 0.2);'; const frameCollapseButton = 'cursor: pointer; position: absolute; right: 4px; top: 2px; width: 24px; height: 24px; text-align: center; color: #555; font-size: 20px;'; const frameExpandButton = @@ -52,5 +52,5 @@ export default { set, translationNode, translationNodeConflicted, - translationNodeUpdated, + translationNodeUpdated }; diff --git a/jipt/src/ui/ui.ts b/jipt/src/ui/ui.ts index 73cdb60bd..36b677958 100644 --- a/jipt/src/ui/ui.ts +++ b/jipt/src/ui/ui.ts @@ -126,7 +126,7 @@ export default class UI { return element; } - private buildFrame(config) { + private buildFrame(config: Config) { const element = document.createElement('iframe'); const query = this.state.getCurrentRevision() ? `?revisionId=${this.state.getCurrentRevision()}` diff --git a/lib/accent.ex b/lib/accent.ex index d5509ddae..ba8c376f5 100644 --- a/lib/accent.ex +++ b/lib/accent.ex @@ -16,6 +16,11 @@ defmodule Accent do {Phoenix.PubSub, [name: Accent.PubSub, adapter: Phoenix.PubSub.PG2]} ] + children = + if Application.get_env(:ueberauth, Ueberauth.Strategy.OIDC)[:default_oidc][:client_id], + do: [{OpenIDConnect.Worker, Application.get_env(:ueberauth, Ueberauth.Strategy.OIDC)} | children], + else: children + if Application.get_env(:sentry, :dsn) do {:ok, _} = Logger.add_backend(Sentry.LoggerBackend) end diff --git a/lib/accent/integrations/execute/aws_s3.ex b/lib/accent/integrations/execute/aws_s3.ex new file mode 100644 index 000000000..b5a206362 --- /dev/null +++ b/lib/accent/integrations/execute/aws_s3.ex @@ -0,0 +1,65 @@ +defmodule Accent.IntegrationManager.Execute.AWSS3 do + @moduledoc false + + alias Accent.Hook + + def upload_translations(integration, user, params) do + {uploads, version_tag} = Accent.IntegrationManager.Execute.UploadDocuments.all(integration, params) + + # To support bucket with '.' in the name, we need to use the region subdomain. + # The us-east-1 subdomain is not s3-us-east-1. It’s s3 only. + base_url = + case integration.data.aws_s3_region do + "us-east-1" -> "https://s3.amazonaws.com/#{integration.data.aws_s3_bucket}" + region -> "https://s3-#{region}.amazonaws.com/#{integration.data.aws_s3_bucket}" + end + + url = + Path.join([ + base_url, + integration.data.aws_s3_path_prefix + ]) + + uri = URI.parse(url) + + document_urls = + for upload <- uploads do + {url, document_name} = Accent.IntegrationManager.Execute.UploadDocuments.url(upload, uri, version_tag) + + headers = + :aws_signature.sign_v4( + integration.data.aws_s3_access_key_id, + integration.data.aws_s3_secret_access_key, + integration.data.aws_s3_region, + "s3", + :calendar.universal_time(), + "put", + url, + [ + {"host", uri.authority}, + {"x-amz-acl", "public-read"}, + {"x-amz-tagging", + "ACCENT_VERSION=#{Application.get_env(:accent, :version)}&USER_ID=#{user.id}&PROJECT_ID=#{integration.project_id}"} + ], + upload.render, + uri_encode_path: false + ) + + {:ok, %{status_code: 200}} = HTTPoison.put(url, {:file, upload.file}, headers) + + %{name: document_name, url: url} + end + + Hook.outbound(%Hook.Context{ + event: "integration_execute_azure_storage_container", + project_id: integration.project_id, + user_id: user.id, + payload: %{ + version_tag: version_tag, + document_urls: document_urls + } + }) + + :ok + end +end diff --git a/lib/accent/integrations/execute/azure_storage_container.ex b/lib/accent/integrations/execute/azure_storage_container.ex index 067087e56..24ea48405 100644 --- a/lib/accent/integrations/execute/azure_storage_container.ex +++ b/lib/accent/integrations/execute/azure_storage_container.ex @@ -1,95 +1,31 @@ defmodule Accent.IntegrationManager.Execute.AzureStorageContainer do @moduledoc false - alias Accent.Document - alias Accent.Repo - alias Accent.Revision - alias Accent.Scopes.Document, as: DocumentScope - alias Accent.Scopes.Revision, as: RevisionScope - alias Accent.Scopes.Translation, as: TranslationScope - alias Accent.Scopes.Version, as: VersionScope - alias Accent.Translation - alias Accent.Version + alias Accent.Hook - def upload_translations(integration, params) do - project = Repo.one!(Ecto.assoc(integration, :project)) - version = fetch_version(project, params) - documents = fetch_documents(project) - revisions = fetch_revisions(project) - master_revision = Repo.preload(Repo.one!(RevisionScope.master(Ecto.assoc(project, :revisions))), :language) + def upload_translations(integration, user, params) do + {uploads, version_tag} = Accent.IntegrationManager.Execute.UploadDocuments.all(integration, params) - uploads = - Enum.flat_map(documents, fn document -> - Enum.flat_map(revisions, fn revision -> - translations = fetch_translations(document, revision, version) + uri = URI.parse(integration.data.azure_storage_container_sas) - if Enum.any?(translations) do - render_options = %{ - translations: translations, - master_language: Revision.language(master_revision), - language: Revision.language(revision), - document: document - } + document_urls = + for upload <- uploads do + {url, document_name} = Accent.IntegrationManager.Execute.UploadDocuments.url(upload, uri, version_tag) + HTTPoison.put(url, {:file, upload.file}, [{"x-ms-blob-type", "BlockBlob"}]) - %{render: render} = Accent.TranslationsRenderer.render_translations(render_options) - [%{document: %{document | render: render}, language: Accent.Revision.language(revision)}] - else - [] - end - end) - end) + %{name: document_name, url: url} + end - for upload <- uploads do - file = Path.join([System.tmp_dir(), Accent.Utils.SecureRandom.urlsafe_base64(16)]) - :ok = File.write(file, upload.document.render) - - uri = URI.parse(integration.data.azure_storage_container_sas) - extension = Accent.DocumentFormat.extension_by_format(upload.document.format) - - path = - Path.join([ - uri.path, - (version && version.tag) || "latest", - upload.language.slug, - upload.document.path <> "." <> extension - ]) - - HTTPoison.put(URI.to_string(%{uri | path: path}), {:file, file}, [{"x-ms-blob-type", "BlockBlob"}]) - end + Hook.outbound(%Hook.Context{ + event: "integration_execute_azure_storage_container", + project_id: integration.project_id, + user_id: user.id, + payload: %{ + version_tag: version_tag, + document_urls: document_urls + } + }) :ok end - - defp fetch_version(project, %{target_version: :specific, tag: tag}) do - Version - |> VersionScope.from_project(project.id) - |> VersionScope.from_tag(tag) - |> Repo.one() - end - - defp fetch_version(_, _) do - nil - end - - defp fetch_documents(project) do - Document - |> DocumentScope.from_project(project.id) - |> Repo.all() - end - - defp fetch_revisions(project) do - Revision - |> RevisionScope.from_project(project.id) - |> Repo.all() - |> Repo.preload(:language) - end - - defp fetch_translations(document, revision, version) do - Translation - |> TranslationScope.active() - |> TranslationScope.from_document(document.id) - |> TranslationScope.from_revision(revision.id) - |> TranslationScope.from_version(version && version.id) - |> Repo.all() - end end diff --git a/lib/accent/integrations/execute/upload_documents.ex b/lib/accent/integrations/execute/upload_documents.ex new file mode 100644 index 000000000..c6c8722d7 --- /dev/null +++ b/lib/accent/integrations/execute/upload_documents.ex @@ -0,0 +1,106 @@ +defmodule Accent.IntegrationManager.Execute.UploadDocuments do + @moduledoc false + + alias Accent.Document + alias Accent.Repo + alias Accent.Revision + alias Accent.Scopes.Document, as: DocumentScope + alias Accent.Scopes.Revision, as: RevisionScope + alias Accent.Scopes.Translation, as: TranslationScope + alias Accent.Scopes.Version, as: VersionScope + alias Accent.Translation + alias Accent.Version + + def url(upload, uri, version_tag) do + extension = Accent.DocumentFormat.extension_by_format(upload.document.format) + document_name = upload.document.path <> "." <> extension + + path = + Path.join([ + uri.path || "/", + version_tag, + upload.language.slug, + document_name + ]) + + {URI.to_string(%{uri | path: path}), document_name} + end + + def all(integration, params) do + project = Repo.one!(Ecto.assoc(integration, :project)) + version = fetch_version(project, params) + documents = fetch_documents(project) + revisions = fetch_revisions(project) + + master_revision = + Repo.preload(Repo.one!(RevisionScope.master(Ecto.assoc(project, :revisions))), :language) + + version_tag = (version && version.tag) || "latest" + + uploads = + Enum.flat_map(documents, fn document -> + Enum.flat_map(revisions, fn revision -> + translations = fetch_translations(document, revision, version) + + if Enum.any?(translations) do + render_options = %{ + translations: translations, + master_language: Revision.language(master_revision), + language: Revision.language(revision), + document: document + } + + %{render: render} = Accent.TranslationsRenderer.render_translations(render_options) + file = Path.join([System.tmp_dir(), Accent.Utils.SecureRandom.urlsafe_base64(16)]) + :ok = File.write(file, render) + + [ + %{ + file: file, + render: render, + document: document, + language: Accent.Revision.language(revision) + } + ] + else + [] + end + end) + end) + + {uploads, version_tag} + end + + defp fetch_version(project, %{target_version: :specific, tag: tag}) do + Version + |> VersionScope.from_project(project.id) + |> VersionScope.from_tag(tag) + |> Repo.one() + end + + defp fetch_version(_, _) do + nil + end + + defp fetch_documents(project) do + Document + |> DocumentScope.from_project(project.id) + |> Repo.all() + end + + defp fetch_revisions(project) do + Revision + |> RevisionScope.from_project(project.id) + |> Repo.all() + |> Repo.preload(:language) + end + + defp fetch_translations(document, revision, version) do + Translation + |> TranslationScope.active() + |> TranslationScope.from_document(document.id) + |> TranslationScope.from_revision(revision.id) + |> TranslationScope.from_version(version && version.id) + |> Repo.all() + end +end diff --git a/lib/accent/integrations/integration_manager.ex b/lib/accent/integrations/integration_manager.ex index 5d7504cfd..5c4194b16 100644 --- a/lib/accent/integrations/integration_manager.ex +++ b/lib/accent/integrations/integration_manager.ex @@ -23,7 +23,7 @@ defmodule Accent.IntegrationManager do @spec execute(Integration.t(), User.t(), map()) :: {:ok, Integration.t()} def execute(integration, user, params) do - case execute_integration(integration, params) do + case execute_integration(integration, user, params) do :ok -> integration |> change(%{last_executed_at: DateTime.utc_now(), last_executed_by_user_id: user.id}) @@ -45,22 +45,33 @@ defmodule Accent.IntegrationManager do defp changeset(model, params) do model |> cast(params, [:project_id, :user_id, :service, :events]) - |> validate_inclusion(:service, ~w(slack github discord azure_storage_container)) + |> validate_inclusion(:service, ~w(slack github discord azure_storage_container aws_s3)) |> cast_embed(:data, with: changeset_data(params[:service] || model.service)) |> foreign_key_constraint(:project_id) |> validate_required([:service, :data]) end - defp execute_integration(%{service: "azure_storage_container"} = integration, params) do + defp execute_integration(%{service: "azure_storage_container"} = integration, user, params) do Accent.IntegrationManager.Execute.AzureStorageContainer.upload_translations( integration, + user, params[:azure_storage_container] ) :ok end - defp execute_integration(_integration, _params) do + defp execute_integration(%{service: "aws_s3"} = integration, user, params) do + Accent.IntegrationManager.Execute.AWSS3.upload_translations( + integration, + user, + params[:aws_s3] + ) + + :ok + end + + defp execute_integration(_integration, _user, _params) do :noop end @@ -88,6 +99,26 @@ defmodule Accent.IntegrationManager do end end + defp changeset_data("aws_s3") do + fn model, params -> + model + |> cast(params, [ + :aws_s3_bucket, + :aws_s3_path_prefix, + :aws_s3_region, + :aws_s3_access_key_id, + :aws_s3_secret_access_key + ]) + |> validate_required([ + :aws_s3_bucket, + :aws_s3_path_prefix, + :aws_s3_region, + :aws_s3_access_key_id, + :aws_s3_secret_access_key + ]) + end + end + defp changeset_data(_) do fn model, params -> cast(model, params, []) end end diff --git a/lib/accent/projects/project_deleter.ex b/lib/accent/projects/project_deleter.ex index 86855b3f9..d1de403c2 100644 --- a/lib/accent/projects/project_deleter.ex +++ b/lib/accent/projects/project_deleter.ex @@ -1,5 +1,8 @@ defmodule Accent.ProjectDeleter do @moduledoc false + import Ecto.Query, only: [from: 2] + + alias Accent.Operation alias Accent.Repo def delete(project: project) do @@ -7,6 +10,17 @@ defmodule Accent.ProjectDeleter do |> Ecto.assoc(:all_collaborators) |> Repo.delete_all() + Repo.transaction(fn -> + Repo.delete_all( + from(operations in Operation, + inner_join: revisions in assoc(operations, :revision), + where: revisions.project_id == ^project.id + ) + ) + + Repo.delete_all(from(operations in Operation, where: operations.project_id == ^project.id)) + end) + {:ok, project} end end diff --git a/lib/accent/prompt_config/prompt_config_manager.ex b/lib/accent/prompt_config/prompt_config_manager.ex index 3c6e8c12b..3a405f466 100644 --- a/lib/accent/prompt_config/prompt_config_manager.ex +++ b/lib/accent/prompt_config/prompt_config_manager.ex @@ -11,7 +11,8 @@ defmodule Accent.PromptConfigManager do "config" => %{ "key" => params[:config_key] }, - "provider" => params[:provider] + "provider" => params[:provider], + "use_platform" => params[:use_platform] } } diff --git a/lib/accent/schemas/integration.ex b/lib/accent/schemas/integration.ex index 0abb689f4..d0b24ba18 100644 --- a/lib/accent/schemas/integration.ex +++ b/lib/accent/schemas/integration.ex @@ -10,6 +10,11 @@ defmodule Accent.Integration do embeds_one(:data, IntegrationData, on_replace: :update) do field(:url) field(:azure_storage_container_sas) + field(:aws_s3_bucket) + field(:aws_s3_path_prefix) + field(:aws_s3_region) + field(:aws_s3_access_key_id) + field(:aws_s3_secret_access_key) end belongs_to(:project, Accent.Project) diff --git a/lib/accent/schemas/version.ex b/lib/accent/schemas/version.ex index 4345e758f..18ffb1404 100644 --- a/lib/accent/schemas/version.ex +++ b/lib/accent/schemas/version.ex @@ -6,6 +6,7 @@ defmodule Accent.Version do field(:name, :string) field(:tag, :string) field(:parsed_tag, :any, virtual: true) + field(:copy_on_update_translation, :boolean) belongs_to(:user, Accent.User) belongs_to(:project, Accent.Project) @@ -17,10 +18,11 @@ defmodule Accent.Version do end @required_fields ~w(project_id user_id name tag)a + @optional_fields ~w(copy_on_update_translation)a def changeset(model, params) do model - |> cast(params, @required_fields) + |> cast(params, @required_fields ++ @optional_fields) |> validate_required(@required_fields) |> unique_constraint(:tag, name: :versions_tag_project_id_index) end diff --git a/lib/accent/scopes/operation.ex b/lib/accent/scopes/operation.ex index 937f1a2fd..06be34e4a 100644 --- a/lib/accent/scopes/operation.ex +++ b/lib/accent/scopes/operation.ex @@ -39,11 +39,19 @@ defmodule Accent.Scopes.Operation do Accent.Operation iex> Accent.Scopes.Operation.filter_from_project(Accent.Operation, "test") #Ecto.Query + iex> Accent.Scopes.Operation.filter_from_project(Accent.Operation, "test", "sync") + #Ecto.Query """ - @spec filter_from_project(Ecto.Queryable.t(), String.t() | nil) :: Ecto.Queryable.t() - def filter_from_project(query, nil), do: query + @spec filter_from_project(Ecto.Queryable.t(), String.t() | nil, String.t() | nil) :: Ecto.Queryable.t() + def filter_from_project(query, project_id, action \\ nil) + def filter_from_project(query, nil, _), do: query - def filter_from_project(query, project_id) do + def filter_from_project(query, project_id, action) + when action in ~w(sync batch_sync batch_merge create_version document_delete merge new_slave version_new) do + from(query, where: [project_id: ^project_id]) + end + + def filter_from_project(query, project_id, _) do from(o in query, left_join: r in assoc(o, :revision), where: r.project_id == ^project_id or o.project_id == ^project_id diff --git a/lib/accent/scopes/translation.ex b/lib/accent/scopes/translation.ex index 29537da5f..08eb1b20b 100644 --- a/lib/accent/scopes/translation.ex +++ b/lib/accent/scopes/translation.ex @@ -421,23 +421,4 @@ defmodule Accent.Scopes.Translation do _ -> query end end - - @doc """ - ## Examples - - iex> Accent.Scopes.Translation.select_key_text(Accent.Translation) - #Ecto.Query - """ - @spec select_key_text(Queryable.t()) :: Queryable.t() - def select_key_text(query) do - from( - translation in query, - select: %{ - id: translation.id, - key: translation.key, - updated_at: translation.updated_at, - corrected_text: translation.corrected_text - } - ) - end end diff --git a/lib/accent/telemetry_ui/ecto_psql_extras.ex b/lib/accent/telemetry_ui/ecto_psql_extras.ex index 833bcf800..16917a604 100644 --- a/lib/accent/telemetry_ui/ecto_psql_extras.ex +++ b/lib/accent/telemetry_ui/ecto_psql_extras.ex @@ -4,25 +4,17 @@ defmodule Accent.TelemetryUI.EctoPSQLExtras do @queries [ bloat: EctoPSQLExtras.Bloat, - blocking: EctoPSQLExtras.Blocking, cache_hit: EctoPSQLExtras.CacheHit, + connections: EctoPSQLExtras.Connections, table_cache_hit: EctoPSQLExtras.TableCacheHit, index_cache_hit: EctoPSQLExtras.IndexCacheHit, index_size: EctoPSQLExtras.IndexSize, index_usage: EctoPSQLExtras.IndexUsage, - locks: EctoPSQLExtras.Locks, - all_locks: EctoPSQLExtras.AllLocks, - long_running_queries: EctoPSQLExtras.LongRunningQueries, records_rank: EctoPSQLExtras.RecordsRank, - seq_scans: EctoPSQLExtras.SeqScans, - table_indexes_size: EctoPSQLExtras.TableIndexesSize, table_size: EctoPSQLExtras.TableSize, - total_index_size: EctoPSQLExtras.TotalIndexSize, + table_indexes_size: EctoPSQLExtras.TableIndexesSize, total_table_size: EctoPSQLExtras.TotalTableSize, - unused_indexes: EctoPSQLExtras.UnusedIndexes, - duplicate_indexes: EctoPSQLExtras.DuplicateIndexes, - null_indexes: EctoPSQLExtras.NullIndexes, - vacuum_stats: EctoPSQLExtras.VacuumStats + unused_indexes: EctoPSQLExtras.UnusedIndexes ] def all(repo), do: Enum.map(Keyword.keys(@queries), &new(repo, &1)) @@ -51,7 +43,7 @@ defmodule Accent.TelemetryUI.EctoPSQLExtras do Enum.map(result.rows, &parse_row(&1, types)) end - TableRex.quick_render!(rows, names) + {rows, names} end defp parse_row(list, types) do @@ -102,13 +94,39 @@ defmodule Accent.TelemetryUI.EctoPSQLExtras do end def to_html(metric, _assigns) do + {rows, names} = metric.data + + names = + Enum.map_join(names, "", fn name -> + "" <> + to_string(name) <> "" + end) + + rows = + for cells <- rows do + "
" <> + Enum.map_join(cells, "", fn cell -> + "" <> + cell <> "" + end) <> "
" + end + {:safe, """
-

#{metric.title}

+

#{metric.title}

-
#{metric.data}
+ + + + #{names} + + + + #{rows} + +
"""} end diff --git a/lib/graphql/mutations/integration.ex b/lib/graphql/mutations/integration.ex index b25289b8f..48ac89d07 100644 --- a/lib/graphql/mutations/integration.ex +++ b/lib/graphql/mutations/integration.ex @@ -12,15 +12,30 @@ defmodule Accent.GraphQL.Mutations.Integration do value(:latest) end + enum :project_integration_execute_aws_s3_target_version do + value(:specific) + value(:latest) + end + input_object :project_integration_execute_azure_storage_container_input do field(:target_version, :project_integration_execute_azure_storage_container_target_version) field(:tag, :string) end + input_object :project_integration_execute_aws_s3_input do + field(:target_version, :project_integration_execute_aws_s3_target_version) + field(:tag, :string) + end + input_object :project_integration_data_input do field(:id, :id) field(:url, :string) field(:azure_storage_container_sas, :string) + field(:aws_s3_region, :string) + field(:aws_s3_bucket, :string) + field(:aws_s3_path_prefix, :string) + field(:aws_s3_access_key_id, :string) + field(:aws_s3_secret_access_key, :string) end payload_object(:project_integration_payload, :project_integration) @@ -39,6 +54,7 @@ defmodule Accent.GraphQL.Mutations.Integration do field :execute_project_integration, :project_integration_payload do arg(:id, non_null(:id)) arg(:azure_storage_container, :project_integration_execute_azure_storage_container_input) + arg(:aws_s3, :project_integration_execute_aws_s3_input) resolve(integration_authorize(:execute_project_integration, &IntegrationResolver.execute/3)) middleware(&build_payload/2) diff --git a/lib/graphql/mutations/prompt.ex b/lib/graphql/mutations/prompt.ex index 52342cc13..b6a52a595 100644 --- a/lib/graphql/mutations/prompt.ex +++ b/lib/graphql/mutations/prompt.ex @@ -42,6 +42,7 @@ defmodule Accent.GraphQL.Mutations.Prompt do arg(:project_id, non_null(:id)) arg(:provider, non_null(:string)) arg(:config_key, :string) + arg(:use_platform, non_null(:boolean)) resolve(project_authorize(:save_project_prompt_config, &Resolver.save_config/3, :project_id)) end diff --git a/lib/graphql/mutations/version.ex b/lib/graphql/mutations/version.ex index 05564f8a8..78a4f3ce9 100644 --- a/lib/graphql/mutations/version.ex +++ b/lib/graphql/mutations/version.ex @@ -9,6 +9,7 @@ defmodule Accent.GraphQL.Mutations.Version do arg(:project_id, non_null(:id)) arg(:name, non_null(:string)) arg(:tag, non_null(:string)) + arg(:copy_on_update_translation, non_null(:boolean)) resolve(project_authorize(:create_version, &Accent.GraphQL.Resolvers.Version.create/3, :project_id)) end @@ -17,6 +18,7 @@ defmodule Accent.GraphQL.Mutations.Version do arg(:id, non_null(:id)) arg(:name, non_null(:string)) arg(:tag, non_null(:string)) + arg(:copy_on_update_translation, non_null(:boolean)) resolve(version_authorize(:update_version, &Accent.GraphQL.Resolvers.Version.update/3)) end diff --git a/lib/graphql/paginated.ex b/lib/graphql/paginated.ex index 84d10cbc5..995276070 100644 --- a/lib/graphql/paginated.ex +++ b/lib/graphql/paginated.ex @@ -15,8 +15,8 @@ defmodule Accent.GraphQL.Paginated do @enforce_keys [:entries, :meta] defstruct entries: [], meta: %{} - def paginate(query, args) do - Accent.Repo.paginate(query, page: args[:page], page_size: args[:page_size]) + def paginate(query, args, options \\ []) do + Accent.Repo.paginate(query, page: args[:page], page_size: args[:page_size], options: options) end def format(paginated_list) do diff --git a/lib/graphql/resolvers/integration.ex b/lib/graphql/resolvers/integration.ex index f946fa10d..c1672a9a9 100644 --- a/lib/graphql/resolvers/integration.ex +++ b/lib/graphql/resolvers/integration.ex @@ -6,6 +6,10 @@ defmodule Accent.GraphQL.Resolvers.Integration do alias Accent.IntegrationManager alias Accent.Plugs.GraphQLContext alias Accent.Project + alias Accent.Repo + alias Accent.Scopes.Integration, as: IntegrationScope + + require Ecto.Query @typep integration_operation :: Accent.GraphQL.Response.t() @@ -38,4 +42,13 @@ defmodule Accent.GraphQL.Resolvers.Integration do |> IntegrationManager.delete() |> build() end + + @spec list_project(Project.t(), map(), GraphQLContext.t()) :: {:ok, [Integration.t()]} + def list_project(project, _args, _) do + Integration + |> IntegrationScope.from_project(project.id) + |> Ecto.Query.order_by(desc: :inserted_at) + |> Repo.all() + |> then(&{:ok, &1}) + end end diff --git a/lib/graphql/resolvers/project.ex b/lib/graphql/resolvers/project.ex index eeed1e094..b91a30c17 100644 --- a/lib/graphql/resolvers/project.ex +++ b/lib/graphql/resolvers/project.ex @@ -62,7 +62,7 @@ defmodule Accent.GraphQL.Resolvers.Project do "name" => args.name, "main_color" => args.main_color, "logo" => args.logo, - "locked_file_operations" => args.is_file_operations_locked + "locked_file_operations" => args.is_file_operations_locked || false } case ProjectUpdater.update( @@ -85,9 +85,8 @@ defmodule Accent.GraphQL.Resolvers.Project do Project |> Query.join(:inner, [p], c in assoc(p, :collaborators)) |> Query.where([_, c], c.user_id == ^viewer.id) - |> Query.order_by([p, _], asc: p.name) + |> Query.order_by([p, _], desc_nulls_first: p.last_synced_at) |> ProjectScope.from_search(args[:query]) - |> ProjectScope.with_stats() |> Paginated.paginate(args) |> Paginated.format() @@ -96,7 +95,6 @@ defmodule Accent.GraphQL.Resolvers.Project do |> Query.join(:inner, [p], c in assoc(p, :collaborators)) |> Query.where([_, c], c.user_id == ^viewer.id) |> ProjectScope.from_ids(args[:node_ids]) - |> ProjectScope.with_stats() |> Repo.all() projects = Map.put(paginated_projects, :nodes, nodes_projects) @@ -115,8 +113,8 @@ defmodule Accent.GraphQL.Resolvers.Project do @spec last_activity(Project.t(), any(), GraphQLContext.t()) :: {:ok, Operation.t() | nil} def last_activity(project, args, _) do Operation - |> OperationScope.filter_from_project(project.id) |> OperationScope.filter_from_action(args[:action]) + |> OperationScope.filter_from_project(project.id, args[:action]) |> OperationScope.order_last_to_first() |> Query.limit(1) |> Repo.one() diff --git a/lib/graphql/resolvers/revision.ex b/lib/graphql/resolvers/revision.ex index 06053ee63..90bace358 100644 --- a/lib/graphql/resolvers/revision.ex +++ b/lib/graphql/resolvers/revision.ex @@ -113,7 +113,7 @@ defmodule Accent.GraphQL.Resolvers.Revision do end @spec show_project(Project.t(), %{id: String.t()}, GraphQLContext.t()) :: {:ok, Revision.t() | nil} - def show_project(project, %{id: id} = args, _) do + def show_project(project, %{id: id} = args, _) when not is_nil(id) do Revision |> RevisionScope.from_project(project.id) |> RevisionScope.with_stats(version_id: args[:version_id]) diff --git a/lib/graphql/resolvers/translation.ex b/lib/graphql/resolvers/translation.ex index 31968b8ed..31190c1d9 100644 --- a/lib/graphql/resolvers/translation.ex +++ b/lib/graphql/resolvers/translation.ex @@ -1,5 +1,8 @@ defmodule Accent.GraphQL.Resolvers.Translation do @moduledoc false + import Absinthe.Resolution.Helpers, only: [batch: 3] + + alias Accent.Document alias Accent.GraphQL.Paginated alias Accent.Plugs.GraphQLContext alias Accent.Project @@ -27,6 +30,59 @@ defmodule Accent.GraphQL.Resolvers.Translation do |> then(&{:ok, &1}) end + def batch_translation_ids(grouped_translation, _args, _resolution) do + ids = Enum.reject(grouped_translation.translation_ids, &is_nil/1) + + batch( + {__MODULE__, :from_translation_ids}, + ids, + fn batch_results -> + translations = + Map.get(batch_results, {grouped_translation.key, grouped_translation.document_id}, []) + + translations = + Enum.sort(translations, fn a, b -> + cond do + a.revision.master == true -> true + DateTime.compare(a.revision.inserted_at, b.revision.inserted_at) === :lt -> true + true -> false + end + end) + + {:ok, translations} + end + ) + end + + def from_translation_ids(_, ids) do + ids = Enum.map(List.flatten(ids), &Ecto.UUID.cast!(&1)) + + query = + Query.from(translations in Translation, + preload: [:revision], + where: translations.id in ^ids + ) + + query + |> Repo.all() + |> Enum.group_by(&{&1.key, &1.document_id}) + end + + def batch_document(grouped_translation, _args, _resolution) do + batch({__MODULE__, :from_document_id}, grouped_translation.document_id, fn batch_results -> + [document | _] = Map.get(batch_results, grouped_translation.document_id) + {:ok, document} + end) + end + + def from_document_id(_, ids) do + query = Query.from(documents in Document, where: documents.id in ^ids) + + query + |> Repo.all() + |> Enum.group_by(& &1.id) + end + @spec correct(Translation.t(), %{text: String.t()}, GraphQLContext.t()) :: translation_operation def correct(translation, %{text: text}, info) do %Context{} @@ -124,12 +180,29 @@ defmodule Accent.GraphQL.Resolvers.Translation do translations = Translation |> list(args, project.id) - |> TranslationScope.from_project(project.id) |> Paginated.paginate(args) {:ok, Paginated.format(translations)} end + @spec list_grouped_project(Project.t(), map(), GraphQLContext.t()) :: {:ok, Paginated.t(Translation.t())} + def list_grouped_project(project, args, _) do + total_entries = + Translation + |> list_grouped_count(args, project.id) + |> Repo.all() + |> Enum.count() + + translations = + Translation + |> list_grouped(args, project.id) + |> Paginated.paginate(args, total_entries: total_entries) + + revisions = grouped_related_revisions(Map.put(args, :project_id, project.id)) + + {:ok, Map.put(Paginated.format(translations), :revisions, revisions)} + end + @spec related_translations(Translation.t(), map(), struct()) :: {:ok, [Translation.t()]} def related_translations(translation, _, _) do translations = @@ -179,7 +252,112 @@ defmodule Accent.GraphQL.Resolvers.Translation do end end + defp grouped_related_revisions(args) do + query_revision_ids = + if Enum.empty?(args[:related_revisions]) do + Query.from( + revisions in Revision, + where: revisions.project_id == ^args[:project_id], + order_by: [asc: :inserted_at], + limit: 2 + ) + else + Query.from( + revisions in Revision, + where: revisions.id in ^args[:related_revisions], + order_by: [asc: :inserted_at], + limit: 2 + ) + end + + Repo.all(query_revision_ids) + end + + defp grouped_related_query(schema, args, project_id) do + revision_ids = + if Enum.empty?(args[:related_revisions]) do + Enum.map(grouped_related_revisions(Map.put(args, :project_id, project_id)), & &1.id) + else + args[:related_revisions] + end + + query = + schema + |> TranslationScope.from_version(args[:version]) + |> TranslationScope.from_project(project_id) + |> TranslationScope.from_revisions(revision_ids) + |> TranslationScope.active() + |> TranslationScope.not_locked() + + {query, revision_ids} + end + + defp list_grouped_count(schema, args, project_id) do + query = list_base_query(schema, args, project_id) + {related_query, revision_ids} = grouped_related_query(schema, args, project_id) + + query = + Query.from( + translations in query, + left_join: related_translations in subquery(related_query), + as: :related_translations, + on: + related_translations.revision_id in ^revision_ids and + related_translations.key == translations.key and + related_translations.document_id == translations.document_id, + distinct: [translations.key, translations.document_id], + select: translations.key, + group_by: [translations.key, translations.document_id] + ) + + if args[:is_conflicted] do + Query.from([related_translations: related_translations] in query, + having: fragment("array_agg(distinct(?))", related_translations.conflicted) != [false] + ) + else + query + end + end + + defp list_grouped(schema, args, project_id) do + query = list_base_query(schema, args, project_id) + {related_query, revision_ids} = grouped_related_query(schema, args, project_id) + + query = + Query.from( + translations in query, + left_join: related_translations in subquery(related_query), + as: :related_translations, + on: + related_translations.revision_id in ^revision_ids and + related_translations.key == translations.key and + related_translations.document_id == translations.document_id, + distinct: translations.key, + select: %{ + key: translations.key, + document_id: translations.document_id, + translation_ids: fragment("array_agg(distinct(?))", related_translations.id) + }, + group_by: [translations.key, translations.document_id] + ) + + if args[:is_conflicted] do + Query.from([related_translations: related_translations] in query, + having: fragment("array_agg(distinct(?))", related_translations.conflicted) != [false] + ) + else + query + end + end + defp list(schema, args, project_id) do + schema + |> list_base_query(args, project_id) + |> Query.distinct(true) + |> Query.preload(:revision) + end + + defp list_base_query(schema, args, project_id) do schema |> TranslationScope.active() |> TranslationScope.not_locked() @@ -193,7 +371,6 @@ defmodule Accent.GraphQL.Resolvers.Translation do |> TranslationScope.parse_empty(args[:is_text_empty]) |> TranslationScope.parse_commented_on(args[:is_commented_on]) |> TranslationScope.from_version(args[:version]) - |> Query.distinct(true) - |> Query.preload(:revision) + |> TranslationScope.from_project(project_id) end end diff --git a/lib/graphql/resolvers/version.ex b/lib/graphql/resolvers/version.ex index 8b7911287..4c4553e08 100644 --- a/lib/graphql/resolvers/version.ex +++ b/lib/graphql/resolvers/version.ex @@ -13,12 +13,17 @@ defmodule Accent.GraphQL.Resolvers.Version do @typep version_operation :: {:ok, %{version: Version.t() | nil, errors: [String.t()] | nil}} - @spec create(Project.t(), %{name: String.t(), tag: String.t()}, GraphQLContext.t()) :: version_operation - def create(project, %{name: name, tag: tag}, info) do + @spec create( + Project.t(), + %{name: String.t(), tag: String.t(), copy_on_update_translation: boolean()}, + GraphQLContext.t() + ) :: version_operation + def create(project, args, info) do %Context{} |> Context.assign(:project, project) - |> Context.assign(:name, name) - |> Context.assign(:tag, tag) + |> Context.assign(:name, args[:name]) + |> Context.assign(:tag, args[:tag]) + |> Context.assign(:copy_on_update_translation, args[:copy_on_update_translation]) |> Context.assign(:user_id, info.context[:conn].assigns[:current_user].id) |> NewVersionBuilder.build() |> NewVersionPersister.persist() @@ -31,10 +36,18 @@ defmodule Accent.GraphQL.Resolvers.Version do end end - @spec update(Version.t(), %{name: String.t(), tag: String.t()}, GraphQLContext.t()) :: version_operation + @spec update( + Version.t(), + %{name: String.t(), tag: String.t(), copy_on_update_translation: boolean()}, + GraphQLContext.t() + ) :: version_operation def update(version, args, _info) do version - |> Version.changeset(%{name: args[:name], tag: args[:tag]}) + |> Version.changeset(%{ + name: args[:name], + tag: args[:tag], + copy_on_update_translation: args[:copy_on_update_translation] || false + }) |> Repo.update() |> case do {:ok, version} -> diff --git a/lib/graphql/schema.ex b/lib/graphql/schema.ex index 86f1e2a5c..b298b0c7f 100644 --- a/lib/graphql/schema.ex +++ b/lib/graphql/schema.ex @@ -30,10 +30,12 @@ defmodule Accent.GraphQL.Schema do import_types(Accent.GraphQL.Types.MutationResult) import_types(Accent.GraphQL.Types.Lint) + @version Application.compile_env!(:accent, :version) + object :application do field(:version, :string, resolve: fn _, _ -> - {:ok, "1.2.3"} + {:ok, @version} end ) end diff --git a/lib/graphql/types/integration.ex b/lib/graphql/types/integration.ex index 075a7f186..2b1606291 100644 --- a/lib/graphql/types/integration.ex +++ b/lib/graphql/types/integration.ex @@ -7,6 +7,7 @@ defmodule Accent.GraphQL.Types.Integration do value(:discord, as: "discord") value(:github, as: "github") value(:azure_storage_container, as: "azure_storage_container") + value(:aws_s3, as: "aws_s3") end enum :project_integration_event do @@ -15,6 +16,8 @@ defmodule Accent.GraphQL.Types.Integration do value(:complete_review, as: "complete_review") value(:create_collaborator, as: "create_collaborator") value(:create_comment, as: "create_comment") + value(:integration_execute_azure_storage_container, as: "integration_execute_azure_storage_container") + value(:integration_execute_aws_s3, as: "integration_execute_aws_s3") end interface :project_integration do @@ -26,6 +29,7 @@ defmodule Accent.GraphQL.Types.Integration do %{service: "slack"}, _ -> :project_integration_slack %{service: "github"}, _ -> :project_integration_github %{service: "azure_storage_container"}, _ -> :project_integration_azure_storage_container + %{service: "aws_s3"}, _ -> :project_integration_aws_s3 end) end @@ -60,7 +64,16 @@ defmodule Accent.GraphQL.Types.Integration do field(:id, non_null(:id)) field(:service, non_null(:project_integration_service)) field(:last_executed_at, :datetime) - field(:data, non_null(:project_integration_azure_data)) + field(:data, non_null(:project_integration_azure_storage_container_data)) + + interfaces([:project_integration]) + end + + object :project_integration_aws_s3 do + field(:id, non_null(:id)) + field(:service, non_null(:project_integration_service)) + field(:last_executed_at, :datetime) + field(:data, non_null(:project_integration_aws_s3_data)) interfaces([:project_integration]) end @@ -74,7 +87,7 @@ defmodule Accent.GraphQL.Types.Integration do field(:id, non_null(:id)) end - object :project_integration_azure_data do + object :project_integration_azure_storage_container_data do field(:id, non_null(:id)) field(:sas_base_url, non_null(:string), @@ -85,4 +98,12 @@ defmodule Accent.GraphQL.Types.Integration do end ) end + + object :project_integration_aws_s3_data do + field(:id, non_null(:id)) + field(:bucket, non_null(:string), resolve: fn data, _, _ -> {:ok, data.aws_s3_bucket} end) + field(:path_prefix, non_null(:string), resolve: fn data, _, _ -> {:ok, data.aws_s3_path_prefix} end) + field(:region, non_null(:string), resolve: fn data, _, _ -> {:ok, data.aws_s3_region} end) + field(:access_key_id, non_null(:string), resolve: fn data, _, _ -> {:ok, data.aws_s3_access_key_id} end) + end end diff --git a/lib/graphql/types/project.ex b/lib/graphql/types/project.ex index 4184647d3..2e866bc28 100644 --- a/lib/graphql/types/project.ex +++ b/lib/graphql/types/project.ex @@ -6,6 +6,8 @@ defmodule Accent.GraphQL.Types.Project do import Accent.GraphQL.Helpers.Authorization import Accent.GraphQL.Helpers.Fields + alias Accent.GraphQL.Resolvers.Translation, as: TranslationResolver + object :projects do field(:meta, non_null(:pagination_meta)) field(:entries, list_of(:project)) @@ -27,6 +29,8 @@ defmodule Accent.GraphQL.Types.Project do object :prompt_config do field(:provider, non_null(:string)) + field(:use_platform, non_null(:boolean)) + field(:use_config_key, non_null(:boolean)) end object :project do @@ -48,7 +52,7 @@ defmodule Accent.GraphQL.Types.Project do %{ provider: project.machine_translations_config["provider"], enabled_actions: project.machine_translations_config["enabled_actions"] || [], - use_platform: project.machine_translations_config["use_platform"], + use_platform: project.machine_translations_config["use_platform"] || false, use_config_key: not is_nil(project.machine_translations_config["config"]["key"]) }} else @@ -62,7 +66,9 @@ defmodule Accent.GraphQL.Types.Project do if project.prompt_config do {:ok, %{ - provider: project.prompt_config["provider"] + provider: project.prompt_config["provider"], + use_platform: project.prompt_config["use_platform"] || false, + use_config_key: not is_nil(project.prompt_config["config"]["key"]) }} else {:ok, nil} @@ -119,7 +125,10 @@ defmodule Accent.GraphQL.Types.Project do end field(:language, :language, resolve: dataloader(Accent.Language)) - field(:integrations, list_of(:project_integration), resolve: dataloader(Accent.Integration)) + + field :integrations, list_of(non_null(:project_integration)) do + resolve(project_authorize(:index_project_integrations, &Accent.GraphQL.Resolvers.Integration.list_project/3)) + end field :document, :document do arg(:id, non_null(:id)) @@ -135,13 +144,14 @@ defmodule Accent.GraphQL.Types.Project do resolve(project_authorize(:index_documents, &Accent.GraphQL.Resolvers.Document.list_project/3)) end - field :translations, :translations do + field :grouped_translations, :grouped_translations do arg(:page, :integer) arg(:page_size, :integer) - arg(:order, :string) arg(:document, :id) arg(:version, :id) + arg(:related_revisions, list_of(non_null(:id))) arg(:query, :string) + arg(:is_translated, :boolean) arg(:is_conflicted, :boolean) arg(:is_text_empty, :boolean) arg(:is_text_not_empty, :boolean) @@ -151,11 +161,28 @@ defmodule Accent.GraphQL.Types.Project do resolve( project_authorize( :index_translations, - &Accent.GraphQL.Resolvers.Translation.list_project/3 + &TranslationResolver.list_grouped_project/3 ) ) end + field :translations, :translations do + arg(:page, :integer) + arg(:page_size, :integer) + arg(:order, :string) + arg(:document, :id) + arg(:version, :id) + arg(:query, :string) + arg(:is_translated, :boolean) + arg(:is_conflicted, :boolean) + arg(:is_text_empty, :boolean) + arg(:is_text_not_empty, :boolean) + arg(:is_added_last_sync, :boolean) + arg(:is_commented_on, :boolean) + + resolve(project_authorize(:index_translations, &TranslationResolver.list_project/3)) + end + field :activities, :activities do arg(:page, :integer) arg(:page_size, :integer) @@ -182,7 +209,7 @@ defmodule Accent.GraphQL.Types.Project do field :translation, :translation do arg(:id, non_null(:id)) - resolve(project_authorize(:show_translation, &Accent.GraphQL.Resolvers.Translation.show_project/3)) + resolve(project_authorize(:show_translation, &TranslationResolver.show_project/3)) end field :activity, :activity do @@ -200,6 +227,7 @@ defmodule Accent.GraphQL.Types.Project do field :revisions, list_of(:revision) do arg(:version_id, :id) + resolve(project_authorize(:index_revisions, &Accent.GraphQL.Resolvers.Revision.list_project/3)) end diff --git a/lib/graphql/types/translation.ex b/lib/graphql/types/translation.ex index 04f884dac..e3a216e65 100644 --- a/lib/graphql/types/translation.ex +++ b/lib/graphql/types/translation.ex @@ -18,6 +18,15 @@ defmodule Accent.GraphQL.Types.Translation do value(:float, as: "float") end + object :grouped_translation do + field(:key, non_null(:string)) + field(:document, non_null(:document), resolve: &Accent.GraphQL.Resolvers.Translation.batch_document/3) + + field(:translations, non_null(list_of(non_null(:translation))), + resolve: &Accent.GraphQL.Resolvers.Translation.batch_translation_ids/3 + ) + end + object :translation do field(:id, non_null(:id)) field(:key, non_null(:string), resolve: &Accent.GraphQL.Resolvers.Translation.key/3) @@ -119,4 +128,10 @@ defmodule Accent.GraphQL.Types.Translation do field(:meta, :pagination_meta) field(:entries, list_of(:translation)) end + + object :grouped_translations do + field(:meta, :pagination_meta) + field(:entries, list_of(:grouped_translation)) + field(:revisions, non_null(list_of(non_null(:revision)))) + end end diff --git a/lib/graphql/types/version.ex b/lib/graphql/types/version.ex index 37a34e03a..0f554dea6 100644 --- a/lib/graphql/types/version.ex +++ b/lib/graphql/types/version.ex @@ -10,6 +10,7 @@ defmodule Accent.GraphQL.Types.Version do field(:tag, non_null(:string)) field(:project, :project, resolve: dataloader(Accent.Project)) field(:user, :user, resolve: dataloader(Accent.User)) + field(:copy_on_update_translation, non_null(:boolean)) field(:inserted_at, non_null(:datetime)) end diff --git a/lib/hook/event.ex b/lib/hook/event.ex new file mode 100644 index 000000000..319190002 --- /dev/null +++ b/lib/hook/event.ex @@ -0,0 +1,6 @@ +defmodule Accent.Hook.Event do + @moduledoc false + @callback name :: String.t() + @callback triggered?(args :: map(), new_state :: map()) :: boolean() + @callback payload(args :: map(), new_state :: map()) :: map() +end diff --git a/lib/hook/events.ex b/lib/hook/events.ex new file mode 100644 index 000000000..2d930c473 --- /dev/null +++ b/lib/hook/events.ex @@ -0,0 +1,24 @@ +defmodule Accent.Hook.Events do + @moduledoc false + path_wildcard = Path.join(Path.dirname(__ENV__.file), "events/*.ex") + paths = Path.wildcard(path_wildcard) + paths_hash = :erlang.md5(paths) + + @callback registered_events :: [String.t()] | :all + + event_modules = + for name <- paths do + Module.concat(__MODULE__, Phoenix.Naming.camelize(Path.basename(name, ".ex"))) + end + + @event_modules event_modules + + def available do + @event_modules + end + + def __mix_recompile__? do + paths = Path.wildcard(unquote(path_wildcard)) + :erlang.md5(paths) != unquote(paths_hash) + end +end diff --git a/lib/hook/events/add_translations.ex b/lib/hook/events/add_translations.ex new file mode 100644 index 000000000..21d6f0706 --- /dev/null +++ b/lib/hook/events/add_translations.ex @@ -0,0 +1,23 @@ +defmodule Accent.Hook.Events.AddTranslations do + @moduledoc false + @behaviour Accent.Hook.Event + + alias Movement.Persisters.ProjectHookWorker + + @impl true + def name do + "add_translations" + end + + @impl true + def triggered?(%ProjectHookWorker.Args{} = args, _new_state) do + args.batch_action === "merge" and args.operations_count > 0 + end + + @impl true + def payload(%ProjectHookWorker.Args{} = args, _new_state) do + %{ + language_name: args.revision.language.name + } + end +end diff --git a/lib/hook/events/complete_review.ex b/lib/hook/events/complete_review.ex new file mode 100644 index 000000000..7b31ded76 --- /dev/null +++ b/lib/hook/events/complete_review.ex @@ -0,0 +1,24 @@ +defmodule Accent.Hook.Events.CompleteReview do + @moduledoc false + @behaviour Accent.Hook.Event + + alias Movement.Persisters.ProjectHookWorker + + @impl true + def name do + "complete_review" + end + + @impl true + def triggered?(%ProjectHookWorker.Args{} = args, %ProjectHookWorker.ProjectState{} = project_state) do + args.previous_project_state.reviewed_count !== args.previous_project_state.translations_count and + project_state.reviewed_count === project_state.translations_count + end + + @impl true + def payload(_args, %ProjectHookWorker.ProjectState{} = project_state) do + %{ + translations_count: project_state.translations_count + } + end +end diff --git a/lib/hook/events/new_conflicts.ex b/lib/hook/events/new_conflicts.ex new file mode 100644 index 000000000..44d6c2a28 --- /dev/null +++ b/lib/hook/events/new_conflicts.ex @@ -0,0 +1,25 @@ +defmodule Accent.Hook.Events.NewConflicts do + @moduledoc false + @behaviour Accent.Hook.Event + + alias Movement.Persisters.ProjectHookWorker + + @impl true + def name do + "new_conflicts" + end + + @impl true + def triggered?(%ProjectHookWorker.Args{} = args, %ProjectHookWorker.ProjectState{} = project_state) do + args.previous_project_state.conflicts_count < project_state.conflicts_count + end + + @impl true + def payload(%ProjectHookWorker.Args{} = args, %ProjectHookWorker.ProjectState{} = project_state) do + %{ + reviewed_count: project_state.reviewed_count, + translations_count: project_state.translations_count, + new_conflicts_count: project_state.conflicts_count - args.previous_project_state.conflicts_count + } + end +end diff --git a/lib/hook/events/sync.ex b/lib/hook/events/sync.ex new file mode 100644 index 000000000..4ec7c58d8 --- /dev/null +++ b/lib/hook/events/sync.ex @@ -0,0 +1,24 @@ +defmodule Accent.Hook.Events.Sync do + @moduledoc false + @behaviour Accent.Hook.Event + + alias Movement.Persisters.ProjectHookWorker + + @impl true + def name do + "sync" + end + + @impl true + def triggered?(%ProjectHookWorker.Args{} = args, _new_state) do + args.batch_action === "sync" and args.operations_count > 0 + end + + @impl true + def payload(%ProjectHookWorker.Args{} = args, _new_state) do + %{ + batch_operation_stats: args.batch_operation.stats, + document_path: args.document.path + } + end +end diff --git a/lib/hook/hook.ex b/lib/hook/hook.ex index a709cc9a1..546be80c5 100644 --- a/lib/hook/hook.ex +++ b/lib/hook/hook.ex @@ -1,19 +1,17 @@ defmodule Accent.Hook do @moduledoc false - def outbound(context), do: run(outbounds_modules(), context) + @outbounds_modules Application.compile_env!(:accent, __MODULE__)[:outbounds] - defp run(modules, context) do + def outbound(context) do jobs = - Enum.reduce(modules, [], fn {module, opts}, acc -> - if context.event in Keyword.fetch!(opts, :events), - do: [module.new(context) | acc], - else: acc + Enum.flat_map(@outbounds_modules, fn module -> + events = module.registered_events() + + if events === :all or context.event in events, + do: [module.new(context)], + else: [] end) Oban.insert_all(jobs) end - - defp outbounds_modules do - Application.get_env(:accent, __MODULE__)[:outbounds] - end end diff --git a/lib/hook/outbounds/discord.ex b/lib/hook/outbounds/discord.ex new file mode 100644 index 000000000..139bdd11b --- /dev/null +++ b/lib/hook/outbounds/discord.ex @@ -0,0 +1,50 @@ +defmodule Accent.Hook.Outbounds.Discord do + @moduledoc false + @behaviour Accent.Hook.Events + + use Oban.Worker, queue: :hook + + defmodule Templates do + @moduledoc false + import Accent.Hook.Outbounds.Helpers.StringTemplate + + deftemplate(:new_conflicts, """ + **<%= @user %>** just added *<%= @new_conflicts_count %> strings* to review. + The project is currently **<%= Float.round(@reviewed_count / @translations_count * 100, 2) %>%** reviewed (<%= @reviewed_count %>/<%= @translations_count %>) + """) + + deftemplate(:sync, """ + **<%= @user %>** just synced a file: *<%= @document_path %>* + + **Stats:**<%= for %{"action" => action, "count" => count} <- @stats do %> + <%= Phoenix.Naming.humanize(action) %>: *<%= count %>*<% end %> + """) + + deftemplate(:complete_review, """ + **<%= @user %>** just finished reviewing all strings! + The project currently has <%= @translations_count %> reviewed translations. + """) + + deftemplate(:integration_execute_azure_storage_container, """ + **<%= @user %>** just uploaded all *<%= @version_tag %>* files to Azure Container Storage. + <%= for %{"name" => document_name, "url" => url} <- @document_urls do %> + [<%= document_name %>](<%= url %>) + <% end %> + """) + end + + @impl Accent.Hook.Events + def registered_events do + ~w(sync complete_review new_conflicts integration_execute_azure_storage_container) + end + + @impl Oban.Worker + def perform(%Oban.Job{args: args}) do + context = Accent.Hook.Context.from_worker(args) + + Accent.Hook.Outbounds.Helpers.PostURL.perform("discord", context, + http_body: &%{content: &1}, + templates: Templates + ) + end +end diff --git a/lib/hook/outbounds/discord/discord.ex b/lib/hook/outbounds/discord/discord.ex deleted file mode 100644 index 396716d8e..000000000 --- a/lib/hook/outbounds/discord/discord.ex +++ /dev/null @@ -1,15 +0,0 @@ -defmodule Accent.Hook.Outbounds.Discord do - @moduledoc false - use Oban.Worker, queue: :hook - - @impl Oban.Worker - def perform(%Oban.Job{args: args}) do - context = Accent.Hook.Context.from_worker(args) - http_body = fn content -> %{content: content} end - - Accent.Hook.Outbounds.PostURL.perform("discord", context, - http_body: http_body, - templates: Hook.Outbounds.Discord.Templates - ) - end -end diff --git a/lib/hook/outbounds/discord/templates.ex b/lib/hook/outbounds/discord/templates.ex deleted file mode 100644 index 54976a895..000000000 --- a/lib/hook/outbounds/discord/templates.ex +++ /dev/null @@ -1,23 +0,0 @@ -defmodule Hook.Outbounds.Discord.Templates do - @moduledoc false - require EEx - - @sync_template """ - **<%= @user %>** just synced a file: *<%= @document_path %>* - - **Stats:**<%= for %{"action" => action, "count" => count} <- @stats do %> - <%= Phoenix.Naming.humanize(action) %>: *<%= count %>*<% end %> - """ - @new_conflicts_template """ - **<%= @user %>** just added *<%= @new_conflicts_count %> strings* to review. - The project is currently **<%= Float.round(@reviewed_count / @translations_count * 100, 2) %>** reviewed (<%= @reviewed_count %>/<%= @translations_count %>) - """ - @complete_review_template """ - **<%= @user %>** just finished reviewing all strings! - The project currently has <%= @translations_count %> reviewed translations. - """ - - EEx.function_from_string(:def, :sync, @sync_template, [:assigns], trim: true) - EEx.function_from_string(:def, :new_conflicts, @new_conflicts_template, [:assigns], trim: true) - EEx.function_from_string(:def, :complete_review, @complete_review_template, [:assigns], trim: true) -end diff --git a/lib/hook/outbounds/email.ex b/lib/hook/outbounds/email.ex index d1325039c..4518d0838 100644 --- a/lib/hook/outbounds/email.ex +++ b/lib/hook/outbounds/email.ex @@ -1,5 +1,7 @@ defmodule Accent.Hook.Outbounds.Email do @moduledoc false + @behaviour Accent.Hook.Events + use Oban.Worker, queue: :hook alias Accent.CreateCommentEmail @@ -7,6 +9,11 @@ defmodule Accent.Hook.Outbounds.Email do alias Accent.Repo alias Accent.Translation + @impl Accent.Hook.Events + def registered_events do + ~w(create_collaborator create_comment) + end + @impl Oban.Worker def perform(%Oban.Job{args: args}) do context = Accent.Hook.Context.from_worker(args) diff --git a/lib/hook/outbounds/post_url.ex b/lib/hook/outbounds/helpers/post_url.ex similarity index 61% rename from lib/hook/outbounds/post_url.ex rename to lib/hook/outbounds/helpers/post_url.ex index 5907245c6..3013d2395 100644 --- a/lib/hook/outbounds/post_url.ex +++ b/lib/hook/outbounds/helpers/post_url.ex @@ -1,4 +1,4 @@ -defmodule Accent.Hook.Outbounds.PostURL do +defmodule Accent.Hook.Outbounds.Helpers.PostURL do @moduledoc false import Ecto.Query, only: [where: 2] @@ -58,27 +58,35 @@ defmodule Accent.Hook.Outbounds.PostURL do defp formatted_diff(diff) when diff > 1000, do: [diff |> div(1000) |> Integer.to_string(), "ms"] defp formatted_diff(diff), do: [Integer.to_string(diff), "µs"] - defp build_content(templates, %{event: "sync", user: user, payload: payload}) do + defp build_content(templates, %{event: "sync"} = context) do templates.sync(%{ - user: User.name_with_fallback(user), - document_path: payload["document_path"], - stats: payload["batch_operation_stats"] + user: User.name_with_fallback(context.user), + document_path: context.payload["document_path"], + stats: context.payload["batch_operation_stats"] }) end - defp build_content(templates, %{event: "new_conflicts", user: user, payload: payload}) do + defp build_content(templates, %{event: "new_conflicts"} = context) do templates.new_conflicts(%{ - user: User.name_with_fallback(user), - reviewed_count: payload["reviewed_count"], - new_conflicts_count: payload["new_conflicts_count"], - translations_count: payload["translations_count"] + user: User.name_with_fallback(context.user), + reviewed_count: context.payload["reviewed_count"], + new_conflicts_count: context.payload["new_conflicts_count"], + translations_count: context.payload["translations_count"] }) end - defp build_content(templates, %{event: "complete_review", user: user, payload: payload}) do + defp build_content(templates, %{event: "complete_review"} = context) do templates.complete_review(%{ - user: User.name_with_fallback(user), - translations_count: payload["translations_count"] + user: User.name_with_fallback(context.user), + translations_count: context.payload["translations_count"] + }) + end + + defp build_content(templates, %{event: "integration_execute_azure_storage_container"} = context) do + templates.integration_execute_azure_storage_container(%{ + user: User.name_with_fallback(context.user), + version_tag: context.payload["version_tag"], + document_urls: context.payload["document_urls"] }) end end diff --git a/lib/hook/outbounds/helpers/string_template.ex b/lib/hook/outbounds/helpers/string_template.ex new file mode 100644 index 000000000..36b7823c7 --- /dev/null +++ b/lib/hook/outbounds/helpers/string_template.ex @@ -0,0 +1,11 @@ +defmodule Accent.Hook.Outbounds.Helpers.StringTemplate do + @moduledoc false + + defmacro deftemplate(name, template) do + quote do + require EEx + + EEx.function_from_string(:def, unquote(name), unquote(template), [:assigns], trim: true) + end + end +end diff --git a/lib/hook/outbounds/slack.ex b/lib/hook/outbounds/slack.ex new file mode 100644 index 000000000..84e5b1182 --- /dev/null +++ b/lib/hook/outbounds/slack.ex @@ -0,0 +1,50 @@ +defmodule Accent.Hook.Outbounds.Slack do + @moduledoc false + @behaviour Accent.Hook.Events + + use Oban.Worker, queue: :hook + + defmodule Templates do + @moduledoc false + import Accent.Hook.Outbounds.Helpers.StringTemplate + + deftemplate(:new_conflicts, """ + *<%= @user %>* just added _<%= @new_conflicts_count %> strings_ to review. + The project is currently *<%= Float.round(@reviewed_count / @translations_count * 100, 2) %>%* reviewed (<%= @reviewed_count %>/<%= @translations_count %>) + """) + + deftemplate(:sync, """ + *<%= @user %>* just synced a file: _<%= @document_path %>_ + + *Stats:*<%= for %{"action" => action, "count" => count} <- @stats do %> + <%= Phoenix.Naming.humanize(action) %>: _<%= count %>_<% end %> + """) + + deftemplate(:complete_review, """ + *<%= @user %>* just finished reviewing all strings! + The project currently has <%= @translations_count %> reviewed translations. + """) + + deftemplate(:integration_execute_azure_storage_container, """ + *<%= @user %>* just uploaded all _<%= @version_tag %>_ files to Azure Container Storage. + <%= for %{"name" => document_name, "url" => url} <- @document_urls do %> + [<%= document_name %>](<%= url %>) + <% end %> + """) + end + + @impl Accent.Hook.Events + def registered_events do + ~w(sync complete_review new_conflicts integration_execute_azure_storage_container) + end + + @impl Oban.Worker + def perform(%Oban.Job{args: args}) do + context = Accent.Hook.Context.from_worker(args) + + Accent.Hook.Outbounds.Helpers.PostURL.perform("slack", context, + http_body: &%{text: &1}, + templates: Templates + ) + end +end diff --git a/lib/hook/outbounds/slack/slack.ex b/lib/hook/outbounds/slack/slack.ex deleted file mode 100644 index 03da43079..000000000 --- a/lib/hook/outbounds/slack/slack.ex +++ /dev/null @@ -1,15 +0,0 @@ -defmodule Accent.Hook.Outbounds.Slack do - @moduledoc false - use Oban.Worker, queue: :hook - - @impl Oban.Worker - def perform(%Oban.Job{args: args}) do - context = Accent.Hook.Context.from_worker(args) - http_body = fn content -> %{text: content} end - - Accent.Hook.Outbounds.PostURL.perform("slack", context, - http_body: http_body, - templates: Hook.Outbounds.Slack.Templates - ) - end -end diff --git a/lib/hook/outbounds/slack/templates.ex b/lib/hook/outbounds/slack/templates.ex deleted file mode 100644 index a21a927cd..000000000 --- a/lib/hook/outbounds/slack/templates.ex +++ /dev/null @@ -1,23 +0,0 @@ -defmodule Hook.Outbounds.Slack.Templates do - @moduledoc false - require EEx - - @sync_template """ - *<%= @user %>* just synced a file: _<%= @document_path %>_ - - *Stats:*<%= for %{"action" => action, "count" => count} <- @stats do %> - <%= Phoenix.Naming.humanize(action) %>: _<%= count %>_<% end %> - """ - @new_conflicts_template """ - *<%= @user %>* just added _<%= @new_conflicts_count %> strings_ to review. - The project is currently *<%= Float.round(@reviewed_count / @translations_count * 100, 2) %>* reviewed (<%= @reviewed_count %>/<%= @translations_count %>) - """ - @complete_review_template """ - *<%= @user %>* just finished reviewing all strings! - The project currently has <%= @translations_count %> reviewed translations. - """ - - EEx.function_from_string(:def, :sync, @sync_template, [:assigns], trim: true) - EEx.function_from_string(:def, :new_conflicts, @new_conflicts_template, [:assigns], trim: true) - EEx.function_from_string(:def, :complete_review, @complete_review_template, [:assigns], trim: true) -end diff --git a/lib/hook/outbounds/websocket.ex b/lib/hook/outbounds/websocket.ex index f70c76883..749bb5030 100644 --- a/lib/hook/outbounds/websocket.ex +++ b/lib/hook/outbounds/websocket.ex @@ -1,7 +1,14 @@ defmodule Accent.Hook.Outbounds.Websocket do @moduledoc false + @behaviour Accent.Hook.Events + use Oban.Worker, queue: :hook + @impl Accent.Hook.Events + def registered_events do + ~w(sync create_collaborator create_comment complete_review new_conflicts) + end + @impl Oban.Worker def perform(%Oban.Job{args: args}) do args diff --git a/lib/langue/formatter/json/parser.ex b/lib/langue/formatter/json/parser.ex index a7d751d6a..4e63f796a 100644 --- a/lib/langue/formatter/json/parser.ex +++ b/lib/langue/formatter/json/parser.ex @@ -19,5 +19,6 @@ defmodule Langue.Formatter.Json.Parser do |> :jsone.decode(object_format: :tuple) |> elem(0) |> NestedParserHelper.parse() + |> Enum.uniq_by(& &1.key) end end diff --git a/lib/langue/utils/parser.ex b/lib/langue/utils/parser.ex index e65cb4de4..0456a1948 100644 --- a/lib/langue/utils/parser.ex +++ b/lib/langue/utils/parser.ex @@ -3,7 +3,5 @@ defmodule Langue.Formatter.Parser do alias Langue.Formatter.ParserResult, as: Output alias Langue.Formatter.SerializerResult, as: Input - @type result :: {:ok, Output.t()} | {:error, any()} - - @callback parse(Input.t()) :: result + @callback parse(Input.t()) :: Output.t() end diff --git a/lib/langue/utils/serializer.ex b/lib/langue/utils/serializer.ex index 88a285303..4328d173a 100644 --- a/lib/langue/utils/serializer.ex +++ b/lib/langue/utils/serializer.ex @@ -3,7 +3,5 @@ defmodule Langue.Formatter.Serializer do alias Langue.Formatter.ParserResult, as: Input alias Langue.Formatter.SerializerResult, as: Output - @type result :: {:ok, Output.t()} | {:error, any()} - - @callback serialize(Input.t()) :: result + @callback serialize(Input.t()) :: Output.t() end diff --git a/lib/machine_translations/machine_translations.ex b/lib/machine_translations/machine_translations.ex index f162da6f7..ff63753cb 100644 --- a/lib/machine_translations/machine_translations.ex +++ b/lib/machine_translations/machine_translations.ex @@ -87,15 +87,15 @@ defmodule Accent.MachineTranslations do defp provider_from_config(nil), do: %Provider.NotImplemented{} - defp provider_from_config(machine_translations_config) do + defp provider_from_config(config) do struct_module = - case machine_translations_config["provider"] do + case config["provider"] do "google_translate" -> Provider.GoogleTranslate "deepl" -> Provider.Deepl _ -> Provider.NotImplemented end - struct!(struct_module, config: fetch_config(machine_translations_config)) + struct!(struct_module, config: fetch_config(config)) end defp fetch_config(%{"provider" => provider, "use_platform" => true}) do diff --git a/lib/machine_translations/provider/deepl.ex b/lib/machine_translations/provider/deepl.ex index ea1e45bd4..8e5399643 100644 --- a/lib/machine_translations/provider/deepl.ex +++ b/lib/machine_translations/provider/deepl.ex @@ -46,7 +46,7 @@ defmodule Accent.MachineTranslations.Provider.Deepl do def translate(provider, contents, source, target) do with {:ok, {source, target}} <- Accent.MachineTranslations.map_source_and_target(source, target, @supported_languages), - params = %{text: contents, source_lang: String.upcase(source), target_lang: String.upcase(target)}, + params = %{text: contents, source_lang: source && String.upcase(source), target_lang: String.upcase(target)}, {:ok, %{body: %{"translations" => translations}}} <- Tesla.post(client(provider.config["key"]), "translate", params) do {:ok, Enum.map(translations, &%TranslatedText{text: &1["text"]})} diff --git a/lib/machine_translations/provider/google_translate.ex b/lib/machine_translations/provider/google_translate.ex index a77f2e771..9aec72a47 100644 --- a/lib/machine_translations/provider/google_translate.ex +++ b/lib/machine_translations/provider/google_translate.ex @@ -137,7 +137,7 @@ defmodule Accent.MachineTranslations.Provider.GoogleTranslate do ), params = %{ contents: contents_with_no_translate, - mimeType: "text/html", + mimeType: "text/plain", sourceLanguageCode: source, targetLanguageCode: target }, @@ -166,7 +166,7 @@ defmodule Accent.MachineTranslations.Provider.GoogleTranslate do end defp unmark_no_translate(value) do - String.replace(value, ~r/([^<]+)<\/span>/, "\\1") + String.replace(value, ~r/([^<]+)<\/span>/i, "\\1") end defmodule Auth do diff --git a/lib/movement/builders/translation_update.ex b/lib/movement/builders/translation_update.ex index e78874007..fc31a24be 100644 --- a/lib/movement/builders/translation_update.ex +++ b/lib/movement/builders/translation_update.ex @@ -2,6 +2,7 @@ defmodule Movement.Builders.TranslationUpdate do @moduledoc false @behaviour Movement.Builder + alias Accent.Version alias Movement.Mappers.Operation, as: OperationMapper @action "update" @@ -17,6 +18,25 @@ defmodule Movement.Builders.TranslationUpdate do value_type = Movement.Mappers.ValueType.from_translation_new_value(translation, text) operation = OperationMapper.map(@action, translation, %{text: text, value_type: value_type}) - %{context | operations: Enum.concat(operations, [operation])} + copy_version_operation = + if copy_translation_update_to_latest_version?(translation) do + copy_translation_update_to_latest_version(translation, text) + end + + %{context | operations: Enum.concat(operations, [operation] ++ List.wrap(copy_version_operation))} + end + + defp copy_translation_update_to_latest_version(translation, text) do + source_translation = Accent.Repo.one!(Ecto.assoc(translation, :source_translation)) + value_type = Movement.Mappers.ValueType.from_translation_new_value(source_translation, text) + OperationMapper.map(@action, source_translation, %{text: text, value_type: value_type}) + end + + defp copy_translation_update_to_latest_version?(translation) do + if translation.version_id do + version = Accent.Repo.get(Version, translation.version_id) + + version.copy_on_update_translation + end end end diff --git a/lib/movement/context.ex b/lib/movement/context.ex index 91c5e7340..995002751 100644 --- a/lib/movement/context.ex +++ b/lib/movement/context.ex @@ -1,6 +1,6 @@ defmodule Movement.Context do @moduledoc false - defstruct entries: [], operations: [], assigns: %{options: []}, render: "" + defstruct entries: [], operations: [], assigns: %{batch_operation: nil, options: []}, render: "" @type t :: %__MODULE__{} diff --git a/lib/movement/migration/translation.ex b/lib/movement/migration/translation.ex index 78486b9c7..e18f250c2 100644 --- a/lib/movement/migration/translation.ex +++ b/lib/movement/migration/translation.ex @@ -56,7 +56,7 @@ defmodule Movement.Migration.Translation do locked: operation.locked, file_index: operation.file_index, file_comment: operation.file_comment, - removed: operation.previous_translation && operation.previous_translation.removed, + removed: (operation.previous_translation && operation.previous_translation.removed) || false, translated: is_nil(operation.translation_id), revision_id: operation.revision_id, document_id: operation.document_id, @@ -82,11 +82,11 @@ defmodule Movement.Migration.Translation do proposed_text: operation.text, corrected_text: operation.text, translated: (operation.previous_translation && operation.previous_translation.translated) || false, - conflicted: operation.previous_translation && operation.previous_translation.conflicted, + conflicted: (operation.previous_translation && operation.previous_translation.conflicted) || false, value_type: operation.value_type, file_index: operation.file_index, file_comment: operation.file_comment, - removed: operation.previous_translation && operation.previous_translation.removed, + removed: (operation.previous_translation && operation.previous_translation.removed) || false, revision_id: operation.revision_id, document_id: operation.document_id, version_id: operation.version_id, diff --git a/lib/movement/persisters/base.ex b/lib/movement/persisters/base.ex index 910a6b8c3..bb8cf6963 100644 --- a/lib/movement/persisters/base.ex +++ b/lib/movement/persisters/base.ex @@ -4,7 +4,7 @@ defmodule Movement.Persisters.Base do alias Accent.Repo alias Movement.Mappers.OperationsStats, as: StatMapper alias Movement.Migrator - alias Movement.Persisters.ProjectStateChangeWorker + alias Movement.Persisters.ProjectHookWorker require Ecto.Query @@ -17,7 +17,8 @@ defmodule Movement.Persisters.Base do @spec execute(Movement.Context.t()) :: {Movement.Context.t(), [Operation.t()]} def execute(%Movement.Context{operations: []} = context), do: {context, []} - def execute(%Movement.Context{assigns: %{batch_action: action} = assigns} = context) when is_binary(action) do + def execute(%Movement.Context{assigns: %{batch_action: action, batch_operation: nil} = assigns} = context) + when is_binary(action) do stats = StatMapper.map(context.operations) batch_operation = @@ -34,12 +35,13 @@ defmodule Movement.Persisters.Base do context |> Movement.Context.assign(:batch_operation, batch_operation) - |> Movement.Context.assign(:batch_action, nil) |> execute() end def execute(context) do - project_state_change_context = %{ + project_context = %{ + batch_action: context.assigns[:batch_action], + operations_count: Enum.count(context.operations), project_id: context.assigns[:project] && context.assigns.project.id, document_id: context.assigns[:document] && context.assigns.document.id, master_revision_id: context.assigns[:master_revision] && context.assigns.master_revision.id, @@ -47,15 +49,14 @@ defmodule Movement.Persisters.Base do version_id: context.assigns[:version] && context.assigns.version.id, batch_operation_id: context.assigns[:batch_operation] && context.assigns.batch_operation.id, user_id: context.assigns[:user_id], - previous_project_state: ProjectStateChangeWorker.get_project_state(context.assigns[:project]) + previous_project_state: ProjectHookWorker.get_project_state(context.assigns[:project]) } context |> persist_operations() |> migrate_up_operations() |> tap(fn _ -> - project_state_change_context.previous_project_state && - Oban.insert(ProjectStateChangeWorker.new(project_state_change_context)) + context.assigns[:project] && Oban.insert(ProjectHookWorker.new(project_context)) end) end diff --git a/lib/movement/persisters/new_version.ex b/lib/movement/persisters/new_version.ex index 97ea01ee4..20bbd6cb6 100644 --- a/lib/movement/persisters/new_version.ex +++ b/lib/movement/persisters/new_version.ex @@ -23,7 +23,8 @@ defmodule Movement.Persisters.NewVersion do "project_id" => assigns[:project].id, "user_id" => assigns[:user_id], "name" => assigns[:name], - "tag" => assigns[:tag] + "tag" => assigns[:tag], + "copy_on_update_translation" => assigns[:copy_on_update_translation] }) |> Repo.insert() |> case do diff --git a/lib/movement/persisters/project_hook_worker.ex b/lib/movement/persisters/project_hook_worker.ex new file mode 100644 index 000000000..db07837a7 --- /dev/null +++ b/lib/movement/persisters/project_hook_worker.ex @@ -0,0 +1,93 @@ +defmodule Movement.Persisters.ProjectHookWorker do + @moduledoc false + use Oban.Worker, queue: :hook + + import Ecto.Query + + alias Accent.Hook + alias Accent.Repo + alias Accent.Scopes.Project, as: ProjectScope + + defmodule ProjectState do + @moduledoc false + @derive Jason.Encoder + defstruct translations_count: 0, reviewed_count: 0, conflicts_count: 0 + + @type t :: %__MODULE__{} + end + + defmodule Args do + @moduledoc false + defstruct previous_project_state: %ProjectState{}, + operations_count: 0, + project: nil, + document: nil, + master_revision: nil, + revision: nil, + version: nil, + batch_operation: nil, + batch_action: nil, + user: nil + + @type t :: %__MODULE__{} + end + + @impl Oban.Worker + def perform(%Oban.Job{args: args}) do + args = cast_args(args) + current_project_state = get_project_state(args.project) + + for module <- Accent.Hook.Events.available() do + if module.triggered?(args, current_project_state) do + Hook.outbound(%Hook.Context{ + event: module.name(), + project_id: args.project.id, + user_id: args.user && args.user.id, + payload: module.payload(args, current_project_state) + }) + end + end + + :ok + end + + def get_project_state(nil), do: nil + + def get_project_state(project) do + project = + Accent.Project + |> from(where: [id: ^project.id]) + |> ProjectScope.with_stats() + |> Repo.one() + + struct!(ProjectState, Map.take(project, ~w(translations_count reviewed_count conflicts_count)a)) + end + + defp cast_args(args) do + %Args{ + previous_project_state: cast_project_state(args["previous_project_state"]), + project: get_record(Accent.Project, args["project_id"]), + document: get_record(Accent.Document, args["document_id"]), + master_revision: get_record(Accent.Revision, args["master_revision_id"]), + revision: get_record(Accent.Revision, args["revision_id"]), + version: get_record(Accent.Version, args["version_id"]), + batch_operation: get_record(Accent.Operation, args["batch_operation_id"]), + operations_count: args["operations_count"], + batch_action: args["batch_action"], + user: get_record(Accent.User, args["user_id"]) + } + end + + defp cast_project_state(nil), do: nil + + defp cast_project_state(args) do + %ProjectState{ + translations_count: args["translations_count"], + reviewed_count: args["reviewed_count"], + conflicts_count: args["conflicts_count"] + } + end + + defp get_record(_schema, nil), do: nil + defp get_record(schema, id), do: Repo.get(schema, id) +end diff --git a/lib/movement/persisters/project_state_change_worker.ex b/lib/movement/persisters/project_state_change_worker.ex deleted file mode 100644 index edc9e99f4..000000000 --- a/lib/movement/persisters/project_state_change_worker.ex +++ /dev/null @@ -1,90 +0,0 @@ -defmodule Movement.Persisters.ProjectStateChangeWorker do - @moduledoc false - use Oban.Worker, queue: :hook - - import Ecto.Query - - alias Accent.Hook - alias Accent.Repo - alias Accent.Scopes.Project, as: ProjectScope - - @impl Oban.Worker - def perform(%Oban.Job{args: args}) do - args = cast_args(args) - current_project_state = get_project_state(args.project) - - if new_conflicts_to_review?(args.previous_project_state, current_project_state) do - Hook.outbound(%Hook.Context{ - event: "new_conflicts", - project_id: args.project.id, - user_id: args.user.id, - payload: %{ - reviewed_count: current_project_state.project.reviewed_count, - translations_count: current_project_state.project.translations_count, - new_conflicts_count: - current_project_state.project.conflicts_count - args.previous_project_state.project.conflicts_count - } - }) - end - - if all_reviewed?(args.previous_project_state, current_project_state) do - Hook.outbound(%Hook.Context{ - event: "complete_review", - project_id: args.project.id, - user_id: args.user.id, - payload: %{ - translations_count: current_project_state.project.translations_count - } - }) - end - - :ok - end - - defp new_conflicts_to_review?(previous_state, current_state) do - previous_state.project.conflicts_count < current_state.project.conflicts_count - end - - defp all_reviewed?(previous_state, current_state) do - previous_state.project.reviewed_count !== previous_state.project.translations_count and - current_state.project.reviewed_count === current_state.project.translations_count - end - - def get_project_state(nil), do: nil - - def get_project_state(project) do - project = - Accent.Project - |> from(where: [id: ^project.id]) - |> ProjectScope.with_stats() - |> Repo.one() - - %{project: Map.take(project, ~w(translations_count reviewed_count conflicts_count)a)} - end - - defp cast_project_state(args) do - %{ - project: %{ - translations_count: args["project"]["translations_count"], - reviewed_count: args["project"]["reviewed_count"], - conflicts_count: args["project"]["conflicts_count"] - } - } - end - - defp cast_args(args) do - %{ - previous_project_state: cast_project_state(args["previous_project_state"]), - project: get_record(Accent.Project, args["project_id"]), - document: get_record(Accent.Document, args["document_id"]), - master_revision: get_record(Accent.Revision, args["master_revision_id"]), - revision: get_record(Accent.Revision, args["revision_id"]), - version: get_record(Accent.Version, args["version_id"]), - batch_operation: get_record(Accent.Operation, args["batch_operation_id"]), - user: get_record(Accent.User, args["user_id"]) - } - end - - defp get_record(_schema, nil), do: nil - defp get_record(schema, id), do: Repo.get(schema, id) -end diff --git a/lib/prompts/prompts.ex b/lib/prompts/prompts.ex index 253c76493..6bf1fa6eb 100644 --- a/lib/prompts/prompts.ex +++ b/lib/prompts/prompts.ex @@ -17,6 +17,8 @@ defmodule Accent.Prompts do Provider.enabled?(provider) end + defp provider_from_config(nil), do: %Provider.NotImplemented{} + defp provider_from_config(config) do struct_module = case config["provider"] do @@ -24,6 +26,12 @@ defmodule Accent.Prompts do _ -> Provider.NotImplemented end - struct!(struct_module, config: config) + struct!(struct_module, config: fetch_config(config)) + end + + defp fetch_config(%{"provider" => provider, "use_platform" => true}) do + Map.get(Application.get_env(:accent, __MODULE__)[:default_providers_config], provider) end + + defp fetch_config(%{"config" => config}), do: config end diff --git a/lib/prompts/provider/open_ai.ex b/lib/prompts/provider/open_ai.ex index ff1131fb3..e09b5f78e 100644 --- a/lib/prompts/provider/open_ai.ex +++ b/lib/prompts/provider/open_ai.ex @@ -9,7 +9,7 @@ defmodule Accent.Prompts.Provider.OpenAI do def enabled?(_), do: true def completions(provider, prompt, user_input) do - config = provider.config["config"] + config = provider.config params = %{ messages: [ diff --git a/lib/web/controllers/merge_controller.ex b/lib/web/controllers/merge_controller.ex index 64541363e..a856ab8e2 100644 --- a/lib/web/controllers/merge_controller.ex +++ b/lib/web/controllers/merge_controller.ex @@ -3,9 +3,9 @@ defmodule Accent.MergeController do import Canary.Plugs - alias Accent.Hook.Context, as: HookContext alias Accent.Project alias Movement.Builders.RevisionMerge, as: RevisionMergeBuilder + alias Movement.Context alias Movement.Persisters.RevisionMerge, as: RevisionMergePersister plug(Plug.Assign, canary_action: :merge) @@ -44,27 +44,14 @@ defmodule Accent.MergeController do """ def create(conn, _params) do conn.assigns[:movement_context] - |> Movement.Context.assign(:revision, conn.assigns[:revision]) - |> Movement.Context.assign(:merge_type, conn.assigns[:merge_type]) - |> Movement.Context.assign(:options, conn.assigns[:merge_options]) - |> Movement.Context.assign(:user_id, conn.assigns[:current_user].id) + |> Context.assign(:revision, conn.assigns[:revision]) + |> Context.assign(:merge_type, conn.assigns[:merge_type]) + |> Context.assign(:options, conn.assigns[:merge_options]) + |> Context.assign(:user_id, conn.assigns[:current_user].id) |> RevisionMergeBuilder.build() |> RevisionMergePersister.persist() |> case do - {:ok, {_context, []}} -> - send_resp(conn, :ok, "") - - {:ok, _} -> - Accent.Hook.outbound(%HookContext{ - event: "add_translations", - project_id: conn.assigns[:project].id, - user_id: conn.assigns[:current_user].id, - payload: %{ - merge_type: conn.assigns[:merge_type], - language_name: conn.assigns[:revision].language.name - } - }) - + {:ok, {_context, _}} -> send_resp(conn, :ok, "") {:error, _reason} -> @@ -74,7 +61,7 @@ defmodule Accent.MergeController do defp assign_comparer(conn, _) do comparer = Movement.Comparer.comparer(:merge, conn.params["merge_type"]) - context = Movement.Context.assign(conn.assigns[:movement_context], :comparer, comparer) + context = Context.assign(conn.assigns[:movement_context], :comparer, comparer) assign(conn, :movement_context, context) end diff --git a/lib/web/controllers/sync_controller.ex b/lib/web/controllers/sync_controller.ex index a110745aa..e0310072e 100644 --- a/lib/web/controllers/sync_controller.ex +++ b/lib/web/controllers/sync_controller.ex @@ -3,9 +3,9 @@ defmodule Accent.SyncController do import Canary.Plugs - alias Accent.Hook.Context, as: HookContext alias Accent.Project alias Movement.Builders.ProjectSync, as: SyncBuilder + alias Movement.Context alias Movement.Persisters.ProjectSync, as: SyncPersister plug(Plug.Assign, canary_action: :sync) @@ -41,25 +41,12 @@ defmodule Accent.SyncController do """ def create(conn, _) do conn.assigns[:movement_context] - |> Movement.Context.assign(:project, conn.assigns[:project]) - |> Movement.Context.assign(:user_id, conn.assigns[:current_user].id) + |> Context.assign(:project, conn.assigns[:project]) + |> Context.assign(:user_id, conn.assigns[:current_user].id) |> SyncBuilder.build() |> SyncPersister.persist() |> case do - {:ok, {_context, []}} -> - send_resp(conn, :ok, "") - - {:ok, {context, _operations}} -> - Accent.Hook.outbound(%HookContext{ - event: "sync", - project_id: conn.assigns[:project].id, - user_id: conn.assigns[:current_user].id, - payload: %{ - batch_operation_stats: context.assigns[:batch_operation].stats, - document_path: context.assigns[:document].path - } - }) - + {:ok, {_context, _}} -> send_resp(conn, :ok, "") {:error, _reason} -> @@ -69,7 +56,7 @@ defmodule Accent.SyncController do defp assign_comparer(conn, _) do comparer = Movement.Comparer.comparer(:sync, conn.params["sync_type"]) - context = Movement.Context.assign(conn.assigns[:movement_context], :comparer, comparer) + context = Context.assign(conn.assigns[:movement_context], :comparer, comparer) assign(conn, :movement_context, context) end diff --git a/lib/web/plugs/movement_context_parser.ex b/lib/web/plugs/movement_context_parser.ex index aa8bf3998..7c5ce4d7d 100644 --- a/lib/web/plugs/movement_context_parser.ex +++ b/lib/web/plugs/movement_context_parser.ex @@ -90,7 +90,7 @@ defmodule Accent.Plugs.MovementContextParser do end def assign_movement_context(conn, _) do - assign(conn, :movement_context, %Context{assigns: %{options: [], project: conn.assigns[:project]}}) + assign(conn, :movement_context, Context.assign(%Context{}, :project, conn.assigns[:project])) end def assign_movement_version(%{assigns: %{version: version, movement_context: context}} = conn, _opts) do diff --git a/mix.exs b/mix.exs index 8ee1c0fbb..8e84e1d20 100644 --- a/mix.exs +++ b/mix.exs @@ -1,7 +1,7 @@ defmodule Accent.Mixfile do use Mix.Project - @version "1.19.12" + @version "1.21.4" def project do [ @@ -82,6 +82,7 @@ defmodule Accent.Mixfile do {:jason, "~> 1.2", override: true}, {:erlsom, "~> 1.5"}, {:xml_builder, "~> 2.0"}, + {:aws_signature, "~> 0.3"}, # Auth {:ueberauth, "~> 0.10"}, @@ -91,6 +92,7 @@ defmodule Accent.Mixfile do {:ueberauth_github, "~> 0.7"}, {:ueberauth_discord, "~> 0.5"}, {:ueberauth_auth0, "~> 2.0"}, + {:ueberauth_oidc, "~> 0.1.7"}, # Errors {:sentry, "~> 7.0"}, @@ -114,6 +116,7 @@ defmodule Accent.Mixfile do # Mock testing {:mox, "~> 1.0", only: :test}, {:mock, "~> 0.3.0", only: :test}, + {:factori, "~> 0.13", only: :test}, # Google API authentication {:goth, "~> 1.4"}, diff --git a/mix.lock b/mix.lock index 32f80e65a..a05a8e0f4 100644 --- a/mix.lock +++ b/mix.lock @@ -2,28 +2,29 @@ "absinthe": {:hex, :absinthe, "1.7.6", "0b897365f98d068cfcb4533c0200a8e58825a4aeeae6ec33633ebed6de11773b", [:mix], [{:dataloader, "~> 1.0.0 or ~> 2.0", [hex: :dataloader, repo: "hexpm", optional: true]}, {:decimal, "~> 1.0 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: true]}, {:nimble_parsec, "~> 1.2.2 or ~> 1.3", [hex: :nimble_parsec, repo: "hexpm", optional: false]}, {:opentelemetry_process_propagator, "~> 0.2.1", [hex: :opentelemetry_process_propagator, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "e7626951ca5eec627da960615b51009f3a774765406ff02722b1d818f17e5778"}, "absinthe_error_payload": {:hex, :absinthe_error_payload, "1.1.4", "502ff239148c8deaac028ddb600d6502d5be68d24fece0c93f4c3cf7e74c1a4d", [:make, :mix], [{:absinthe, "~> 1.3", [hex: :absinthe, repo: "hexpm", optional: false]}, {:ecto, "~> 3.1", [hex: :ecto, repo: "hexpm", optional: false]}], "hexpm", "9e262ef2fd4a2c644075e0cdde2573b1f713c0676ab905c8640eaa8a882b2aca"}, "absinthe_plug": {:hex, :absinthe_plug, "1.5.8", "38d230641ba9dca8f72f1fed2dfc8abd53b3907d1996363da32434ab6ee5d6ab", [:mix], [{:absinthe, "~> 1.5", [hex: :absinthe, repo: "hexpm", optional: false]}, {:plug, "~> 1.4", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "bbb04176647b735828861e7b2705465e53e2cf54ccf5a73ddd1ebd855f996e5a"}, + "aws_signature": {:hex, :aws_signature, "0.3.2", "adf33bc4af00b2089b7708bf20e3246f09c639a905a619b3689f0a0a22c3ef8f", [:rebar3], [], "hexpm", "b0daf61feb4250a8ab0adea60db3e336af732ff71dd3fb22e45ae3dcbd071e44"}, "bamboo": {:hex, :bamboo, "2.3.0", "d2392a2cabe91edf488553d3c70638b532e8db7b76b84b0a39e3dfe492ffd6fc", [:mix], [{:hackney, ">= 1.15.2", [hex: :hackney, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:mime, "~> 1.4 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:plug, "~> 1.0", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "dd0037e68e108fd04d0e8773921512c940e35d981e097b5793543e3b2f9cd3f6"}, "bamboo_phoenix": {:hex, :bamboo_phoenix, "1.0.0", "f3cc591ffb163ed0bf935d256f1f4645cd870cf436545601215745fb9cc9953f", [:mix], [{:bamboo, ">= 2.0.0", [hex: :bamboo, repo: "hexpm", optional: false]}, {:phoenix, ">= 1.3.0", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "6db88fbb26019c84a47994bb2bd879c0887c29ce6c559bc6385fd54eb8b37dee"}, "bamboo_smtp": {:hex, :bamboo_smtp, "4.2.2", "e9f57a2300df9cb496c48751bd7668a86a2b89aa2e79ccaa34e0c46a5f64c3ae", [:mix], [{:bamboo, "~> 2.2.0", [hex: :bamboo, repo: "hexpm", optional: false]}, {:gen_smtp, "~> 1.2.0", [hex: :gen_smtp, repo: "hexpm", optional: false]}], "hexpm", "28cac2ec8adaae02aed663bf68163992891a3b44cfd7ada0bebe3e09bed7207f"}, - "bandit": {:hex, :bandit, "1.2.0", "2b5784909cc25b2514868055ff27458cdc63314514b90d86448ff91d18bece80", [:mix], [{:hpax, "~> 0.1.1", [hex: :hpax, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:thousand_island, "~> 1.0", [hex: :thousand_island, repo: "hexpm", optional: false]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "05688b883d87cc3b32991517a61e8c2ce8ee2dd6aa6eb73635426002a6661491"}, + "bandit": {:hex, :bandit, "1.3.0", "6a4e8d7c9ea721edd02c389e2cc867890cd96f83116e71ddf1ccbdd80661550c", [:mix], [{:hpax, "~> 0.1.1", [hex: :hpax, repo: "hexpm", optional: false]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:thousand_island, "~> 1.0", [hex: :thousand_island, repo: "hexpm", optional: false]}, {:websock, "~> 0.5", [hex: :websock, repo: "hexpm", optional: false]}], "hexpm", "bda37d6c614d74778a5dc43b8bcdc3245cd30619eab0342f58042f968f2165da"}, "bunt": {:hex, :bunt, "1.0.0", "081c2c665f086849e6d57900292b3a161727ab40431219529f13c4ddcf3e7a44", [:mix], [], "hexpm", "dc5f86aa08a5f6fa6b8096f0735c4e76d54ae5c9fa2c143e5a1fc7c1cd9bb6b5"}, "cachex": {:hex, :cachex, "3.6.0", "14a1bfbeee060dd9bec25a5b6f4e4691e3670ebda28c8ba2884b12fe30b36bf8", [:mix], [{:eternal, "~> 1.2", [hex: :eternal, repo: "hexpm", optional: false]}, {:jumper, "~> 1.0", [hex: :jumper, repo: "hexpm", optional: false]}, {:sleeplocks, "~> 1.1", [hex: :sleeplocks, repo: "hexpm", optional: false]}, {:unsafe, "~> 1.0", [hex: :unsafe, repo: "hexpm", optional: false]}], "hexpm", "ebf24e373883bc8e0c8d894a63bbe102ae13d918f790121f5cfe6e485cc8e2e2"}, "canada": {:hex, :canada, "2.0.0", "ce5e058f576a0625959fc5427fcde15311fb28a5ebc13775eafd13468ad16553", [:mix], [], "hexpm", "49a648c48d8b0864380f38f02a7f316bd30fd45602205c48197432b5225d8596"}, "canary": {:git, "https://github.com/runhyve/canary.git", "b81d780e1cb7a1c276599f980ab9c9a7c9cd8c12", []}, - "castore": {:hex, :castore, "1.0.5", "9eeebb394cc9a0f3ae56b813459f990abb0a3dedee1be6b27fdb50301930502f", [:mix], [], "hexpm", "8d7c597c3e4a64c395980882d4bca3cebb8d74197c590dc272cfd3b6a6310578"}, + "castore": {:hex, :castore, "1.0.6", "ffc42f110ebfdafab0ea159cd43d31365fa0af0ce4a02ecebf1707ae619ee727", [:mix], [], "hexpm", "374c6e7ca752296be3d6780a6d5b922854ffcc74123da90f2f328996b962d33a"}, "certifi": {:hex, :certifi, "2.12.0", "2d1cca2ec95f59643862af91f001478c9863c2ac9cb6e2f89780bfd8de987329", [:rebar3], [], "hexpm", "ee68d85df22e554040cdb4be100f33873ac6051387baf6a8f6ce82272340ff1c"}, "cloak": {:hex, :cloak, "1.1.2", "7e0006c2b0b98d976d4f559080fabefd81f0e0a50a3c4b621f85ceeb563e80bb", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}], "hexpm", "940d5ac4fcd51b252930fd112e319ea5ae6ab540b722f3ca60a85666759b9585"}, "cloak_ecto": {:hex, :cloak_ecto, "1.2.0", "e86a3df3bf0dc8980f70406bcb0af2858bac247d55494d40bc58a152590bd402", [:mix], [{:cloak, "~> 1.1.1", [hex: :cloak, repo: "hexpm", optional: false]}, {:ecto, "~> 3.0", [hex: :ecto, repo: "hexpm", optional: false]}], "hexpm", "8bcc677185c813fe64b786618bd6689b1707b35cd95acaae0834557b15a0c62f"}, "combine": {:hex, :combine, "0.10.0", "eff8224eeb56498a2af13011d142c5e7997a80c8f5b97c499f84c841032e429f", [:mix], [], "hexpm", "1b1dbc1790073076580d0d1d64e42eae2366583e7aecd455d1215b0d16f2451b"}, "corsica": {:hex, :corsica, "2.1.3", "dccd094ffce38178acead9ae743180cdaffa388f35f0461ba1e8151d32e190e6", [:mix], [{:plug, "~> 1.0", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "616c08f61a345780c2cf662ff226816f04d8868e12054e68963e95285b5be8bc"}, - "credo": {:hex, :credo, "1.7.3", "05bb11eaf2f2b8db370ecaa6a6bda2ec49b2acd5e0418bc106b73b07128c0436", [:mix], [{:bunt, "~> 0.2.1 or ~> 1.0", [hex: :bunt, repo: "hexpm", optional: false]}, {:file_system, "~> 0.2 or ~> 1.0", [hex: :file_system, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "35ea675a094c934c22fb1dca3696f3c31f2728ae6ef5a53b5d648c11180a4535"}, + "credo": {:hex, :credo, "1.7.5", "643213503b1c766ec0496d828c90c424471ea54da77c8a168c725686377b9545", [:mix], [{:bunt, "~> 0.2.1 or ~> 1.0", [hex: :bunt, repo: "hexpm", optional: false]}, {:file_system, "~> 0.2 or ~> 1.0", [hex: :file_system, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "f799e9b5cd1891577d8c773d245668aa74a2fcd15eb277f51a0131690ebfb3fd"}, "credo_envvar": {:hex, :credo_envvar, "0.1.4", "40817c10334e400f031012c0510bfa0d8725c19d867e4ae39cf14f2cbebc3b20", [:mix], [{:credo, "~> 1.0", [hex: :credo, repo: "hexpm", optional: false]}], "hexpm", "5055cdb4bcbaf7d423bc2bb3ac62b4e2d825e2b1e816884c468dee59d0363009"}, "csv": {:hex, :csv, "2.5.0", "c47b5a5221bf2e56d6e8eb79e77884046d7fd516280dc7d9b674251e0ae46246", [:mix], [{:parallel_stream, "~> 1.0.4 or ~> 1.1.0", [hex: :parallel_stream, repo: "hexpm", optional: false]}], "hexpm", "e821f541487045c7591a1963eeb42afff0dfa99bdcdbeb3410795a2f59c77d34"}, "dataloader": {:hex, :dataloader, "2.0.0", "49b42d60b9bb06d761a71d7b034c4b34787957e713d4fae15387a25fcd639112", [:mix], [{:ecto, ">= 3.4.3 and < 4.0.0", [hex: :ecto, repo: "hexpm", optional: true]}, {:opentelemetry_process_propagator, "~> 0.2.1", [hex: :opentelemetry_process_propagator, repo: "hexpm", optional: true]}, {:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "09d61781b76ce216e395cdbc883ff00d00f46a503e215c22722dba82507dfef0"}, "db_connection": {:hex, :db_connection, "2.6.0", "77d835c472b5b67fc4f29556dee74bf511bbafecdcaf98c27d27fa5918152086", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "c2f992d15725e721ec7fbc1189d4ecdb8afef76648c746a8e1cad35e3b8a35f3"}, "decimal": {:hex, :decimal, "2.1.1", "5611dca5d4b2c3dd497dec8f68751f1f1a54755e8ed2a966c2633cf885973ad6", [:mix], [], "hexpm", "53cfe5f497ed0e7771ae1a475575603d77425099ba5faef9394932b35020ffcc"}, "dialyxir": {:hex, :dialyxir, "1.4.3", "edd0124f358f0b9e95bfe53a9fcf806d615d8f838e2202a9f430d59566b6b53b", [:mix], [{:erlex, ">= 0.2.6", [hex: :erlex, repo: "hexpm", optional: false]}], "hexpm", "bf2cfb75cd5c5006bec30141b131663299c661a864ec7fbbc72dfa557487a986"}, - "ecto": {:hex, :ecto, "3.11.1", "4b4972b717e7ca83d30121b12998f5fcdc62ba0ed4f20fd390f16f3270d85c3e", [:mix], [{:decimal, "~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "ebd3d3772cd0dfcd8d772659e41ed527c28b2a8bde4b00fe03e0463da0f1983b"}, + "ecto": {:hex, :ecto, "3.11.2", "e1d26be989db350a633667c5cda9c3d115ae779b66da567c68c80cfb26a8c9ee", [:mix], [{:decimal, "~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "3c38bca2c6f8d8023f2145326cc8a80100c3ffe4dcbd9842ff867f7fc6156c65"}, "ecto_dev_logger": {:hex, :ecto_dev_logger, "0.10.0", "5b3a3900b845e0d40127bed9bdf9d02bf20aa38198a60fe108cddff63ed0048f", [:mix], [{:ecto, "~> 3.7", [hex: :ecto, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "a55e58bad5d5c9b8ef2a3c3347dbdf7efa880a5371cf1457e44b41f489a43927"}, "ecto_psql_extras": {:hex, :ecto_psql_extras, "0.7.15", "0fc29dbae0e444a29bd6abeee4cf3c4c037e692a272478a234a1cc765077dbb1", [:mix], [{:ecto_sql, "~> 3.7", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16.0 or ~> 0.17.0", [hex: :postgrex, repo: "hexpm", optional: false]}, {:table_rex, "~> 3.1.1 or ~> 4.0.0", [hex: :table_rex, repo: "hexpm", optional: false]}], "hexpm", "b6127f3a5c6fc3d84895e4768cc7c199f22b48b67d6c99b13fbf4a374e73f039"}, "ecto_sql": {:hex, :ecto_sql, "3.11.1", "e9abf28ae27ef3916b43545f9578b4750956ccea444853606472089e7d169470", [:mix], [{:db_connection, "~> 2.4.1 or ~> 2.5", [hex: :db_connection, repo: "hexpm", optional: false]}, {:ecto, "~> 3.11.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:myxql, "~> 0.6.0", [hex: :myxql, repo: "hexpm", optional: true]}, {:postgrex, "~> 0.16.0 or ~> 0.17.0 or ~> 1.0", [hex: :postgrex, repo: "hexpm", optional: true]}, {:tds, "~> 2.1.1 or ~> 2.2", [hex: :tds, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.0 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "ce14063ab3514424276e7e360108ad6c2308f6d88164a076aac8a387e1fea634"}, @@ -33,8 +34,10 @@ "eternal": {:hex, :eternal, "1.2.2", "d1641c86368de99375b98d183042dd6c2b234262b8d08dfd72b9eeaafc2a1abd", [:mix], [], "hexpm", "2c9fe32b9c3726703ba5e1d43a1d255a4f3f2d8f8f9bc19f094c7cb1a7a9e782"}, "excoveralls": {:hex, :excoveralls, "0.18.0", "b92497e69465dc51bc37a6422226ee690ab437e4c06877e836f1c18daeb35da9", [:mix], [{:castore, "~> 1.0", [hex: :castore, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}], "hexpm", "1109bb911f3cb583401760be49c02cbbd16aed66ea9509fc5479335d284da60b"}, "exile": {:hex, :exile, "0.9.1", "832b6340cf800661e90e52cebc760b795450f803c0e9265ccdc54150423fbb32", [:make, :mix], [{:elixir_make, "~> 0.6", [hex: :elixir_make, repo: "hexpm", optional: false]}], "hexpm", "553a1847b27118c843d3dc6912adbc36d60336811d15ad70a31b82eb5a416328"}, + "factori": {:hex, :factori, "0.13.0", "989602327020e6ad9cb3b5403a30591cc3b81528477d894f7d11f3c0eb3c3d18", [:make, :mix], [{:ecto, "~> 3.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:ecto_sql, "~> 3.0", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:faker, "~> 0.16", [hex: :faker, repo: "hexpm", optional: false]}], "hexpm", "cedb92ff18ad2a5e854e46344ad59bdbfb2729bef7bf557bc06e89e89c98df1a"}, + "faker": {:hex, :faker, "0.18.0", "943e479319a22ea4e8e39e8e076b81c02827d9302f3d32726c5bf82f430e6e14", [:mix], [], "hexpm", "bfbdd83958d78e2788e99ec9317c4816e651ad05e24cfd1196ce5db5b3e81797"}, "fast_yaml": {:git, "https://github.com/processone/fast_yaml.git", "e789f68895f71b7ad31057177810ca0161bf790e", [ref: "e789f68895f71b7ad31057177810ca0161bf790e"]}, - "file_system": {:hex, :file_system, "0.2.10", "fb082005a9cd1711c05b5248710f8826b02d7d1784e7c3451f9c1231d4fc162d", [:mix], [], "hexpm", "41195edbfb562a593726eda3b3e8b103a309b733ad25f3d642ba49696bf715dc"}, + "file_system": {:hex, :file_system, "1.0.0", "b689cc7dcee665f774de94b5a832e578bd7963c8e637ef940cd44327db7de2cd", [:mix], [], "hexpm", "6752092d66aec5a10e662aefeed8ddb9531d79db0bc145bb8c40325ca1d8536d"}, "finch": {:hex, :finch, "0.17.0", "17d06e1d44d891d20dbd437335eebe844e2426a0cd7e3a3e220b461127c73f70", [:mix], [{:castore, "~> 0.1 or ~> 1.0", [hex: :castore, repo: "hexpm", optional: false]}, {:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:mint, "~> 1.3", [hex: :mint, repo: "hexpm", optional: false]}, {:nimble_options, "~> 0.4 or ~> 1.0", [hex: :nimble_options, repo: "hexpm", optional: false]}, {:nimble_pool, "~> 0.2.6 or ~> 1.0", [hex: :nimble_pool, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "8d014a661bb6a437263d4b5abf0bcbd3cf0deb26b1e8596f2a271d22e48934c7"}, "gen_smtp": {:hex, :gen_smtp, "1.2.0", "9cfc75c72a8821588b9b9fe947ae5ab2aed95a052b81237e0928633a13276fd3", [:rebar3], [{:ranch, ">= 1.8.0", [hex: :ranch, repo: "hexpm", optional: false]}], "hexpm", "5ee0375680bca8f20c4d85f58c2894441443a743355430ff33a783fe03296779"}, "gettext": {:hex, :gettext, "0.20.0", "75ad71de05f2ef56991dbae224d35c68b098dd0e26918def5bb45591d5c8d429", [:mix], [], "hexpm", "1c03b177435e93a47441d7f681a7040bd2a816ece9e2666d1c9001035121eb3d"}, @@ -56,20 +59,21 @@ "mock": {:hex, :mock, "0.3.8", "7046a306b71db2488ef54395eeb74df0a7f335a7caca4a3d3875d1fc81c884dd", [:mix], [{:meck, "~> 0.9.2", [hex: :meck, repo: "hexpm", optional: false]}], "hexpm", "7fa82364c97617d79bb7d15571193fc0c4fe5afd0c932cef09426b3ee6fe2022"}, "mox": {:hex, :mox, "1.1.0", "0f5e399649ce9ab7602f72e718305c0f9cdc351190f72844599545e4996af73c", [:mix], [], "hexpm", "d44474c50be02d5b72131070281a5d3895c0e7a95c780e90bc0cfe712f633a13"}, "new_relic_absinthe": {:hex, :new_relic_absinthe, "0.0.4", "57917f99789d9b36e4beb599deba495a474e5bf99a5c70a33717b0e17f1c5d4d", [:mix], [{:absinthe, "~> 1.4", [hex: :absinthe, repo: "hexpm", optional: false]}, {:new_relic_agent, "~> 1.19", [hex: :new_relic_agent, repo: "hexpm", optional: false]}], "hexpm", "6b796662e550ddd07e98ff3df95803a6b2a023605e78e0a45261d3e66341c296"}, - "new_relic_agent": {:hex, :new_relic_agent, "1.28.0", "eb015edb4f4887a31ee0488cf9ce97b7e46c9e0208eeff8737d8ea09cd506d09", [:mix], [{:castore, ">= 0.1.0", [hex: :castore, repo: "hexpm", optional: false]}, {:ecto, ">= 3.4.1", [hex: :ecto, repo: "hexpm", optional: true]}, {:ecto_sql, ">= 3.4.0", [hex: :ecto_sql, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:phoenix, ">= 1.5.5", [hex: :phoenix, repo: "hexpm", optional: true]}, {:plug, ">= 1.10.4", [hex: :plug, repo: "hexpm", optional: true]}, {:plug_cowboy, ">= 2.4.0", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:redix, ">= 0.11.0", [hex: :redix, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "ef220f429f2d673e78679ec96cd4e8979b2cb2166b204bfe1a43ca974520743a"}, + "new_relic_agent": {:hex, :new_relic_agent, "1.29.0", "928795ce65f2a28688100973e70ba96001288df66be8fb87903d1f49b7ed4cc0", [:mix], [{:castore, ">= 0.1.0", [hex: :castore, repo: "hexpm", optional: false]}, {:ecto, ">= 3.9.5", [hex: :ecto, repo: "hexpm", optional: true]}, {:ecto_sql, ">= 3.4.0", [hex: :ecto_sql, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:phoenix, ">= 1.5.5", [hex: :phoenix, repo: "hexpm", optional: true]}, {:plug, ">= 1.10.4", [hex: :plug, repo: "hexpm", optional: true]}, {:plug_cowboy, ">= 2.4.0", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:redix, ">= 0.11.0", [hex: :redix, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "ff90ba00671b84039141e662699d337254c4c4f7285c417839aad2e166e51e66"}, "nimble_options": {:hex, :nimble_options, "1.1.0", "3b31a57ede9cb1502071fade751ab0c7b8dbe75a9a4c2b5bbb0943a690b63172", [:mix], [], "hexpm", "8bbbb3941af3ca9acc7835f5655ea062111c9c27bcac53e004460dfd19008a99"}, "nimble_parsec": {:hex, :nimble_parsec, "1.4.0", "51f9b613ea62cfa97b25ccc2c1b4216e81df970acd8e16e8d1bdc58fef21370d", [:mix], [], "hexpm", "9c565862810fb383e9838c1dd2d7d2c437b3d13b267414ba6af33e50d2d1cf28"}, "nimble_pool": {:hex, :nimble_pool, "1.0.0", "5eb82705d138f4dd4423f69ceb19ac667b3b492ae570c9f5c900bb3d2f50a847", [:mix], [], "hexpm", "80be3b882d2d351882256087078e1b1952a28bf98d0a287be87e4a24a710b67a"}, "oauth2": {:hex, :oauth2, "2.1.0", "beb657f393814a3a7a8a15bd5e5776ecae341fd344df425342a3b6f1904c2989", [:mix], [{:tesla, "~> 1.5", [hex: :tesla, repo: "hexpm", optional: false]}], "hexpm", "8ac07f85b3307dd1acfeb0ec852f64161b22f57d0ce0c15e616a1dfc8ebe2b41"}, - "oban": {:hex, :oban, "2.17.3", "ddfd5710aadcd550d2e174c8d73ce5f1865601418cf54a91775f20443fb832b7", [:mix], [{:ecto_sql, "~> 3.6", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:ecto_sqlite3, "~> 0.9", [hex: :ecto_sqlite3, repo: "hexpm", optional: true]}, {:jason, "~> 1.1", [hex: :jason, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16", [hex: :postgrex, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "452eada8bfe0d0fefd0740ab5fa8cf3ef6c375df0b4a3c3805d179022a04738a"}, + "oban": {:hex, :oban, "2.17.6", "bac1dacd836edbf6a200ddd880db10faa2d39bb2e550ec6d19b3eb9c43852c2a", [:mix], [{:ecto_sql, "~> 3.10", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:ecto_sqlite3, "~> 0.9", [hex: :ecto_sqlite3, repo: "hexpm", optional: true]}, {:jason, "~> 1.1", [hex: :jason, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16", [hex: :postgrex, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "623f3554212e9a776e015156c47f076d66c7b74115ac47a7d3acba0294e65acb"}, + "openid_connect": {:hex, :openid_connect, "0.2.2", "c05055363330deab39ffd89e609db6b37752f255a93802006d83b45596189c0b", [:mix], [{:httpoison, "~> 1.2", [hex: :httpoison, repo: "hexpm", optional: false]}, {:jason, ">= 1.0.0", [hex: :jason, repo: "hexpm", optional: false]}, {:jose, "~> 1.8", [hex: :jose, repo: "hexpm", optional: false]}], "hexpm", "735769b6d592124b58edd0582554ce638524c0214cd783d8903d33357d74cc13"}, "p1_utils": {:hex, :p1_utils, "1.0.15", "731f76ae1f31f4554afb2ae629cb5589d53bd13efc72b11f5a7c3b1242f91046", [:rebar3], [], "hexpm", "1d308c3f37d7f770fb39abe3b86701b82d54414bc2499d9499edde3cb50bcf19"}, "parallel_stream": {:hex, :parallel_stream, "1.1.0", "f52f73eb344bc22de335992377413138405796e0d0ad99d995d9977ac29f1ca9", [:mix], [], "hexpm", "684fd19191aedfaf387bbabbeb8ff3c752f0220c8112eb907d797f4592d6e871"}, "parse_trans": {:hex, :parse_trans, "3.4.1", "6e6aa8167cb44cc8f39441d05193be6e6f4e7c2946cb2759f015f8c56b76e5ff", [:rebar3], [], "hexpm", "620a406ce75dada827b82e453c19cf06776be266f5a67cff34e1ef2cbb60e49a"}, "phoenix": {:hex, :phoenix, "1.7.11", "1d88fc6b05ab0c735b250932c4e6e33bfa1c186f76dcf623d8dd52f07d6379c7", [:mix], [{:castore, ">= 0.0.0", [hex: :castore, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix_pubsub, "~> 2.1", [hex: :phoenix_pubsub, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.14", [hex: :plug, repo: "hexpm", optional: false]}, {:plug_cowboy, "~> 2.7", [hex: :plug_cowboy, repo: "hexpm", optional: true]}, {:plug_crypto, "~> 1.2 or ~> 2.0", [hex: :plug_crypto, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:websock_adapter, "~> 0.5.3", [hex: :websock_adapter, repo: "hexpm", optional: false]}], "hexpm", "b1ec57f2e40316b306708fe59b92a16b9f6f4bf50ccfa41aa8c7feb79e0ec02a"}, - "phoenix_ecto": {:hex, :phoenix_ecto, "4.4.3", "86e9878f833829c3f66da03d75254c155d91d72a201eb56ae83482328dc7ca93", [:mix], [{:ecto, "~> 3.5", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "d36c401206f3011fefd63d04e8ef626ec8791975d9d107f9a0817d426f61ac07"}, + "phoenix_ecto": {:hex, :phoenix_ecto, "4.5.1", "6fdbc334ea53620e71655664df6f33f670747b3a7a6c4041cdda3e2c32df6257", [:mix], [{:ecto, "~> 3.5", [hex: :ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.1", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:plug, "~> 1.9", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "ebe43aa580db129e54408e719fb9659b7f9e0d52b965c5be26cdca416ecead28"}, "phoenix_html": {:hex, :phoenix_html, "3.3.3", "380b8fb45912b5638d2f1d925a3771b4516b9a78587249cabe394e0a5d579dc9", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: true]}], "hexpm", "923ebe6fec6e2e3b3e569dfbdc6560de932cd54b000ada0208b5f45024bdd76c"}, - "phoenix_live_reload": {:hex, :phoenix_live_reload, "1.4.1", "2aff698f5e47369decde4357ba91fc9c37c6487a512b41732818f2204a8ef1d3", [:mix], [{:file_system, "~> 0.2.1 or ~> 0.3", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "9bffb834e7ddf08467fe54ae58b5785507aaba6255568ae22b4d46e2bb3615ab"}, - "phoenix_live_view": {:hex, :phoenix_live_view, "0.20.0", "3f3531c835e46a3b45b4c3ca4a09cef7ba1d0f0d0035eef751c7084b8adb1299", [:mix], [{:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6.15 or ~> 1.7.0", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.3", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "29875f8a58fb031f2dc8f3be025c92ed78d342b46f9bbf6dfe579549d7c81050"}, + "phoenix_live_reload": {:hex, :phoenix_live_reload, "1.5.2", "354460993a480656b71c3887f5565f612b3bdbdd8688c83f9e6f512307067dd4", [:mix], [{:file_system, "~> 0.3 or ~> 1.0", [hex: :file_system, repo: "hexpm", optional: false]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}], "hexpm", "2bb3722f327e14a7aa47b1acf27ed633c8cd27b167e18b8237954b9b4804af39"}, + "phoenix_live_view": {:hex, :phoenix_live_view, "0.20.14", "70fa101aa0539e81bed4238777498f6215e9dda3461bdaa067cad6908110c364", [:mix], [{:floki, "~> 0.36", [hex: :floki, repo: "hexpm", optional: true]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.6.15 or ~> 1.7.0", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.3 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: true]}, {:plug, "~> 1.15", [hex: :plug, repo: "hexpm", optional: false]}, {:telemetry, "~> 0.4.2 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "82f6d006c5264f979ed5eb75593d808bbe39020f20df2e78426f4f2d570e2402"}, "phoenix_pubsub": {:hex, :phoenix_pubsub, "2.1.3", "3168d78ba41835aecad272d5e8cd51aa87a7ac9eb836eabc42f6e57538e3731d", [:mix], [], "hexpm", "bba06bc1dcfd8cb086759f0edc94a8ba2bc8896d5331a1e2c2902bf8e36ee502"}, "phoenix_template": {:hex, :phoenix_template, "1.0.4", "e2092c132f3b5e5b2d49c96695342eb36d0ed514c5b252a77048d5969330d639", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}], "hexpm", "2c0c81f0e5c6753faf5cca2f229c9709919aba34fab866d3bc05060c9c444206"}, "phoenix_view": {:hex, :phoenix_view, "2.0.3", "4d32c4817fce933693741deeb99ef1392619f942633dde834a5163124813aad3", [:mix], [{:phoenix_html, "~> 2.14.2 or ~> 3.0 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: true]}, {:phoenix_template, "~> 1.0", [hex: :phoenix_template, repo: "hexpm", optional: false]}], "hexpm", "cd34049af41be2c627df99cd4eaa71fc52a328c0c3d8e7d4aa28f880c30e7f64"}, @@ -78,7 +82,7 @@ "plug_assign": {:hex, :plug_assign, "2.0.2", "19e827d254711a93b8ef1eab07b6b0932f2bc649f80b8245ad9e73e381a3064f", [:mix], [{:plug, "~> 1.15", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "0d8d72961d2d838aaddf020d49ad6f83270d7d9e2bc420458d33bb795d012810"}, "plug_canonical_host": {:hex, :plug_canonical_host, "2.0.3", "3d96c3340cc8a434eb6758a4a34de6c152bd781be96bb8439545da2d17ecf576", [:make, :mix], [{:plug, "~> 1.0", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "aca98ac6d0036391b84d5a40af6f946c839fb0d588bf0064029a2e8931431ea6"}, "plug_crypto": {:hex, :plug_crypto, "2.0.0", "77515cc10af06645abbfb5e6ad7a3e9714f805ae118fa1a70205f80d2d70fe73", [:mix], [], "hexpm", "53695bae57cc4e54566d993eb01074e4d894b65a3766f1c43e2c61a1b0f45ea9"}, - "postgrex": {:hex, :postgrex, "0.17.4", "5777781f80f53b7c431a001c8dad83ee167bcebcf3a793e3906efff680ab62b3", [:mix], [{:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "6458f7d5b70652bc81c3ea759f91736c16a31be000f306d3c64bcdfe9a18b3cc"}, + "postgrex": {:hex, :postgrex, "0.17.5", "0483d054938a8dc069b21bdd636bf56c487404c241ce6c319c1f43588246b281", [:mix], [{:db_connection, "~> 2.1", [hex: :db_connection, repo: "hexpm", optional: false]}, {:decimal, "~> 1.5 or ~> 2.0", [hex: :decimal, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: true]}, {:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: true]}], "hexpm", "50b8b11afbb2c4095a3ba675b4f055c416d0f3d7de6633a595fc131a828a67eb"}, "ranch": {:hex, :ranch, "2.1.0", "2261f9ed9574dcfcc444106b9f6da155e6e540b2f82ba3d42b339b93673b72a3", [:make, :rebar3], [], "hexpm", "244ee3fa2a6175270d8e1fc59024fd9dbc76294a321057de8f803b1479e76916"}, "scrivener": {:hex, :scrivener, "2.7.2", "1d913c965ec352650a7f864ad7fd8d80462f76a32f33d57d1e48bc5e9d40aba2", [:mix], [], "hexpm", "7866a0ec4d40274efbee1db8bead13a995ea4926ecd8203345af8f90d2b620d9"}, "scrivener_ecto": {:hex, :scrivener_ecto, "2.7.0", "cf64b8cb8a96cd131cdbcecf64e7fd395e21aaa1cb0236c42a7c2e34b0dca580", [:mix], [{:ecto, "~> 3.3", [hex: :ecto, repo: "hexpm", optional: false]}, {:scrivener, "~> 2.4", [hex: :scrivener, repo: "hexpm", optional: false]}], "hexpm", "e809f171687806b0031129034352f5ae44849720c48dd839200adeaf0ac3e260"}, @@ -89,20 +93,21 @@ "table": {:hex, :table, "0.1.2", "87ad1125f5b70c5dea0307aa633194083eb5182ec537efc94e96af08937e14a8", [:mix], [], "hexpm", "7e99bc7efef806315c7e65640724bf165c3061cdc5d854060f74468367065029"}, "table_rex": {:hex, :table_rex, "4.0.0", "3c613a68ebdc6d4d1e731bc973c233500974ec3993c99fcdabb210407b90959b", [:mix], [], "hexpm", "c35c4d5612ca49ebb0344ea10387da4d2afe278387d4019e4d8111e815df8f55"}, "telemetry": {:hex, :telemetry, "1.2.1", "68fdfe8d8f05a8428483a97d7aab2f268aaff24b49e0f599faa091f1d4e7f61c", [:rebar3], [], "hexpm", "dad9ce9d8effc621708f99eac538ef1cbe05d6a874dd741de2e689c47feafed5"}, - "telemetry_metrics": {:hex, :telemetry_metrics, "0.6.1", "315d9163a1d4660aedc3fee73f33f1d355dcc76c5c3ab3d59e76e3edf80eef1f", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "7be9e0871c41732c233be71e4be11b96e56177bf15dde64a8ac9ce72ac9834c6"}, + "telemetry_metrics": {:hex, :telemetry_metrics, "0.6.2", "2caabe9344ec17eafe5403304771c3539f3b6e2f7fb6a6f602558c825d0d0bfb", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "9b43db0dc33863930b9ef9d27137e78974756f5f198cae18409970ed6fa5b561"}, "telemetry_poller": {:hex, :telemetry_poller, "1.0.0", "db91bb424e07f2bb6e73926fcafbfcbcb295f0193e0a00e825e589a0a47e8453", [:rebar3], [{:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "b3a24eafd66c3f42da30fc3ca7dda1e9d546c12250a2d60d7b81d264fbec4f6e"}, - "telemetry_ui": {:hex, :telemetry_ui, "4.1.0", "d69be6a89fd78c7030011363dbed735afb558a925dac24ee2615aae588375d9e", [:mix], [{:ecto, "~> 3.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:ecto_sql, "~> 3.0", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:oban, "~> 2.13", [hex: :oban, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_ecto, "~> 4.4", [hex: :phoenix_ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.0", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 1.0 or ~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16", [hex: :postgrex, repo: "hexpm", optional: false]}, {:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:telemetry_metrics, "~> 0.6", [hex: :telemetry_metrics, repo: "hexpm", optional: false]}, {:telemetry_poller, "~> 1.0", [hex: :telemetry_poller, repo: "hexpm", optional: false]}, {:timex, "~> 3.7", [hex: :timex, repo: "hexpm", optional: false]}, {:vega_lite, "~> 0.1", [hex: :vega_lite, repo: "hexpm", optional: false]}, {:vega_lite_convert, "~> 0.6", [hex: :vega_lite_convert, repo: "hexpm", optional: true]}, {:vix, "~> 0.16", [hex: :vix, repo: "hexpm", optional: true]}], "hexpm", "6e5af411bdc7bbd4ab88294c4f58391ce7ec6cba7119fcb3799e2ec2de7104fa"}, + "telemetry_ui": {:hex, :telemetry_ui, "4.2.0", "a1e883440818bb84a78718fbd9d5c6917988d5ff4e1c19687fd9ad3ab1a108e8", [:mix], [{:ecto, "~> 3.0", [hex: :ecto, repo: "hexpm", optional: false]}, {:ecto_sql, "~> 3.0", [hex: :ecto_sql, repo: "hexpm", optional: false]}, {:jason, "~> 1.0", [hex: :jason, repo: "hexpm", optional: false]}, {:oban, "~> 2.13", [hex: :oban, repo: "hexpm", optional: true]}, {:phoenix, "~> 1.4", [hex: :phoenix, repo: "hexpm", optional: false]}, {:phoenix_ecto, "~> 4.4", [hex: :phoenix_ecto, repo: "hexpm", optional: false]}, {:phoenix_html, "~> 3.0 or ~> 4.0", [hex: :phoenix_html, repo: "hexpm", optional: false]}, {:phoenix_live_view, "~> 0.18", [hex: :phoenix_live_view, repo: "hexpm", optional: false]}, {:phoenix_view, "~> 1.0 or ~> 2.0", [hex: :phoenix_view, repo: "hexpm", optional: false]}, {:postgrex, "~> 0.16", [hex: :postgrex, repo: "hexpm", optional: false]}, {:telemetry, "~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}, {:telemetry_metrics, "~> 0.6", [hex: :telemetry_metrics, repo: "hexpm", optional: false]}, {:telemetry_poller, "~> 1.0", [hex: :telemetry_poller, repo: "hexpm", optional: false]}, {:timex, "~> 3.7", [hex: :timex, repo: "hexpm", optional: false]}, {:vega_lite, "~> 0.1", [hex: :vega_lite, repo: "hexpm", optional: false]}, {:vega_lite_convert, "~> 0.6", [hex: :vega_lite_convert, repo: "hexpm", optional: true]}, {:vix, "~> 0.16", [hex: :vix, repo: "hexpm", optional: true]}], "hexpm", "b321a4915b91a9dc6c3763a2b5bd5e2f9425c7e4680203075684b327bfce9534"}, "tesla": {:hex, :tesla, "1.8.0", "d511a4f5c5e42538d97eef7c40ec4f3e44effdc5068206f42ed859e09e51d1fd", [:mix], [{:castore, "~> 0.1 or ~> 1.0", [hex: :castore, repo: "hexpm", optional: true]}, {:exjsx, ">= 3.0.0", [hex: :exjsx, repo: "hexpm", optional: true]}, {:finch, "~> 0.13", [hex: :finch, repo: "hexpm", optional: true]}, {:fuse, "~> 2.4", [hex: :fuse, repo: "hexpm", optional: true]}, {:gun, ">= 1.0.0", [hex: :gun, repo: "hexpm", optional: true]}, {:hackney, "~> 1.6", [hex: :hackney, repo: "hexpm", optional: true]}, {:ibrowse, "4.4.2", [hex: :ibrowse, repo: "hexpm", optional: true]}, {:jason, ">= 1.0.0", [hex: :jason, repo: "hexpm", optional: true]}, {:mime, "~> 1.0 or ~> 2.0", [hex: :mime, repo: "hexpm", optional: false]}, {:mint, "~> 1.0", [hex: :mint, repo: "hexpm", optional: true]}, {:msgpax, "~> 2.3", [hex: :msgpax, repo: "hexpm", optional: true]}, {:poison, ">= 1.0.0", [hex: :poison, repo: "hexpm", optional: true]}, {:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: true]}], "hexpm", "10501f360cd926a309501287470372af1a6e1cbed0f43949203a4c13300bc79f"}, - "thousand_island": {:hex, :thousand_island, "1.3.2", "bc27f9afba6e1a676dd36507d42e429935a142cf5ee69b8e3f90bff1383943cd", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "0e085b93012cd1057b378fce40cbfbf381ff6d957a382bfdd5eca1a98eec2535"}, + "thousand_island": {:hex, :thousand_island, "1.3.5", "6022b6338f1635b3d32406ff98d68b843ba73b3aa95cfc27154223244f3a6ca5", [:mix], [{:telemetry, "~> 0.4 or ~> 1.0", [hex: :telemetry, repo: "hexpm", optional: false]}], "hexpm", "2be6954916fdfe4756af3239fb6b6d75d0b8063b5df03ba76fd8a4c87849e180"}, "timex": {:hex, :timex, "3.7.11", "bb95cb4eb1d06e27346325de506bcc6c30f9c6dea40d1ebe390b262fad1862d1", [:mix], [{:combine, "~> 0.10", [hex: :combine, repo: "hexpm", optional: false]}, {:gettext, "~> 0.20", [hex: :gettext, repo: "hexpm", optional: false]}, {:tzdata, "~> 1.1", [hex: :tzdata, repo: "hexpm", optional: false]}], "hexpm", "8b9024f7efbabaf9bd7aa04f65cf8dcd7c9818ca5737677c7b76acbc6a94d1aa"}, - "tls_certificate_check": {:hex, :tls_certificate_check, "1.21.0", "042ab2c0c860652bc5cf69c94e3a31f96676d14682e22ec7813bd173ceff1788", [:rebar3], [{:ssl_verify_fun, "~> 1.1", [hex: :ssl_verify_fun, repo: "hexpm", optional: false]}], "hexpm", "6cee6cffc35a390840d48d463541d50746a7b0e421acaadb833cfc7961e490e7"}, + "tls_certificate_check": {:hex, :tls_certificate_check, "1.22.1", "0f450cc1568a67a65ce5e15df53c53f9a098c3da081c5f126199a72505858dc1", [:rebar3], [{:ssl_verify_fun, "~> 1.1", [hex: :ssl_verify_fun, repo: "hexpm", optional: false]}], "hexpm", "3092be0babdc0e14c2e900542351e066c0fa5a9cf4b3597559ad1e67f07938c0"}, "tzdata": {:hex, :tzdata, "1.1.1", "20c8043476dfda8504952d00adac41c6eda23912278add38edc140ae0c5bcc46", [:mix], [{:hackney, "~> 1.17", [hex: :hackney, repo: "hexpm", optional: false]}], "hexpm", "a69cec8352eafcd2e198dea28a34113b60fdc6cb57eb5ad65c10292a6ba89787"}, - "ueberauth": {:hex, :ueberauth, "0.10.7", "5a31cbe11e7ce5c7484d745dc9e1f11948e89662f8510d03c616de03df581ebd", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "0bccf73e2ffd6337971340832947ba232877aa8122dba4c95be9f729c8987377"}, + "ueberauth": {:hex, :ueberauth, "0.10.8", "ba78fbcbb27d811a6cd06ad851793aaf7d27c3b30c9e95349c2c362b344cd8f0", [:mix], [{:plug, "~> 1.5", [hex: :plug, repo: "hexpm", optional: false]}], "hexpm", "f2d3172e52821375bccb8460e5fa5cb91cfd60b19b636b6e57e9759b6f8c10c1"}, "ueberauth_auth0": {:hex, :ueberauth_auth0, "2.1.0", "0632d5844049fa2f26823f15e1120aa32f27df6f27ce515a4b04641736594bf4", [:mix], [{:oauth2, "~> 2.0", [hex: :oauth2, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.7", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "8d3b30fa27c95c9e82c30c4afb016251405706d2e9627e603c3c9787fd1314fc"}, "ueberauth_discord": {:hex, :ueberauth_discord, "0.7.0", "463f6dfe1ed10a76739331ce8e1dd3600ab611f10524dd828eb3aa50e76e9d43", [:mix], [{:oauth2, "~> 1.0 or ~> 2.0", [hex: :oauth2, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.7", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "d6f98ef91abb4ddceada4b7acba470e0e68c4d2de9735ff2f24172a8e19896b4"}, "ueberauth_github": {:hex, :ueberauth_github, "0.8.3", "1c478629b4c1dae446c68834b69194ad5cead3b6c67c913db6fdf64f37f0328f", [:mix], [{:oauth2, "~> 1.0 or ~> 2.0", [hex: :oauth2, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.7", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "ae0ab2879c32cfa51d7287a48219b262bfdab0b7ec6629f24160564247493cc6"}, "ueberauth_google": {:hex, :ueberauth_google, "0.12.1", "90cf49743588193334f7a00da252f92d90bfd178d766c0e4291361681fafec7d", [:mix], [{:oauth2, "~> 1.0 or ~> 2.0", [hex: :oauth2, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.10.0", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "7f7deacd679b2b66e3bffb68ecc77aa1b5396a0cbac2941815f253128e458c38"}, "ueberauth_microsoft": {:hex, :ueberauth_microsoft, "0.23.0", "5c78e02a83d821ee45f96216bb6140ba688cc79b8b26e7ff438e3abe24615e1d", [:mix], [{:oauth2, "~> 1.0 or ~> 2.0", [hex: :oauth2, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.7", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "0c08d98203e6d3069f30306f09a6cb55b95c2bda94d6f8e90f05bd442ee96b82"}, + "ueberauth_oidc": {:hex, :ueberauth_oidc, "0.1.7", "d610cbe5ef09881dff52126906b130307adcf02791ce158c1847fd50949b283a", [:mix], [{:httpoison, "~> 1.8", [hex: :httpoison, repo: "hexpm", optional: false]}, {:jose, "~> 1.11", [hex: :jose, repo: "hexpm", optional: false]}, {:openid_connect, "~> 0.2.2", [hex: :openid_connect, repo: "hexpm", optional: false]}, {:plug, "~> 1.11", [hex: :plug, repo: "hexpm", optional: false]}, {:ueberauth, "~> 0.7", [hex: :ueberauth, repo: "hexpm", optional: false]}], "hexpm", "34d612f66a5425af4142d6c9dece887c60188c31e1dc113e5ee8cecdc6c5e8a9"}, "unicode_util_compat": {:hex, :unicode_util_compat, "0.7.0", "bc84380c9ab48177092f43ac89e4dfa2c6d62b40b8bd132b1059ecc7232f9a78", [:rebar3], [], "hexpm", "25eee6d67df61960cf6a794239566599b09e17e668d3700247bc498638152521"}, "unsafe": {:hex, :unsafe, "1.0.2", "23c6be12f6c1605364801f4b47007c0c159497d0446ad378b5cf05f1855c0581", [:mix], [], "hexpm", "b485231683c3ab01a9cd44cb4a79f152c6f3bb87358439c6f68791b85c2df675"}, "vega_lite": {:hex, :vega_lite, "0.1.8", "7f6119126ecaf4bc2c1854084370d7091424f5cce4795fbac044eee9963f0752", [:mix], [{:table, "~> 0.1.0", [hex: :table, repo: "hexpm", optional: false]}], "hexpm", "6c8a9271f850612dd8a90de8d1ebd433590ed07ffef76fc2397c240dc04d3fdc"}, diff --git a/package-lock.json b/package-lock.json index d105a6187..d3eb54982 100644 --- a/package-lock.json +++ b/package-lock.json @@ -11,15 +11,15 @@ "@babel/eslint-parser": "7.17.0", "@typescript-eslint/eslint-plugin": "5.12.1", "@typescript-eslint/parser": "5.12.1", - "ember-template-lint": "5.2.0", - "ember-template-lint-plugin-prettier": "4.1.0", + "ember-template-lint": "5.13.0", + "ember-template-lint-plugin-prettier": "5.0.0", "eslint": "8.10.0", "eslint-config-prettier": "8.4.0", "eslint-plugin-ember": "10.5.9", "eslint-plugin-mirego": "^1.0.0", "eslint-plugin-node": "11.1.0", "eslint-plugin-sonarjs": "0.12.0", - "prettier": "2.5.1", + "prettier": "3.2.5", "typescript": "4.7.4" }, "engines": { @@ -741,6 +741,18 @@ "node": ">=6.0.0" } }, + "node_modules/@babel/runtime": { + "version": "7.23.9", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.23.9.tgz", + "integrity": "sha512-0CX6F+BI2s9dkUqr08KFrAIZgNFj75rdBU/DjCyYLIaV/quFjkk6T+EJ2LkZHyZTbEV4L5p97mNkUsHl2wLFAw==", + "dev": true, + "dependencies": { + "regenerator-runtime": "^0.14.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, "node_modules/@ember-data/rfc395-data": { "version": "0.0.4", "resolved": "https://registry.npmjs.org/@ember-data/rfc395-data/-/rfc395-data-0.0.4.tgz", @@ -798,67 +810,67 @@ "dev": true }, "node_modules/@glimmer/global-context": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/global-context/-/global-context-0.83.1.tgz", - "integrity": "sha512-OwlgqpbOJU73EjZOZdftab0fKbtdJ4x/QQeJseL9cvaAUiK3+w52M5ONFxD1T/yPBp2Mf7NCYqA/uL8tRbzY2A==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/global-context/-/global-context-0.84.3.tgz", + "integrity": "sha512-8Oy9Wg5IZxMEeAnVmzD2NkObf89BeHoFSzJgJROE/deutd3rxg83mvlOez4zBBGYwnTb+VGU2LYRpet92egJjA==", "dev": true, "dependencies": { "@glimmer/env": "^0.1.7" } }, "node_modules/@glimmer/interfaces": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/interfaces/-/interfaces-0.83.1.tgz", - "integrity": "sha512-rjAztghzX97v8I4rk3+NguM3XGYcFjc/GbJ8qrEj19KF2lUDoDBW1sB7f0tov3BD5HlrGXei/vOh4+DHfjeB5w==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/interfaces/-/interfaces-0.84.3.tgz", + "integrity": "sha512-dk32ykoNojt0mvEaIW6Vli5MGTbQo58uy3Epj7ahCgTHmWOKuw/0G83f2UmFprRwFx689YTXG38I/vbpltEjzg==", "dev": true, "dependencies": { "@simple-dom/interface": "^1.4.0" } }, "node_modules/@glimmer/reference": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/reference/-/reference-0.83.1.tgz", - "integrity": "sha512-BThEwDlMkJB1WBPWDrww+VxgGyDbwxh5FFPvGhkovvCZnCb7fAMUCt9pi6CUZtviugkWOBFtE9P4eZZbOLkXeg==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/reference/-/reference-0.84.3.tgz", + "integrity": "sha512-lV+p/aWPVC8vUjmlvYVU7WQJsLh319SdXuAWoX/SE3pq340BJlAJiEcAc6q52y9JNhT57gMwtjMX96W5Xcx/qw==", "dev": true, "dependencies": { "@glimmer/env": "^0.1.7", - "@glimmer/global-context": "0.83.1", - "@glimmer/interfaces": "0.83.1", - "@glimmer/util": "0.83.1", - "@glimmer/validator": "0.83.1" + "@glimmer/global-context": "0.84.3", + "@glimmer/interfaces": "0.84.3", + "@glimmer/util": "0.84.3", + "@glimmer/validator": "0.84.3" } }, "node_modules/@glimmer/syntax": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/syntax/-/syntax-0.83.1.tgz", - "integrity": "sha512-n3vEd0GtjtgkOsd2gqkSimp8ecqq5KrHyana/s1XJZvVAPD5rMWT9WvAVWG8XAktns8BxjwLIUoj/vkOfA+eHg==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/syntax/-/syntax-0.84.3.tgz", + "integrity": "sha512-ioVbTic6ZisLxqTgRBL2PCjYZTFIwobifCustrozRU2xGDiYvVIL0vt25h2c1ioDsX59UgVlDkIK4YTAQQSd2A==", "dev": true, "dependencies": { - "@glimmer/interfaces": "0.83.1", - "@glimmer/util": "0.83.1", + "@glimmer/interfaces": "0.84.3", + "@glimmer/util": "0.84.3", "@handlebars/parser": "~2.0.0", "simple-html-tokenizer": "^0.5.11" } }, "node_modules/@glimmer/util": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/util/-/util-0.83.1.tgz", - "integrity": "sha512-amvjtl9dvrkxsoitXAly9W5NUaLIE3A2J2tWhBWIL1Z6DOFotfX7ytIosOIcPhJLZCtiXPHzMutQRv0G/MSMsA==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/util/-/util-0.84.3.tgz", + "integrity": "sha512-qFkh6s16ZSRuu2rfz3T4Wp0fylFj3HBsONGXQcrAdZjdUaIS6v3pNj6mecJ71qRgcym9Hbaq/7/fefIwECUiKw==", "dev": true, "dependencies": { "@glimmer/env": "0.1.7", - "@glimmer/interfaces": "0.83.1", + "@glimmer/interfaces": "0.84.3", "@simple-dom/interface": "^1.4.0" } }, "node_modules/@glimmer/validator": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/validator/-/validator-0.83.1.tgz", - "integrity": "sha512-LaILSNnQgDHZpaUsfjVndbS1JfVn0xdTlJdFJblPbhoVklOBSReZVekens3EQ6xOr3BC612sRm1hBnEPixOY6A==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/validator/-/validator-0.84.3.tgz", + "integrity": "sha512-RTBV4TokUB0vI31UC7ikpV7lOYpWUlyqaKV//pRC4pexYMlmqnVhkFrdiimB/R1XyNdUOQUmnIAcdic39NkbhQ==", "dev": true, "dependencies": { "@glimmer/env": "^0.1.7", - "@glimmer/global-context": "0.83.1" + "@glimmer/global-context": "0.84.3" } }, "node_modules/@handlebars/parser": { @@ -916,17 +928,17 @@ } }, "node_modules/@lint-todo/utils": { - "version": "13.1.0", - "resolved": "https://registry.npmjs.org/@lint-todo/utils/-/utils-13.1.0.tgz", - "integrity": "sha512-uzcZPIPH7hcs+hKMiHfp58MosJpI9sTTgl1pGYau4zq34q1ppswJ6nLeohv/cDhqEBrHjtvldt8zDnVJXRvBlA==", + "version": "13.1.1", + "resolved": "https://registry.npmjs.org/@lint-todo/utils/-/utils-13.1.1.tgz", + "integrity": "sha512-F5z53uvRIF4dYfFfJP3a2Cqg+4P1dgJchJsFnsZE0eZp0LK8X7g2J0CsJHRgns+skpXOlM7n5vFGwkWCWj8qJg==", "dev": true, "dependencies": { - "@types/eslint": "^7.2.13", + "@types/eslint": "^8.4.9", "find-up": "^5.0.0", "fs-extra": "^9.1.0", "proper-lockfile": "^4.1.2", "slash": "^3.0.0", - "tslib": "^2.4.0", + "tslib": "^2.4.1", "upath": "^2.0.1" }, "engines": { @@ -1004,9 +1016,9 @@ } }, "node_modules/@lint-todo/utils/node_modules/tslib": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.0.tgz", - "integrity": "sha512-336iVw3rtn2BUK7ORdIAHTyxHGRIHVReokCR3XjbckJMK7ms8FysBfhLR8IXnAgy7T0PTPNBWKiH514FOW/WSg==", + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==", "dev": true }, "node_modules/@lint-todo/utils/node_modules/yocto-queue": { @@ -1056,6 +1068,18 @@ "node": ">= 8" } }, + "node_modules/@prettier/sync": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/@prettier/sync/-/sync-0.2.1.tgz", + "integrity": "sha512-7ls1R6//+GPYD9vof1XaL5psViv83CwpdwlS8oUkWldYgbPhzZ3WgxIQMWqGyBmWPmoBfQg8C7jj7KI/ZuDHhQ==", + "dev": true, + "funding": { + "url": "https://github.com/prettier/prettier-synchronized?sponsor=1" + }, + "peerDependencies": { + "prettier": "^3.0.0" + } + }, "node_modules/@simple-dom/interface": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/@simple-dom/interface/-/interface-1.4.0.tgz", @@ -1063,9 +1087,9 @@ "dev": true }, "node_modules/@types/eslint": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-7.29.0.tgz", - "integrity": "sha512-VNcvioYDH8/FxaeTKkM4/TiTwt6pBV9E3OfGmvaw8tPl0rrHCJ4Ll15HRT+pMiFAf/MLQvAzC+6RzUMEL9Ceng==", + "version": "8.56.3", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.56.3.tgz", + "integrity": "sha512-PvSf1wfv2wJpVIFUMSb+i4PvqNYkB9Rkp9ZDO3oaWzq4SKhsQk4mrMBr3ZH06I0hKrVGLBacmgl8JM4WVjb9dg==", "dev": true, "dependencies": { "@types/estree": "*", @@ -1073,9 +1097,9 @@ } }, "node_modules/@types/estree": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.0.tgz", - "integrity": "sha512-WulqXMDUTYAXCjZnk6JtIHPigp55cVtDgDrO2gHRwhyJto21+1zbVCtOYB2L1F9w4qCQ0rOGWBnBe0FNTiEJIQ==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", "dev": true }, "node_modules/@types/json-schema": { @@ -1091,9 +1115,9 @@ "dev": true }, "node_modules/@types/symlink-or-copy": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@types/symlink-or-copy/-/symlink-or-copy-1.2.0.tgz", - "integrity": "sha512-Lja2xYuuf2B3knEsga8ShbOdsfNOtzT73GyJmZyY7eGl2+ajOqrs8yM5ze0fsSoYwvA6bw7/Qr7OZ7PEEmYwWg==", + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@types/symlink-or-copy/-/symlink-or-copy-1.2.2.tgz", + "integrity": "sha512-MQ1AnmTLOncwEf9IVU+B2e4Hchrku5N67NkgcAHW0p3sdzPe0FNMANxEm6OJUzPniEQGkeT3OROLlCwZJLWFZA==", "dev": true }, "node_modules/@typescript-eslint/eslint-plugin": { @@ -1438,19 +1462,38 @@ "dev": true }, "node_modules/aria-query": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.1.3.tgz", - "integrity": "sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==", + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", "dev": true, "dependencies": { - "deep-equal": "^2.0.5" + "dequal": "^2.0.3" + } + }, + "node_modules/array-buffer-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz", + "integrity": "sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "is-array-buffer": "^3.0.4" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" } }, "node_modules/array-equal": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.0.tgz", - "integrity": "sha512-H3LU5RLiSsGXPhN+Nipar0iR0IofH+8r89G2y1tBKxQ/agagKyAjhkAFDRBfodP2caPrNKHpAWNIM/c9yeL7uA==", - "dev": true + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.2.tgz", + "integrity": "sha512-gUHx76KtnhEgB3HOuFYiCm3FIdEs6ocM2asHvNTkfu/Y09qQVrrVVaOKENmS2KkSaGoxgXNqC+ZVtR/n0MOkSA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } }, "node_modules/array-union": { "version": "2.1.0", @@ -1461,6 +1504,28 @@ "node": ">=8" } }, + "node_modules/arraybuffer.prototype.slice": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz", + "integrity": "sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.2.1", + "get-intrinsic": "^1.2.3", + "is-array-buffer": "^3.0.4", + "is-shared-array-buffer": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/async": { "version": "2.6.4", "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", @@ -1556,10 +1621,13 @@ } }, "node_modules/available-typed-arrays": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz", - "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", "dev": true, + "dependencies": { + "possible-typed-array-names": "^1.0.0" + }, "engines": { "node": ">= 0.4" }, @@ -2025,13 +2093,19 @@ } }, "node_modules/call-bind": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", - "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", "dev": true, "dependencies": { - "function-bind": "^1.1.1", - "get-intrinsic": "^1.0.2" + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -2163,9 +2237,9 @@ } }, "node_modules/cli-spinners": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", - "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==", + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", "dev": true, "engines": { "node": ">=6" @@ -2275,10 +2349,13 @@ } }, "node_modules/date-fns": { - "version": "2.29.3", - "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.29.3.tgz", - "integrity": "sha512-dDCnyH2WnnKusqvZZ6+jA1O51Ibt8ZMRNkDZdyAyK4YfbDwa/cEmuztzG5pk6hqlp9aSBPYcjOlktquahGwGeA==", + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.30.0.tgz", + "integrity": "sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==", "dev": true, + "dependencies": { + "@babel/runtime": "^7.21.0" + }, "engines": { "node": ">=0.11" }, @@ -2304,34 +2381,6 @@ } } }, - "node_modules/deep-equal": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.2.0.tgz", - "integrity": "sha512-RdpzE0Hv4lhowpIUKKMJfeH6C1pXdtT1/it80ubgWqwI3qpuxUBpC1S4hnHg+zjnuOoDkzUtUCEEkG+XG5l3Mw==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "es-get-iterator": "^1.1.2", - "get-intrinsic": "^1.1.3", - "is-arguments": "^1.1.1", - "is-array-buffer": "^3.0.1", - "is-date-object": "^1.0.5", - "is-regex": "^1.1.4", - "is-shared-array-buffer": "^1.0.2", - "isarray": "^2.0.5", - "object-is": "^1.1.5", - "object-keys": "^1.1.1", - "object.assign": "^4.1.4", - "regexp.prototype.flags": "^1.4.3", - "side-channel": "^1.0.4", - "which-boxed-primitive": "^1.0.2", - "which-collection": "^1.0.1", - "which-typed-array": "^1.1.9" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/deep-is": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", @@ -2350,12 +2399,30 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", + "dev": true, + "dependencies": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/define-properties": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz", - "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", "dev": true, "dependencies": { + "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" }, @@ -2366,6 +2433,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, "node_modules/dir-glob": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", @@ -2440,9 +2516,9 @@ } }, "node_modules/ember-cli-version-checker/node_modules/semver": { - "version": "7.3.8", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.8.tgz", - "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "dependencies": { "lru-cache": "^6.0.0" @@ -2461,9 +2537,9 @@ "dev": true }, "node_modules/ember-template-imports": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/ember-template-imports/-/ember-template-imports-3.4.1.tgz", - "integrity": "sha512-KXnBFTAVxCfXnSCUgd/iuic9ajWbmFkRUBEeorJAMqxvougsPoK22s5ygE9O3GnzYdPpMwn+8v+/NAGy8HRBGA==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/ember-template-imports/-/ember-template-imports-3.4.2.tgz", + "integrity": "sha512-OS8TUVG2kQYYwP3netunLVfeijPoOKIs1SvPQRTNOQX4Pu8xGGBEZmrv0U1YTnQn12Eg+p6w/0UdGbUnITjyzw==", "dev": true, "dependencies": { "babel-import-util": "^0.2.0", @@ -2481,29 +2557,29 @@ } }, "node_modules/ember-template-lint": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ember-template-lint/-/ember-template-lint-5.2.0.tgz", - "integrity": "sha512-A0wQ1cmaTshIiVhl90QFiaTp1aLw0jy8/uUNWNwD/0PnnaKAi4Oz/wbfjKX/4ZKnhFHgjTFeVyeve1ItoqR7cg==", - "dev": true, - "dependencies": { - "@lint-todo/utils": "^13.0.3", - "aria-query": "^5.0.2", - "chalk": "^5.1.2", - "ci-info": "^3.7.0", - "date-fns": "^2.29.2", - "ember-template-imports": "^3.4.0", - "ember-template-recast": "^6.1.3", + "version": "5.13.0", + "resolved": "https://registry.npmjs.org/ember-template-lint/-/ember-template-lint-5.13.0.tgz", + "integrity": "sha512-AYxz9S9fVZfHPmTsymc7NwsD7FVmDUZyfC+KYpxDlK0wic7JSQx2FNQNqQSBFRLOuzn7VQ0/+1pX6DGqKDGswg==", + "dev": true, + "dependencies": { + "@lint-todo/utils": "^13.1.1", + "aria-query": "^5.3.0", + "chalk": "^5.3.0", + "ci-info": "^3.8.0", + "date-fns": "^2.30.0", + "ember-template-imports": "^3.4.2", + "ember-template-recast": "^6.1.4", "eslint-formatter-kakoune": "^1.0.0", "find-up": "^6.3.0", "fuse.js": "^6.5.3", "get-stdin": "^9.0.0", - "globby": "^13.1.2", + "globby": "^13.2.2", "is-glob": "^4.0.3", - "language-tags": "^1.0.6", + "language-tags": "^1.0.8", "micromatch": "^4.0.5", - "resolve": "^1.22.1", + "resolve": "^1.22.3", "v8-compile-cache": "^2.3.0", - "yargs": "^17.5.1" + "yargs": "^17.7.2" }, "bin": { "ember-template-lint": "bin/ember-template-lint.js" @@ -2513,25 +2589,26 @@ } }, "node_modules/ember-template-lint-plugin-prettier": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ember-template-lint-plugin-prettier/-/ember-template-lint-plugin-prettier-4.1.0.tgz", - "integrity": "sha512-LGGn1SXS5vhH60Tp37D4yG8z9ftVMCSSBn5qst+7rjMtLAZuZYwq+q+fs46op745ds0NNtaj+3OzSpBj+RVL2w==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ember-template-lint-plugin-prettier/-/ember-template-lint-plugin-prettier-5.0.0.tgz", + "integrity": "sha512-aXUYM4yuIdPZ80+AsAU8QBwGSJJ/aAkRsNcQ5vI5HmXiBjzHlDc/ZhmP6iVcYuCmoA/3iKcssMAYwIDbuby4pg==", "dev": true, "dependencies": { + "@prettier/sync": "^0.2.1", "prettier-linter-helpers": "^1.0.0" }, "engines": { - "node": "^12.22.0 || ^14.17.0 || >=16.0.0" + "node": "^16.0.0 || ^18.0.0 || >=20.0.0" }, "peerDependencies": { "ember-template-lint": ">= 4.0.0", - "prettier": ">= 1.18.1" + "prettier": ">= 3.0.0" } }, "node_modules/ember-template-lint/node_modules/chalk": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.2.0.tgz", - "integrity": "sha512-ree3Gqw/nazQAPuJJEy+avdl7QfZMcUvmHIKgEZkGL+xOBzRvup5Hxo6LHuMceSxOabuJLJm5Yp/92R9eMmMvA==", + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.3.0.tgz", + "integrity": "sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==", "dev": true, "engines": { "node": "^12.17.0 || ^14.13 || >=16.0.0" @@ -2541,14 +2618,14 @@ } }, "node_modules/ember-template-recast": { - "version": "6.1.3", - "resolved": "https://registry.npmjs.org/ember-template-recast/-/ember-template-recast-6.1.3.tgz", - "integrity": "sha512-45lkfjrWlrMPlOd5rLFeQeePZwAvcS//x1x15kaiQTlqQdYWiYNXwbpWHqV+p9fXY6bEjl6EbyPhG/zBkgh8MA==", + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/ember-template-recast/-/ember-template-recast-6.1.4.tgz", + "integrity": "sha512-fCh+rOK6z+/tsdkTbOE+e7f84P6ObnIRQrCCrnu21E4X05hPeradikIkRMhJdxn4NWrxitfZskQDd37TR/lsNQ==", "dev": true, "dependencies": { - "@glimmer/reference": "^0.83.1", - "@glimmer/syntax": "^0.83.1", - "@glimmer/validator": "^0.83.0", + "@glimmer/reference": "^0.84.3", + "@glimmer/syntax": "^0.84.3", + "@glimmer/validator": "^0.84.3", "async-promise-queue": "^1.0.5", "colors": "^1.4.0", "commander": "^8.3.0", @@ -2556,7 +2633,7 @@ "ora": "^5.4.0", "slash": "^3.0.0", "tmp": "^0.2.1", - "workerpool": "^6.1.5" + "workerpool": "^6.4.0" }, "bin": { "ember-template-recast": "lib/bin.js" @@ -2598,44 +2675,52 @@ "dev": true }, "node_modules/es-abstract": { - "version": "1.21.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.21.1.tgz", - "integrity": "sha512-QudMsPOz86xYz/1dG1OuGBKOELjCh99IIWHLzy5znUB6j8xG2yMA7bfTV86VSqKF+Y/H08vQPR+9jyXpuC6hfg==", - "dev": true, - "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "es-set-tostringtag": "^2.0.1", + "version": "1.22.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.22.4.tgz", + "integrity": "sha512-vZYJlk2u6qHYxBOTjAeg7qUxHdNfih64Uu2J8QqWgXZ2cri0ZpJAkzDUK/q593+mvKwlxyaxr6F1Q+3LKoQRgg==", + "dev": true, + "dependencies": { + "array-buffer-byte-length": "^1.0.1", + "arraybuffer.prototype.slice": "^1.0.3", + "available-typed-arrays": "^1.0.6", + "call-bind": "^1.0.7", + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "es-set-tostringtag": "^2.0.2", "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "function.prototype.name": "^1.1.5", - "get-intrinsic": "^1.1.3", - "get-symbol-description": "^1.0.0", + "function.prototype.name": "^1.1.6", + "get-intrinsic": "^1.2.4", + "get-symbol-description": "^1.0.2", "globalthis": "^1.0.3", "gopd": "^1.0.1", - "has": "^1.0.3", - "has-property-descriptors": "^1.0.0", + "has-property-descriptors": "^1.0.2", "has-proto": "^1.0.1", "has-symbols": "^1.0.3", - "internal-slot": "^1.0.4", - "is-array-buffer": "^3.0.1", + "hasown": "^2.0.1", + "internal-slot": "^1.0.7", + "is-array-buffer": "^3.0.4", "is-callable": "^1.2.7", "is-negative-zero": "^2.0.2", "is-regex": "^1.1.4", "is-shared-array-buffer": "^1.0.2", "is-string": "^1.0.7", - "is-typed-array": "^1.1.10", + "is-typed-array": "^1.1.13", "is-weakref": "^1.0.2", - "object-inspect": "^1.12.2", + "object-inspect": "^1.13.1", "object-keys": "^1.1.1", - "object.assign": "^4.1.4", - "regexp.prototype.flags": "^1.4.3", - "safe-regex-test": "^1.0.0", - "string.prototype.trimend": "^1.0.6", - "string.prototype.trimstart": "^1.0.6", + "object.assign": "^4.1.5", + "regexp.prototype.flags": "^1.5.2", + "safe-array-concat": "^1.1.0", + "safe-regex-test": "^1.0.3", + "string.prototype.trim": "^1.2.8", + "string.prototype.trimend": "^1.0.7", + "string.prototype.trimstart": "^1.0.7", + "typed-array-buffer": "^1.0.1", + "typed-array-byte-length": "^1.0.0", + "typed-array-byte-offset": "^1.0.0", "typed-array-length": "^1.0.4", "unbox-primitive": "^1.0.2", - "which-typed-array": "^1.1.9" + "which-typed-array": "^1.1.14" }, "engines": { "node": ">= 0.4" @@ -2644,35 +2729,36 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/es-get-iterator": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/es-get-iterator/-/es-get-iterator-1.1.3.tgz", - "integrity": "sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==", + "node_modules/es-define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", + "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", - "has-symbols": "^1.0.3", - "is-arguments": "^1.1.1", - "is-map": "^2.0.2", - "is-set": "^2.0.2", - "is-string": "^1.0.7", - "isarray": "^2.0.5", - "stop-iteration-iterator": "^1.0.0" + "get-intrinsic": "^1.2.4" }, - "funding": { - "url": "https://github.com/sponsors/ljharb" + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "dev": true, + "engines": { + "node": ">= 0.4" } }, "node_modules/es-set-tostringtag": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz", - "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", "dev": true, "dependencies": { - "get-intrinsic": "^1.1.3", - "has": "^1.0.3", - "has-tostringtag": "^1.0.0" + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" }, "engines": { "node": ">= 0.4" @@ -3119,9 +3205,9 @@ "dev": true }, "node_modules/fast-glob": { - "version": "3.2.12", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.12.tgz", - "integrity": "sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==", + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", + "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", "dev": true, "dependencies": { "@nodelib/fs.stat": "^2.0.2", @@ -3294,21 +3380,24 @@ "dev": true }, "node_modules/function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", - "dev": true + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", + "dev": true, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } }, "node_modules/function.prototype.name": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", - "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.6.tgz", + "integrity": "sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==", "dev": true, "dependencies": { "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.0", - "functions-have-names": "^1.2.2" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "functions-have-names": "^1.2.3" }, "engines": { "node": ">= 0.4" @@ -3361,14 +3450,19 @@ } }, "node_modules/get-intrinsic": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.0.tgz", - "integrity": "sha512-L049y6nFOuom5wGyRc3/gdTLO94dySVKRACj1RmJZBQXlbTMhtNIgkWkUHq+jYmZvKf14EW1EoJnnjbmoHij0Q==", + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", "dev": true, "dependencies": { - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.3" + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3387,13 +3481,14 @@ } }, "node_modules/get-symbol-description": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", - "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.2.tgz", + "integrity": "sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" + "call-bind": "^1.0.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4" }, "engines": { "node": ">= 0.4" @@ -3457,14 +3552,14 @@ } }, "node_modules/globby": { - "version": "13.1.3", - "resolved": "https://registry.npmjs.org/globby/-/globby-13.1.3.tgz", - "integrity": "sha512-8krCNHXvlCgHDpegPzleMq07yMYTO2sXKASmZmquEYWEmCx6J5UTRbp5RwMJkTJGtcQ44YpiUYUiN0b9mzy8Bw==", + "version": "13.2.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-13.2.2.tgz", + "integrity": "sha512-Y1zNGV+pzQdh7H39l9zgB4PJqjRNqydvdYCDG4HFXM4XuvSaQQlEc91IU1yALL8gUTDomgBAfz3XJdmUS+oo0w==", "dev": true, "dependencies": { "dir-glob": "^3.0.1", - "fast-glob": "^3.2.11", - "ignore": "^5.2.0", + "fast-glob": "^3.3.0", + "ignore": "^5.2.4", "merge2": "^1.4.1", "slash": "^4.0.0" }, @@ -3500,23 +3595,11 @@ } }, "node_modules/graceful-fs": { - "version": "4.2.10", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", - "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "dev": true }, - "node_modules/has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "dev": true, - "dependencies": { - "function-bind": "^1.1.1" - }, - "engines": { - "node": ">= 0.4.0" - } - }, "node_modules/has-bigints": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", @@ -3536,21 +3619,21 @@ } }, "node_modules/has-property-descriptors": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz", - "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", "dev": true, "dependencies": { - "get-intrinsic": "^1.1.1" + "es-define-property": "^1.0.0" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/has-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz", - "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", "dev": true, "engines": { "node": ">= 0.4" @@ -3572,12 +3655,12 @@ } }, "node_modules/has-tostringtag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", - "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", "dev": true, "dependencies": { - "has-symbols": "^1.0.2" + "has-symbols": "^1.0.3" }, "engines": { "node": ">= 0.4" @@ -3610,6 +3693,18 @@ "resolve": "^1.10.0" } }, + "node_modules/hasown": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.1.tgz", + "integrity": "sha512-1/th4MHjnwncwXsIW6QMzlvYL9kG5e/CpVvLRZe4XPa8TOUNbCELqmvhDmnkNsAjwaG4+I8gJJL0JBvTTLO9qA==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.2" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/heimdalljs": { "version": "0.2.6", "resolved": "https://registry.npmjs.org/heimdalljs/-/heimdalljs-0.2.6.tgz", @@ -3671,9 +3766,9 @@ ] }, "node_modules/ignore": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz", - "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz", + "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", "dev": true, "engines": { "node": ">= 4" @@ -3718,27 +3813,27 @@ "dev": true }, "node_modules/internal-slot": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz", - "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.7.tgz", + "integrity": "sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==", "dev": true, "dependencies": { - "get-intrinsic": "^1.2.0", - "has": "^1.0.3", + "es-errors": "^1.3.0", + "hasown": "^2.0.0", "side-channel": "^1.0.4" }, "engines": { "node": ">= 0.4" } }, - "node_modules/is-arguments": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz", - "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==", + "node_modules/is-array-buffer": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.4.tgz", + "integrity": "sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==", "dev": true, "dependencies": { "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" + "get-intrinsic": "^1.2.1" }, "engines": { "node": ">= 0.4" @@ -3747,20 +3842,6 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-array-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.1.tgz", - "integrity": "sha512-ASfLknmY8Xa2XtB4wmbz13Wu202baeA18cJBCeCy0wXUHZF0IPyVEXqKEcd+t2fNSLLL1vC6k7lxZEojNbISXQ==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", - "is-typed-array": "^1.1.10" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-bigint": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", @@ -3802,12 +3883,12 @@ } }, "node_modules/is-core-module": { - "version": "2.11.0", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz", - "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==", + "version": "2.13.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.1.tgz", + "integrity": "sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==", "dev": true, "dependencies": { - "has": "^1.0.3" + "hasown": "^2.0.0" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3867,19 +3948,10 @@ "node": ">=8" } }, - "node_modules/is-map": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.2.tgz", - "integrity": "sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-negative-zero": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", - "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", "dev": true, "engines": { "node": ">= 0.4" @@ -3928,22 +4000,16 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-set": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.2.tgz", - "integrity": "sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-shared-array-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", - "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", "dev": true, "dependencies": { - "call-bind": "^1.0.2" + "call-bind": "^1.0.7" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -3980,16 +4046,12 @@ } }, "node_modules/is-typed-array": { - "version": "1.1.10", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz", - "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==", + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", "dev": true, "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0" + "which-typed-array": "^1.1.14" }, "engines": { "node": ">= 0.4" @@ -4010,15 +4072,6 @@ "url": "https://github.com/sponsors/sindresorhus" } }, - "node_modules/is-weakmap": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.1.tgz", - "integrity": "sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/is-weakref": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", @@ -4031,23 +4084,10 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/is-weakset": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.2.tgz", - "integrity": "sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==", - "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", "dev": true }, "node_modules/isexe": { @@ -4068,12 +4108,6 @@ "node": ">=0.10.0" } }, - "node_modules/isobject/node_modules/isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", - "dev": true - }, "node_modules/istextorbinary": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/istextorbinary/-/istextorbinary-2.1.0.tgz", @@ -4188,12 +4222,6 @@ "isobject": "^2.0.0" } }, - "node_modules/line-column/node_modules/isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", - "dev": true - }, "node_modules/locate-path": { "version": "7.2.0", "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-7.2.0.tgz", @@ -4426,26 +4454,10 @@ } }, "node_modules/object-inspect": { - "version": "1.12.3", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz", - "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==", - "dev": true, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, - "node_modules/object-is": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.5.tgz", - "integrity": "sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw==", + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", + "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==", "dev": true, - "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3" - }, - "engines": { - "node": ">= 0.4" - }, "funding": { "url": "https://github.com/sponsors/ljharb" } @@ -4460,13 +4472,13 @@ } }, "node_modules/object.assign": { - "version": "4.1.4", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", - "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", "has-symbols": "^1.0.3", "object-keys": "^1.1.1" }, @@ -4702,6 +4714,15 @@ "url": "https://github.com/sponsors/jonschlinkert" } }, + "node_modules/possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "dev": true, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", @@ -4712,15 +4733,18 @@ } }, "node_modules/prettier": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.5.1.tgz", - "integrity": "sha512-vBZcPRUR5MZJwoyi3ZoyQlc1rXeEck8KgeC9AwwOn+exuxLxq5toTRDTSaVrXHxelDMHy9zlicw8u66yxoSUFg==", + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.2.5.tgz", + "integrity": "sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A==", "dev": true, "bin": { - "prettier": "bin-prettier.js" + "prettier": "bin/prettier.cjs" }, "engines": { - "node": ">=10.13.0" + "node": ">=14" + }, + "funding": { + "url": "https://github.com/prettier/prettier?sponsor=1" } }, "node_modules/prettier-linter-helpers": { @@ -4817,9 +4841,9 @@ } }, "node_modules/readable-stream": { - "version": "3.6.1", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.1.tgz", - "integrity": "sha512-+rQmrWMYGA90yenhTYsLWAsLsqVC8osOw6PKE1HDYiO0gdPeKe/xDHNzIAIn4C91YQ6oenEhfYqqc1883qHbjQ==", + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", "dev": true, "dependencies": { "inherits": "^2.0.3", @@ -4830,15 +4854,22 @@ "node": ">= 6" } }, + "node_modules/regenerator-runtime": { + "version": "0.14.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz", + "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==", + "dev": true + }, "node_modules/regexp.prototype.flags": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", - "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz", + "integrity": "sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "functions-have-names": "^1.2.2" + "call-bind": "^1.0.6", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "set-function-name": "^2.0.1" }, "engines": { "node": ">= 0.4" @@ -4878,12 +4909,12 @@ } }, "node_modules/resolve": { - "version": "1.22.1", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", - "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", "dev": true, "dependencies": { - "is-core-module": "^2.9.0", + "is-core-module": "^2.13.0", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" }, @@ -4995,6 +5026,30 @@ "queue-microtask": "^1.2.2" } }, + "node_modules/safe-array-concat": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.0.tgz", + "integrity": "sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.5", + "get-intrinsic": "^1.2.2", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + }, + "engines": { + "node": ">=0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/safe-array-concat/node_modules/isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true + }, "node_modules/safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", @@ -5003,15 +5058,18 @@ "peer": true }, "node_modules/safe-regex-test": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz", - "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.3.tgz", + "integrity": "sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", "is-regex": "^1.1.4" }, + "engines": { + "node": ">= 0.4" + }, "funding": { "url": "https://github.com/sponsors/ljharb" } @@ -5025,6 +5083,38 @@ "semver": "bin/semver.js" } }, + "node_modules/set-function-length": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.1.tgz", + "integrity": "sha512-j4t6ccc+VsKwYHso+kElc5neZpjtq9EnRICFZtWyBsLojhmeF/ZBd/elqm22WJh/BziDe/SBiOeAt0m2mfLD0g==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.2", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.3", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.1" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "dependencies": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + } + }, "node_modules/shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -5047,14 +5137,18 @@ } }, "node_modules/side-channel": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", - "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.5.tgz", + "integrity": "sha512-QcgiIWV4WV7qWExbN5llt6frQB/lBven9pqliLXfGPB+K9ZYXxDozp0wLkHS24kWCm+6YXH/f0HhnObZnZOBnQ==", "dev": true, "dependencies": { - "call-bind": "^1.0.0", - "get-intrinsic": "^1.0.2", - "object-inspect": "^1.9.0" + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -5142,23 +5236,11 @@ "dev": true }, "node_modules/sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", + "integrity": "sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==", "dev": true }, - "node_modules/stop-iteration-iterator": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.0.0.tgz", - "integrity": "sha512-iCGQj+0l0HOdZ2AEeBADlsRC+vsnDsZsbdSiH1yNSjcfKM7fdpCMfqAL/dwF5BLiw/XhRft/Wax6zQbhq2BcjQ==", - "dev": true, - "dependencies": { - "internal-slot": "^1.0.4" - }, - "engines": { - "node": ">= 0.4" - } - }, "node_modules/string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -5203,47 +5285,65 @@ } }, "node_modules/string.prototype.matchall": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.8.tgz", - "integrity": "sha512-6zOCOcJ+RJAQshcTvXPHoxoQGONa3e/Lqx90wUA+wEzX78sg5Bo+1tQo4N0pohS0erG9qtCqJDjNCQBjeWVxyg==", + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz", + "integrity": "sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==", "dev": true, "dependencies": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4", - "get-intrinsic": "^1.1.3", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "get-intrinsic": "^1.2.1", "has-symbols": "^1.0.3", - "internal-slot": "^1.0.3", - "regexp.prototype.flags": "^1.4.3", + "internal-slot": "^1.0.5", + "regexp.prototype.flags": "^1.5.0", + "set-function-name": "^2.0.0", "side-channel": "^1.0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/string.prototype.trim": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.8.tgz", + "integrity": "sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/string.prototype.trimend": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz", - "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.7.tgz", + "integrity": "sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA==", "dev": true, "dependencies": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" }, "funding": { "url": "https://github.com/sponsors/ljharb" } }, "node_modules/string.prototype.trimstart": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz", - "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.7.tgz", + "integrity": "sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg==", "dev": true, "dependencies": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -5487,15 +5587,74 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/typed-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + } + }, + "node_modules/typed-array-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/typed-array-byte-offset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", + "dev": true, + "dependencies": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, "node_modules/typed-array-length": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz", - "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.5.tgz", + "integrity": "sha512-yMi0PlwuznKHxKmcpoOdeLwxBoVPkqZxd7q2FgMkmD3bNwvF5VW0+UlUQ1k1vmktTu4Yu13Q0RIxEP8+B+wloA==", "dev": true, "dependencies": { - "call-bind": "^1.0.2", + "call-bind": "^1.0.7", "for-each": "^0.3.3", - "is-typed-array": "^1.1.9" + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" }, "funding": { "url": "https://github.com/sponsors/ljharb" @@ -5543,9 +5702,9 @@ } }, "node_modules/universalify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", - "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", "dev": true, "engines": { "node": ">= 10.0.0" @@ -5599,9 +5758,9 @@ } }, "node_modules/validate-peer-dependencies/node_modules/semver": { - "version": "7.3.8", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.8.tgz", - "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "dependencies": { "lru-cache": "^6.0.0" @@ -5664,33 +5823,17 @@ "url": "https://github.com/sponsors/ljharb" } }, - "node_modules/which-collection": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.1.tgz", - "integrity": "sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==", - "dev": true, - "dependencies": { - "is-map": "^2.0.1", - "is-set": "^2.0.1", - "is-weakmap": "^2.0.1", - "is-weakset": "^2.0.1" - }, - "funding": { - "url": "https://github.com/sponsors/ljharb" - } - }, "node_modules/which-typed-array": { - "version": "1.1.9", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.9.tgz", - "integrity": "sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.14.tgz", + "integrity": "sha512-VnXFiIW8yNn9kIHN88xvZ4yOWchftKDsRJ8fEPacX/wl1lOvBrhsJ/OeJCXq7B0AaijRuqgzSKalJoPk+D8MPg==", "dev": true, "dependencies": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", + "available-typed-arrays": "^1.0.6", + "call-bind": "^1.0.5", "for-each": "^0.3.3", "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0", - "is-typed-array": "^1.1.10" + "has-tostringtag": "^1.0.1" }, "engines": { "node": ">= 0.4" @@ -5709,9 +5852,9 @@ } }, "node_modules/workerpool": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.3.1.tgz", - "integrity": "sha512-0x7gJm1rhpn5SPG9NENOxPtbfUZZtK/qOg6gEdSqeDBA3dTeR91RJqSPjccPRCkhNfrnnl/dWxSSj5w9CtdzNA==", + "version": "6.5.1", + "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.5.1.tgz", + "integrity": "sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==", "dev": true }, "node_modules/wrap-ansi": { @@ -5753,9 +5896,9 @@ "dev": true }, "node_modules/yargs": { - "version": "17.7.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.1.tgz", - "integrity": "sha512-cwiTb08Xuv5fqF4AovYacTFNxk62th7LKJ6BL9IGUpTJrWoU7/7WdQGTP2SjKf1dUNBGzDd28p/Yfs/GI6JrLw==", + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dev": true, "dependencies": { "cliui": "^8.0.1", @@ -6369,6 +6512,15 @@ "dev": true, "peer": true }, + "@babel/runtime": { + "version": "7.23.9", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.23.9.tgz", + "integrity": "sha512-0CX6F+BI2s9dkUqr08KFrAIZgNFj75rdBU/DjCyYLIaV/quFjkk6T+EJ2LkZHyZTbEV4L5p97mNkUsHl2wLFAw==", + "dev": true, + "requires": { + "regenerator-runtime": "^0.14.0" + } + }, "@ember-data/rfc395-data": { "version": "0.0.4", "resolved": "https://registry.npmjs.org/@ember-data/rfc395-data/-/rfc395-data-0.0.4.tgz", @@ -6416,67 +6568,67 @@ "dev": true }, "@glimmer/global-context": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/global-context/-/global-context-0.83.1.tgz", - "integrity": "sha512-OwlgqpbOJU73EjZOZdftab0fKbtdJ4x/QQeJseL9cvaAUiK3+w52M5ONFxD1T/yPBp2Mf7NCYqA/uL8tRbzY2A==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/global-context/-/global-context-0.84.3.tgz", + "integrity": "sha512-8Oy9Wg5IZxMEeAnVmzD2NkObf89BeHoFSzJgJROE/deutd3rxg83mvlOez4zBBGYwnTb+VGU2LYRpet92egJjA==", "dev": true, "requires": { "@glimmer/env": "^0.1.7" } }, "@glimmer/interfaces": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/interfaces/-/interfaces-0.83.1.tgz", - "integrity": "sha512-rjAztghzX97v8I4rk3+NguM3XGYcFjc/GbJ8qrEj19KF2lUDoDBW1sB7f0tov3BD5HlrGXei/vOh4+DHfjeB5w==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/interfaces/-/interfaces-0.84.3.tgz", + "integrity": "sha512-dk32ykoNojt0mvEaIW6Vli5MGTbQo58uy3Epj7ahCgTHmWOKuw/0G83f2UmFprRwFx689YTXG38I/vbpltEjzg==", "dev": true, "requires": { "@simple-dom/interface": "^1.4.0" } }, "@glimmer/reference": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/reference/-/reference-0.83.1.tgz", - "integrity": "sha512-BThEwDlMkJB1WBPWDrww+VxgGyDbwxh5FFPvGhkovvCZnCb7fAMUCt9pi6CUZtviugkWOBFtE9P4eZZbOLkXeg==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/reference/-/reference-0.84.3.tgz", + "integrity": "sha512-lV+p/aWPVC8vUjmlvYVU7WQJsLh319SdXuAWoX/SE3pq340BJlAJiEcAc6q52y9JNhT57gMwtjMX96W5Xcx/qw==", "dev": true, "requires": { "@glimmer/env": "^0.1.7", - "@glimmer/global-context": "0.83.1", - "@glimmer/interfaces": "0.83.1", - "@glimmer/util": "0.83.1", - "@glimmer/validator": "0.83.1" + "@glimmer/global-context": "0.84.3", + "@glimmer/interfaces": "0.84.3", + "@glimmer/util": "0.84.3", + "@glimmer/validator": "0.84.3" } }, "@glimmer/syntax": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/syntax/-/syntax-0.83.1.tgz", - "integrity": "sha512-n3vEd0GtjtgkOsd2gqkSimp8ecqq5KrHyana/s1XJZvVAPD5rMWT9WvAVWG8XAktns8BxjwLIUoj/vkOfA+eHg==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/syntax/-/syntax-0.84.3.tgz", + "integrity": "sha512-ioVbTic6ZisLxqTgRBL2PCjYZTFIwobifCustrozRU2xGDiYvVIL0vt25h2c1ioDsX59UgVlDkIK4YTAQQSd2A==", "dev": true, "requires": { - "@glimmer/interfaces": "0.83.1", - "@glimmer/util": "0.83.1", + "@glimmer/interfaces": "0.84.3", + "@glimmer/util": "0.84.3", "@handlebars/parser": "~2.0.0", "simple-html-tokenizer": "^0.5.11" } }, "@glimmer/util": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/util/-/util-0.83.1.tgz", - "integrity": "sha512-amvjtl9dvrkxsoitXAly9W5NUaLIE3A2J2tWhBWIL1Z6DOFotfX7ytIosOIcPhJLZCtiXPHzMutQRv0G/MSMsA==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/util/-/util-0.84.3.tgz", + "integrity": "sha512-qFkh6s16ZSRuu2rfz3T4Wp0fylFj3HBsONGXQcrAdZjdUaIS6v3pNj6mecJ71qRgcym9Hbaq/7/fefIwECUiKw==", "dev": true, "requires": { "@glimmer/env": "0.1.7", - "@glimmer/interfaces": "0.83.1", + "@glimmer/interfaces": "0.84.3", "@simple-dom/interface": "^1.4.0" } }, "@glimmer/validator": { - "version": "0.83.1", - "resolved": "https://registry.npmjs.org/@glimmer/validator/-/validator-0.83.1.tgz", - "integrity": "sha512-LaILSNnQgDHZpaUsfjVndbS1JfVn0xdTlJdFJblPbhoVklOBSReZVekens3EQ6xOr3BC612sRm1hBnEPixOY6A==", + "version": "0.84.3", + "resolved": "https://registry.npmjs.org/@glimmer/validator/-/validator-0.84.3.tgz", + "integrity": "sha512-RTBV4TokUB0vI31UC7ikpV7lOYpWUlyqaKV//pRC4pexYMlmqnVhkFrdiimB/R1XyNdUOQUmnIAcdic39NkbhQ==", "dev": true, "requires": { "@glimmer/env": "^0.1.7", - "@glimmer/global-context": "0.83.1" + "@glimmer/global-context": "0.84.3" } }, "@handlebars/parser": { @@ -6528,17 +6680,17 @@ } }, "@lint-todo/utils": { - "version": "13.1.0", - "resolved": "https://registry.npmjs.org/@lint-todo/utils/-/utils-13.1.0.tgz", - "integrity": "sha512-uzcZPIPH7hcs+hKMiHfp58MosJpI9sTTgl1pGYau4zq34q1ppswJ6nLeohv/cDhqEBrHjtvldt8zDnVJXRvBlA==", + "version": "13.1.1", + "resolved": "https://registry.npmjs.org/@lint-todo/utils/-/utils-13.1.1.tgz", + "integrity": "sha512-F5z53uvRIF4dYfFfJP3a2Cqg+4P1dgJchJsFnsZE0eZp0LK8X7g2J0CsJHRgns+skpXOlM7n5vFGwkWCWj8qJg==", "dev": true, "requires": { - "@types/eslint": "^7.2.13", + "@types/eslint": "^8.4.9", "find-up": "^5.0.0", "fs-extra": "^9.1.0", "proper-lockfile": "^4.1.2", "slash": "^3.0.0", - "tslib": "^2.4.0", + "tslib": "^2.4.1", "upath": "^2.0.1" }, "dependencies": { @@ -6586,9 +6738,9 @@ "dev": true }, "tslib": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.5.0.tgz", - "integrity": "sha512-336iVw3rtn2BUK7ORdIAHTyxHGRIHVReokCR3XjbckJMK7ms8FysBfhLR8IXnAgy7T0PTPNBWKiH514FOW/WSg==", + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.6.2.tgz", + "integrity": "sha512-AEYxH93jGFPn/a2iVAwW87VuUIkR1FVUKB77NwMF7nBTDkDrrT/Hpt/IrCJ0QXhW27jTBDcf5ZY7w6RiqTMw2Q==", "dev": true }, "yocto-queue": { @@ -6625,6 +6777,13 @@ "fastq": "^1.6.0" } }, + "@prettier/sync": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/@prettier/sync/-/sync-0.2.1.tgz", + "integrity": "sha512-7ls1R6//+GPYD9vof1XaL5psViv83CwpdwlS8oUkWldYgbPhzZ3WgxIQMWqGyBmWPmoBfQg8C7jj7KI/ZuDHhQ==", + "dev": true, + "requires": {} + }, "@simple-dom/interface": { "version": "1.4.0", "resolved": "https://registry.npmjs.org/@simple-dom/interface/-/interface-1.4.0.tgz", @@ -6632,9 +6791,9 @@ "dev": true }, "@types/eslint": { - "version": "7.29.0", - "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-7.29.0.tgz", - "integrity": "sha512-VNcvioYDH8/FxaeTKkM4/TiTwt6pBV9E3OfGmvaw8tPl0rrHCJ4Ll15HRT+pMiFAf/MLQvAzC+6RzUMEL9Ceng==", + "version": "8.56.3", + "resolved": "https://registry.npmjs.org/@types/eslint/-/eslint-8.56.3.tgz", + "integrity": "sha512-PvSf1wfv2wJpVIFUMSb+i4PvqNYkB9Rkp9ZDO3oaWzq4SKhsQk4mrMBr3ZH06I0hKrVGLBacmgl8JM4WVjb9dg==", "dev": true, "requires": { "@types/estree": "*", @@ -6642,9 +6801,9 @@ } }, "@types/estree": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.0.tgz", - "integrity": "sha512-WulqXMDUTYAXCjZnk6JtIHPigp55cVtDgDrO2gHRwhyJto21+1zbVCtOYB2L1F9w4qCQ0rOGWBnBe0FNTiEJIQ==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/@types/estree/-/estree-1.0.5.tgz", + "integrity": "sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw==", "dev": true }, "@types/json-schema": { @@ -6660,9 +6819,9 @@ "dev": true }, "@types/symlink-or-copy": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/@types/symlink-or-copy/-/symlink-or-copy-1.2.0.tgz", - "integrity": "sha512-Lja2xYuuf2B3knEsga8ShbOdsfNOtzT73GyJmZyY7eGl2+ajOqrs8yM5ze0fsSoYwvA6bw7/Qr7OZ7PEEmYwWg==", + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/@types/symlink-or-copy/-/symlink-or-copy-1.2.2.tgz", + "integrity": "sha512-MQ1AnmTLOncwEf9IVU+B2e4Hchrku5N67NkgcAHW0p3sdzPe0FNMANxEm6OJUzPniEQGkeT3OROLlCwZJLWFZA==", "dev": true }, "@typescript-eslint/eslint-plugin": { @@ -6880,18 +7039,28 @@ "dev": true }, "aria-query": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.1.3.tgz", - "integrity": "sha512-R5iJ5lkuHybztUfuOAznmboyjWq8O6sqNqtK7CLOqdydi54VNbORp49mb14KbWgG1QD3JFO9hJdZ+y4KutfdOQ==", + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-5.3.0.tgz", + "integrity": "sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A==", + "dev": true, + "requires": { + "dequal": "^2.0.3" + } + }, + "array-buffer-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/array-buffer-byte-length/-/array-buffer-byte-length-1.0.1.tgz", + "integrity": "sha512-ahC5W1xgou+KTXix4sAO8Ki12Q+jf4i0+tmk3sC+zgcynshkHxzpXdImBehiUYKKKDwvfFiJl1tZt6ewscS1Mg==", "dev": true, "requires": { - "deep-equal": "^2.0.5" + "call-bind": "^1.0.5", + "is-array-buffer": "^3.0.4" } }, "array-equal": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.0.tgz", - "integrity": "sha512-H3LU5RLiSsGXPhN+Nipar0iR0IofH+8r89G2y1tBKxQ/agagKyAjhkAFDRBfodP2caPrNKHpAWNIM/c9yeL7uA==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-equal/-/array-equal-1.0.2.tgz", + "integrity": "sha512-gUHx76KtnhEgB3HOuFYiCm3FIdEs6ocM2asHvNTkfu/Y09qQVrrVVaOKENmS2KkSaGoxgXNqC+ZVtR/n0MOkSA==", "dev": true }, "array-union": { @@ -6900,6 +7069,22 @@ "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", "dev": true }, + "arraybuffer.prototype.slice": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.3.tgz", + "integrity": "sha512-bMxMKAjg13EBSVscxTaYA4mRc5t1UAXa2kXiGTNfZ079HIWXEkKmkgFrh/nJqamaLSrXO5H4WFFkPEaLJWbs3A==", + "dev": true, + "requires": { + "array-buffer-byte-length": "^1.0.1", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", + "es-abstract": "^1.22.3", + "es-errors": "^1.2.1", + "get-intrinsic": "^1.2.3", + "is-array-buffer": "^3.0.4", + "is-shared-array-buffer": "^1.0.2" + } + }, "async": { "version": "2.6.4", "resolved": "https://registry.npmjs.org/async/-/async-2.6.4.tgz", @@ -6990,10 +7175,13 @@ "dev": true }, "available-typed-arrays": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.5.tgz", - "integrity": "sha512-DMD0KiN46eipeziST1LPP/STfDU0sufISXmjSgvVsoU2tqxctQeASejWcfNtxYKqETM1UxQ8sp2OrSBWpHY6sw==", - "dev": true + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz", + "integrity": "sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ==", + "dev": true, + "requires": { + "possible-typed-array-names": "^1.0.0" + } }, "babel-import-util": { "version": "0.2.0", @@ -7375,13 +7563,16 @@ } }, "call-bind": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", - "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.7.tgz", + "integrity": "sha512-GHTSNSYICQ7scH7sZ+M2rFopRoLh8t2bLSW6BbgrtLsahOIB5iyAVJf9GjWK3cYTDaMj4XdBpM1cA6pIS0Kv2w==", "dev": true, "requires": { - "function-bind": "^1.1.1", - "get-intrinsic": "^1.0.2" + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.4", + "set-function-length": "^1.2.1" } }, "callsites": { @@ -7470,9 +7661,9 @@ } }, "cli-spinners": { - "version": "2.7.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.7.0.tgz", - "integrity": "sha512-qu3pN8Y3qHNgE2AFweciB1IfMnmZ/fsNTEE+NOFjmGB2F/7rLhnhzppvpCnN4FovtP26k8lHyy9ptEbNwWFLzw==", + "version": "2.9.2", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.9.2.tgz", + "integrity": "sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg==", "dev": true }, "cliui": { @@ -7557,10 +7748,13 @@ } }, "date-fns": { - "version": "2.29.3", - "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.29.3.tgz", - "integrity": "sha512-dDCnyH2WnnKusqvZZ6+jA1O51Ibt8ZMRNkDZdyAyK4YfbDwa/cEmuztzG5pk6hqlp9aSBPYcjOlktquahGwGeA==", - "dev": true + "version": "2.30.0", + "resolved": "https://registry.npmjs.org/date-fns/-/date-fns-2.30.0.tgz", + "integrity": "sha512-fnULvOpxnC5/Vg3NCiWelDsLiUc9bRwAPs/+LfTLNvetFCtCTN+yQz15C/fs4AwX1R9K5GLtLfn8QW+dWisaAw==", + "dev": true, + "requires": { + "@babel/runtime": "^7.21.0" + } }, "debug": { "version": "4.3.4", @@ -7571,31 +7765,6 @@ "ms": "2.1.2" } }, - "deep-equal": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-2.2.0.tgz", - "integrity": "sha512-RdpzE0Hv4lhowpIUKKMJfeH6C1pXdtT1/it80ubgWqwI3qpuxUBpC1S4hnHg+zjnuOoDkzUtUCEEkG+XG5l3Mw==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "es-get-iterator": "^1.1.2", - "get-intrinsic": "^1.1.3", - "is-arguments": "^1.1.1", - "is-array-buffer": "^3.0.1", - "is-date-object": "^1.0.5", - "is-regex": "^1.1.4", - "is-shared-array-buffer": "^1.0.2", - "isarray": "^2.0.5", - "object-is": "^1.1.5", - "object-keys": "^1.1.1", - "object.assign": "^4.1.4", - "regexp.prototype.flags": "^1.4.3", - "side-channel": "^1.0.4", - "which-boxed-primitive": "^1.0.2", - "which-collection": "^1.0.1", - "which-typed-array": "^1.1.9" - } - }, "deep-is": { "version": "0.1.4", "resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz", @@ -7611,16 +7780,34 @@ "clone": "^1.0.2" } }, + "define-data-property": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/define-data-property/-/define-data-property-1.1.4.tgz", + "integrity": "sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==", + "dev": true, + "requires": { + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "gopd": "^1.0.1" + } + }, "define-properties": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.0.tgz", - "integrity": "sha512-xvqAVKGfT1+UAvPwKTVw/njhdQ8ZhXK4lI0bCIuCMrp2up9nPnaDftrLtmpTazqd1o+UY4zgzU+avtMbDP+ldA==", + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/define-properties/-/define-properties-1.2.1.tgz", + "integrity": "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==", "dev": true, "requires": { + "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" } }, + "dequal": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/dequal/-/dequal-2.0.3.tgz", + "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==", + "dev": true + }, "dir-glob": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", @@ -7680,9 +7867,9 @@ }, "dependencies": { "semver": { - "version": "7.3.8", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.8.tgz", - "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -7697,9 +7884,9 @@ "dev": true }, "ember-template-imports": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/ember-template-imports/-/ember-template-imports-3.4.1.tgz", - "integrity": "sha512-KXnBFTAVxCfXnSCUgd/iuic9ajWbmFkRUBEeorJAMqxvougsPoK22s5ygE9O3GnzYdPpMwn+8v+/NAGy8HRBGA==", + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/ember-template-imports/-/ember-template-imports-3.4.2.tgz", + "integrity": "sha512-OS8TUVG2kQYYwP3netunLVfeijPoOKIs1SvPQRTNOQX4Pu8xGGBEZmrv0U1YTnQn12Eg+p6w/0UdGbUnITjyzw==", "dev": true, "requires": { "babel-import-util": "^0.2.0", @@ -7714,57 +7901,58 @@ } }, "ember-template-lint": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ember-template-lint/-/ember-template-lint-5.2.0.tgz", - "integrity": "sha512-A0wQ1cmaTshIiVhl90QFiaTp1aLw0jy8/uUNWNwD/0PnnaKAi4Oz/wbfjKX/4ZKnhFHgjTFeVyeve1ItoqR7cg==", - "dev": true, - "requires": { - "@lint-todo/utils": "^13.0.3", - "aria-query": "^5.0.2", - "chalk": "^5.1.2", - "ci-info": "^3.7.0", - "date-fns": "^2.29.2", - "ember-template-imports": "^3.4.0", - "ember-template-recast": "^6.1.3", + "version": "5.13.0", + "resolved": "https://registry.npmjs.org/ember-template-lint/-/ember-template-lint-5.13.0.tgz", + "integrity": "sha512-AYxz9S9fVZfHPmTsymc7NwsD7FVmDUZyfC+KYpxDlK0wic7JSQx2FNQNqQSBFRLOuzn7VQ0/+1pX6DGqKDGswg==", + "dev": true, + "requires": { + "@lint-todo/utils": "^13.1.1", + "aria-query": "^5.3.0", + "chalk": "^5.3.0", + "ci-info": "^3.8.0", + "date-fns": "^2.30.0", + "ember-template-imports": "^3.4.2", + "ember-template-recast": "^6.1.4", "eslint-formatter-kakoune": "^1.0.0", "find-up": "^6.3.0", "fuse.js": "^6.5.3", "get-stdin": "^9.0.0", - "globby": "^13.1.2", + "globby": "^13.2.2", "is-glob": "^4.0.3", - "language-tags": "^1.0.6", + "language-tags": "^1.0.8", "micromatch": "^4.0.5", - "resolve": "^1.22.1", + "resolve": "^1.22.3", "v8-compile-cache": "^2.3.0", - "yargs": "^17.5.1" + "yargs": "^17.7.2" }, "dependencies": { "chalk": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.2.0.tgz", - "integrity": "sha512-ree3Gqw/nazQAPuJJEy+avdl7QfZMcUvmHIKgEZkGL+xOBzRvup5Hxo6LHuMceSxOabuJLJm5Yp/92R9eMmMvA==", + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-5.3.0.tgz", + "integrity": "sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w==", "dev": true } } }, "ember-template-lint-plugin-prettier": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/ember-template-lint-plugin-prettier/-/ember-template-lint-plugin-prettier-4.1.0.tgz", - "integrity": "sha512-LGGn1SXS5vhH60Tp37D4yG8z9ftVMCSSBn5qst+7rjMtLAZuZYwq+q+fs46op745ds0NNtaj+3OzSpBj+RVL2w==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ember-template-lint-plugin-prettier/-/ember-template-lint-plugin-prettier-5.0.0.tgz", + "integrity": "sha512-aXUYM4yuIdPZ80+AsAU8QBwGSJJ/aAkRsNcQ5vI5HmXiBjzHlDc/ZhmP6iVcYuCmoA/3iKcssMAYwIDbuby4pg==", "dev": true, "requires": { + "@prettier/sync": "^0.2.1", "prettier-linter-helpers": "^1.0.0" } }, "ember-template-recast": { - "version": "6.1.3", - "resolved": "https://registry.npmjs.org/ember-template-recast/-/ember-template-recast-6.1.3.tgz", - "integrity": "sha512-45lkfjrWlrMPlOd5rLFeQeePZwAvcS//x1x15kaiQTlqQdYWiYNXwbpWHqV+p9fXY6bEjl6EbyPhG/zBkgh8MA==", + "version": "6.1.4", + "resolved": "https://registry.npmjs.org/ember-template-recast/-/ember-template-recast-6.1.4.tgz", + "integrity": "sha512-fCh+rOK6z+/tsdkTbOE+e7f84P6ObnIRQrCCrnu21E4X05hPeradikIkRMhJdxn4NWrxitfZskQDd37TR/lsNQ==", "dev": true, "requires": { - "@glimmer/reference": "^0.83.1", - "@glimmer/syntax": "^0.83.1", - "@glimmer/validator": "^0.83.0", + "@glimmer/reference": "^0.84.3", + "@glimmer/syntax": "^0.84.3", + "@glimmer/validator": "^0.84.3", "async-promise-queue": "^1.0.5", "colors": "^1.4.0", "commander": "^8.3.0", @@ -7772,7 +7960,7 @@ "ora": "^5.4.0", "slash": "^3.0.0", "tmp": "^0.2.1", - "workerpool": "^6.1.5" + "workerpool": "^6.4.0" }, "dependencies": { "globby": { @@ -7804,72 +7992,78 @@ "dev": true }, "es-abstract": { - "version": "1.21.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.21.1.tgz", - "integrity": "sha512-QudMsPOz86xYz/1dG1OuGBKOELjCh99IIWHLzy5znUB6j8xG2yMA7bfTV86VSqKF+Y/H08vQPR+9jyXpuC6hfg==", - "dev": true, - "requires": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "es-set-tostringtag": "^2.0.1", + "version": "1.22.4", + "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.22.4.tgz", + "integrity": "sha512-vZYJlk2u6qHYxBOTjAeg7qUxHdNfih64Uu2J8QqWgXZ2cri0ZpJAkzDUK/q593+mvKwlxyaxr6F1Q+3LKoQRgg==", + "dev": true, + "requires": { + "array-buffer-byte-length": "^1.0.1", + "arraybuffer.prototype.slice": "^1.0.3", + "available-typed-arrays": "^1.0.6", + "call-bind": "^1.0.7", + "es-define-property": "^1.0.0", + "es-errors": "^1.3.0", + "es-set-tostringtag": "^2.0.2", "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "function.prototype.name": "^1.1.5", - "get-intrinsic": "^1.1.3", - "get-symbol-description": "^1.0.0", + "function.prototype.name": "^1.1.6", + "get-intrinsic": "^1.2.4", + "get-symbol-description": "^1.0.2", "globalthis": "^1.0.3", "gopd": "^1.0.1", - "has": "^1.0.3", - "has-property-descriptors": "^1.0.0", + "has-property-descriptors": "^1.0.2", "has-proto": "^1.0.1", "has-symbols": "^1.0.3", - "internal-slot": "^1.0.4", - "is-array-buffer": "^3.0.1", + "hasown": "^2.0.1", + "internal-slot": "^1.0.7", + "is-array-buffer": "^3.0.4", "is-callable": "^1.2.7", "is-negative-zero": "^2.0.2", "is-regex": "^1.1.4", "is-shared-array-buffer": "^1.0.2", "is-string": "^1.0.7", - "is-typed-array": "^1.1.10", + "is-typed-array": "^1.1.13", "is-weakref": "^1.0.2", - "object-inspect": "^1.12.2", + "object-inspect": "^1.13.1", "object-keys": "^1.1.1", - "object.assign": "^4.1.4", - "regexp.prototype.flags": "^1.4.3", - "safe-regex-test": "^1.0.0", - "string.prototype.trimend": "^1.0.6", - "string.prototype.trimstart": "^1.0.6", + "object.assign": "^4.1.5", + "regexp.prototype.flags": "^1.5.2", + "safe-array-concat": "^1.1.0", + "safe-regex-test": "^1.0.3", + "string.prototype.trim": "^1.2.8", + "string.prototype.trimend": "^1.0.7", + "string.prototype.trimstart": "^1.0.7", + "typed-array-buffer": "^1.0.1", + "typed-array-byte-length": "^1.0.0", + "typed-array-byte-offset": "^1.0.0", "typed-array-length": "^1.0.4", "unbox-primitive": "^1.0.2", - "which-typed-array": "^1.1.9" + "which-typed-array": "^1.1.14" } }, - "es-get-iterator": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/es-get-iterator/-/es-get-iterator-1.1.3.tgz", - "integrity": "sha512-sPZmqHBe6JIiTfN5q2pEi//TwxmAFHwj/XEuYjTuse78i8KxaqMTTzxPoFKuzRpDpTJ+0NAbpfenkmH2rePtuw==", + "es-define-property": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.0.tgz", + "integrity": "sha512-jxayLKShrEqqzJ0eumQbVhTYQM27CfT1T35+gCgDFoL82JLsXqTJ76zv6A0YLOgEnLUMvLzsDsGIrl8NFpT2gQ==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", - "has-symbols": "^1.0.3", - "is-arguments": "^1.1.1", - "is-map": "^2.0.2", - "is-set": "^2.0.2", - "is-string": "^1.0.7", - "isarray": "^2.0.5", - "stop-iteration-iterator": "^1.0.0" + "get-intrinsic": "^1.2.4" } }, + "es-errors": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz", + "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==", + "dev": true + }, "es-set-tostringtag": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.1.tgz", - "integrity": "sha512-g3OMbtlwY3QewlqAiMLI47KywjWZoEytKr8pf6iTC8uJq5bIAH52Z9pnQ8pVL6whrCto53JZDuUIsifGeLorTg==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/es-set-tostringtag/-/es-set-tostringtag-2.0.3.tgz", + "integrity": "sha512-3T8uNMC3OQTHkFUsFq8r/BwAXLHvU/9O9mE0fBc/MY5iq/8H7ncvO947LmYA6ldWw9Uh8Yhf25zu6n7nML5QWQ==", "dev": true, "requires": { - "get-intrinsic": "^1.1.3", - "has": "^1.0.3", - "has-tostringtag": "^1.0.0" + "get-intrinsic": "^1.2.4", + "has-tostringtag": "^1.0.2", + "hasown": "^2.0.1" } }, "es-to-primitive": { @@ -8199,9 +8393,9 @@ "dev": true }, "fast-glob": { - "version": "3.2.12", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.12.tgz", - "integrity": "sha512-DVj4CQIYYow0BlaelwK1pHl5n5cRSJfM60UA0zK891sVInoPri2Ekj7+e1CT3/3qxXenpI+nBBmQAcJPJgaj4w==", + "version": "3.3.2", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz", + "integrity": "sha512-oX2ruAFQwf/Orj8m737Y5adxDQO0LAB7/S5MnxCdTNDd4p6BsyIVsv9JQsATbTSq8KHRpLwIHbVlUNatxd+1Ow==", "dev": true, "requires": { "@nodelib/fs.stat": "^2.0.2", @@ -8349,21 +8543,21 @@ "dev": true }, "function-bind": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz", + "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==", "dev": true }, "function.prototype.name": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.5.tgz", - "integrity": "sha512-uN7m/BzVKQnCUF/iW8jYea67v++2u7m5UgENbHRtdDVclOUP+FMPlCNdmk0h/ysGyo2tavMJEDqJAkJdRa1vMA==", + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/function.prototype.name/-/function.prototype.name-1.1.6.tgz", + "integrity": "sha512-Z5kx79swU5P27WEayXM1tBi5Ze/lbIyiNgU3qyXUOf9b2rgXYyF9Dy9Cx+IQv/Lc8WCG6L82zwUPpSS9hGehIg==", "dev": true, "requires": { "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "es-abstract": "^1.19.0", - "functions-have-names": "^1.2.2" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "functions-have-names": "^1.2.3" } }, "functional-red-black-tree": { @@ -8398,14 +8592,16 @@ "dev": true }, "get-intrinsic": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.0.tgz", - "integrity": "sha512-L049y6nFOuom5wGyRc3/gdTLO94dySVKRACj1RmJZBQXlbTMhtNIgkWkUHq+jYmZvKf14EW1EoJnnjbmoHij0Q==", + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.2.4.tgz", + "integrity": "sha512-5uYhsJH8VJBTv7oslg4BznJYhDoRI6waYCxMmCdnTrcCrHA/fCFKoTFz2JKKE0HdDFUF7/oQuhzumXJK7paBRQ==", "dev": true, "requires": { - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.3" + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "has-proto": "^1.0.1", + "has-symbols": "^1.0.3", + "hasown": "^2.0.0" } }, "get-stdin": { @@ -8415,13 +8611,14 @@ "dev": true }, "get-symbol-description": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.0.tgz", - "integrity": "sha512-2EmdH1YvIQiZpltCNgkuiUnyukzxM/R6NDJX31Ke3BG1Nq5b0S2PhX59UKi9vZpPDQVdqn+1IcaAwnzTT5vCjw==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/get-symbol-description/-/get-symbol-description-1.0.2.tgz", + "integrity": "sha512-g0QYk1dZBxGwk+Ngc+ltRH2IBp2f7zBkBMBJZCDerh6EhlhSR6+9irMCuT/09zD6qkarHUSn529sK/yL4S27mg==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" + "call-bind": "^1.0.5", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4" } }, "glob": { @@ -8464,14 +8661,14 @@ } }, "globby": { - "version": "13.1.3", - "resolved": "https://registry.npmjs.org/globby/-/globby-13.1.3.tgz", - "integrity": "sha512-8krCNHXvlCgHDpegPzleMq07yMYTO2sXKASmZmquEYWEmCx6J5UTRbp5RwMJkTJGtcQ44YpiUYUiN0b9mzy8Bw==", + "version": "13.2.2", + "resolved": "https://registry.npmjs.org/globby/-/globby-13.2.2.tgz", + "integrity": "sha512-Y1zNGV+pzQdh7H39l9zgB4PJqjRNqydvdYCDG4HFXM4XuvSaQQlEc91IU1yALL8gUTDomgBAfz3XJdmUS+oo0w==", "dev": true, "requires": { "dir-glob": "^3.0.1", - "fast-glob": "^3.2.11", - "ignore": "^5.2.0", + "fast-glob": "^3.3.0", + "ignore": "^5.2.4", "merge2": "^1.4.1", "slash": "^4.0.0" }, @@ -8494,20 +8691,11 @@ } }, "graceful-fs": { - "version": "4.2.10", - "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.10.tgz", - "integrity": "sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA==", + "version": "4.2.11", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz", + "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==", "dev": true }, - "has": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", - "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", - "dev": true, - "requires": { - "function-bind": "^1.1.1" - } - }, "has-bigints": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", @@ -8521,18 +8709,18 @@ "dev": true }, "has-property-descriptors": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.0.tgz", - "integrity": "sha512-62DVLZGoiEBDHQyqG4w9xCuZ7eJEwNmJRWw2VY84Oedb7WFcA27fiEVe8oUQx9hAUJ4ekurquucTGwsyO1XGdQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz", + "integrity": "sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==", "dev": true, "requires": { - "get-intrinsic": "^1.1.1" + "es-define-property": "^1.0.0" } }, "has-proto": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.1.tgz", - "integrity": "sha512-7qE+iP+O+bgF9clE5+UoBFzE65mlBiVj3tKCrlNQ0Ogwm0BjpT/gK4SlLYDMybDh5I3TCTKnPPa0oMG7JDYrhg==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has-proto/-/has-proto-1.0.3.tgz", + "integrity": "sha512-SJ1amZAJUiZS+PhsVLf5tGydlaVB8EdFpaSO4gmiUKUOxk8qzn5AIy4ZeJUmh22znIdk/uMAUT2pl3FxzVUH+Q==", "dev": true }, "has-symbols": { @@ -8542,12 +8730,12 @@ "dev": true }, "has-tostringtag": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.0.tgz", - "integrity": "sha512-kFjcSNhnlGV1kyoGk7OXKSawH5JOb/LzUc5w9B02hOTO0dfFRjbHQKvg1d6cf3HbeUmtU9VbbV3qzZ2Teh97WQ==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-tostringtag/-/has-tostringtag-1.0.2.tgz", + "integrity": "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw==", "dev": true, "requires": { - "has-symbols": "^1.0.2" + "has-symbols": "^1.0.3" } }, "hash-for-dep": { @@ -8576,6 +8764,15 @@ } } }, + "hasown": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.1.tgz", + "integrity": "sha512-1/th4MHjnwncwXsIW6QMzlvYL9kG5e/CpVvLRZe4XPa8TOUNbCELqmvhDmnkNsAjwaG4+I8gJJL0JBvTTLO9qA==", + "dev": true, + "requires": { + "function-bind": "^1.1.2" + } + }, "heimdalljs": { "version": "0.2.6", "resolved": "https://registry.npmjs.org/heimdalljs/-/heimdalljs-0.2.6.tgz", @@ -8627,9 +8824,9 @@ "dev": true }, "ignore": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.2.0.tgz", - "integrity": "sha512-CmxgYGiEPCLhfLnpPp1MoRmifwEIOgjcHXxOBjv7mY96c+eWScsOP9c112ZyLdWHi0FxHjI+4uVhKYp/gcdRmQ==", + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.3.1.tgz", + "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", "dev": true }, "import-fresh": { @@ -8665,35 +8862,24 @@ "dev": true }, "internal-slot": { - "version": "1.0.5", - "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.5.tgz", - "integrity": "sha512-Y+R5hJrzs52QCG2laLn4udYVnxsfny9CpOhNhUvk/SSSVyF6T27FzRbF0sroPidSu3X8oEAkOn2K804mjpt6UQ==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/internal-slot/-/internal-slot-1.0.7.tgz", + "integrity": "sha512-NGnrKwXzSms2qUUih/ILZ5JBqNTSa1+ZmP6flaIp6KmSElgE9qdndzS3cqjrDovwFdmwsGsLdeFgB6suw+1e9g==", "dev": true, "requires": { - "get-intrinsic": "^1.2.0", - "has": "^1.0.3", + "es-errors": "^1.3.0", + "hasown": "^2.0.0", "side-channel": "^1.0.4" } }, - "is-arguments": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.1.tgz", - "integrity": "sha512-8Q7EARjzEnKpt/PCD7e1cgUS0a6X8u5tdSiMqXhojOdoV9TsMsiO+9VLC5vAmO8N7/GmXn7yjR8qnA6bVAEzfA==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "has-tostringtag": "^1.0.0" - } - }, "is-array-buffer": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.1.tgz", - "integrity": "sha512-ASfLknmY8Xa2XtB4wmbz13Wu202baeA18cJBCeCy0wXUHZF0IPyVEXqKEcd+t2fNSLLL1vC6k7lxZEojNbISXQ==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/is-array-buffer/-/is-array-buffer-3.0.4.tgz", + "integrity": "sha512-wcjaerHw0ydZwfhiKbXJWLDY8A7yV7KhjQOpb83hGgGfId/aQa4TOvwyzn2PuswW2gPCYEL/nEAiSVpdOj1lXw==", "dev": true, "requires": { "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", - "is-typed-array": "^1.1.10" + "get-intrinsic": "^1.2.1" } }, "is-bigint": { @@ -8722,12 +8908,12 @@ "dev": true }, "is-core-module": { - "version": "2.11.0", - "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.11.0.tgz", - "integrity": "sha512-RRjxlvLDkD1YJwDbroBHMb+cukurkDWNyHx7D3oNB5x9rb5ogcksMC5wHCadcXoo67gVr/+3GFySh3134zi6rw==", + "version": "2.13.1", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.13.1.tgz", + "integrity": "sha512-hHrIjvZsftOsvKSn2TRYl63zvxsgE0K+0mYMoH6gD4omR5IWB2KynivBQczo3+wF1cCkjzvptnI9Q0sPU66ilw==", "dev": true, "requires": { - "has": "^1.0.3" + "hasown": "^2.0.0" } }, "is-date-object": { @@ -8766,16 +8952,10 @@ "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", "dev": true }, - "is-map": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-map/-/is-map-2.0.2.tgz", - "integrity": "sha512-cOZFQQozTha1f4MxLFzlgKYPTyj26picdZTx82hbc/Xf4K/tZOOXSCkMvU4pKioRXGDLJRn0GM7Upe7kR721yg==", - "dev": true - }, "is-negative-zero": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.2.tgz", - "integrity": "sha512-dqJvarLawXsFbNDeJW7zAz8ItJ9cd28YufuuFzh0G8pNHjJMnY08Dv7sYX2uF5UpQOwieAeOExEYAWWfu7ZZUA==", + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.3.tgz", + "integrity": "sha512-5KoIu2Ngpyek75jXodFvnafB6DJgr3u8uuK0LEZJjrU19DrMD3EVERaR8sjz8CCGgpZvxPl9SuE1GMVPFHx1mw==", "dev": true }, "is-number": { @@ -8803,19 +8983,13 @@ "has-tostringtag": "^1.0.0" } }, - "is-set": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-set/-/is-set-2.0.2.tgz", - "integrity": "sha512-+2cnTEZeY5z/iXGbLhPrOAaK/Mau5k5eXq9j14CpRTftq0pAJu2MwVRSZhyZWBzx3o6X795Lz6Bpb6R0GKf37g==", - "dev": true - }, "is-shared-array-buffer": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", - "integrity": "sha512-sqN2UDu1/0y6uvXyStCOzyhAjCSlHceFoMKJW8W9EU9cvic/QdsZ0kEU93HEy3IUEFZIiH/3w+AH/UQbPHNdhA==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.3.tgz", + "integrity": "sha512-nA2hv5XIhLR3uVzDDfCIknerhx8XUKnstuOERPNNIinXG7v9u+ohXF67vxm4TPTEPU6lm61ZkwP3c9PCB97rhg==", "dev": true, "requires": { - "call-bind": "^1.0.2" + "call-bind": "^1.0.7" } }, "is-string": { @@ -8837,16 +9011,12 @@ } }, "is-typed-array": { - "version": "1.1.10", - "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.10.tgz", - "integrity": "sha512-PJqgEHiWZvMpaFZ3uTc8kHPM4+4ADTlDniuQL7cU/UDA0Ql7F70yGfHph3cLNe+c9toaigv+DFzTJKhc2CtO6A==", + "version": "1.1.13", + "resolved": "https://registry.npmjs.org/is-typed-array/-/is-typed-array-1.1.13.tgz", + "integrity": "sha512-uZ25/bUAlUY5fR4OKT4rZQEBrzQWYV9ZJYGGsUmEJ6thodVJ1HX64ePQ6Z0qPWP+m+Uq6e9UugrE38jeYsDSMw==", "dev": true, "requires": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", - "for-each": "^0.3.3", - "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0" + "which-typed-array": "^1.1.14" } }, "is-unicode-supported": { @@ -8855,12 +9025,6 @@ "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", "dev": true }, - "is-weakmap": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/is-weakmap/-/is-weakmap-2.0.1.tgz", - "integrity": "sha512-NSBR4kH5oVj1Uwvv970ruUkCV7O1mzgVFO4/rev2cLRda9Tm9HrL70ZPut4rOHgY0FNrUu9BCbXA2sdQ+x0chA==", - "dev": true - }, "is-weakref": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-weakref/-/is-weakref-1.0.2.tgz", @@ -8870,20 +9034,10 @@ "call-bind": "^1.0.2" } }, - "is-weakset": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/is-weakset/-/is-weakset-2.0.2.tgz", - "integrity": "sha512-t2yVvttHkQktwnNNmBQ98AhENLdPUTDTE21uPqAQ0ARwQfGeQKRVS0NNurH7bTf7RrvcVn1OOge45CnBeHCSmg==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.1" - } - }, "isarray": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", - "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", "dev": true }, "isexe": { @@ -8899,14 +9053,6 @@ "dev": true, "requires": { "isarray": "1.0.0" - }, - "dependencies": { - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", - "dev": true - } } }, "istextorbinary": { @@ -8998,14 +9144,6 @@ "requires": { "isarray": "^1.0.0", "isobject": "^2.0.0" - }, - "dependencies": { - "isarray": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", - "integrity": "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==", - "dev": true - } } }, "locate-path": { @@ -9197,21 +9335,11 @@ "dev": true }, "object-inspect": { - "version": "1.12.3", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.12.3.tgz", - "integrity": "sha512-geUvdk7c+eizMNUDkRpW1wJwgfOiOeHbxBR/hLXK1aT6zmVSO0jsQcs7fj6MGw89jC/cjGfLcNOrtMYtGqm81g==", + "version": "1.13.1", + "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.1.tgz", + "integrity": "sha512-5qoj1RUiKOMsCCNLV1CBiPYE10sziTsnmNxkAI/rZhiD63CF7IqdFGC/XzjWjpSgLf0LxXX3bDFIh0E18f6UhQ==", "dev": true }, - "object-is": { - "version": "1.1.5", - "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.5.tgz", - "integrity": "sha512-3cyDsyHgtmi7I7DfSSI2LDp6SK2lwvtbg0p0R1e0RvTqF5ceGx+K2dfSjm1bKDMVCFEDAQvy+o8c6a7VujOddw==", - "dev": true, - "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3" - } - }, "object-keys": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/object-keys/-/object-keys-1.1.1.tgz", @@ -9219,13 +9347,13 @@ "dev": true }, "object.assign": { - "version": "4.1.4", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.4.tgz", - "integrity": "sha512-1mxKf0e58bvyjSCtKYY4sRe9itRk3PJpquJOjeIkz885CczcI4IvJJDLPS72oowuSh+pBxUFROpX+TU++hxhZQ==", + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.5.tgz", + "integrity": "sha512-byy+U7gp+FVwmyzKPYhW2h5l3crpmGsxl7X2s8y43IgxvG4g3QZ6CffDtsNQy1WsmZpQbO+ybo0AlW7TY6DcBQ==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.4", + "call-bind": "^1.0.5", + "define-properties": "^1.2.1", "has-symbols": "^1.0.3", "object-keys": "^1.1.1" } @@ -9394,6 +9522,12 @@ "integrity": "sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==", "dev": true }, + "possible-typed-array-names": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/possible-typed-array-names/-/possible-typed-array-names-1.0.0.tgz", + "integrity": "sha512-d7Uw+eZoloe0EHDIYoe+bQ5WXnGMOpmiZFTuMWCwpjzzkL2nTjcKiAk4hh8TjnGye2TwWOk3UXucZ+3rbmBa8Q==", + "dev": true + }, "prelude-ls": { "version": "1.2.1", "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz", @@ -9401,9 +9535,9 @@ "dev": true }, "prettier": { - "version": "2.5.1", - "resolved": "https://registry.npmjs.org/prettier/-/prettier-2.5.1.tgz", - "integrity": "sha512-vBZcPRUR5MZJwoyi3ZoyQlc1rXeEck8KgeC9AwwOn+exuxLxq5toTRDTSaVrXHxelDMHy9zlicw8u66yxoSUFg==", + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.2.5.tgz", + "integrity": "sha512-3/GWa9aOC0YeD7LUfvOG2NiDyhOWRvt1k+rcKhOuYnMY24iiCphgneUfJDyFXd6rZCAnuLBv6UeAULtrhT/F4A==", "dev": true }, "prettier-linter-helpers": { @@ -9478,9 +9612,9 @@ } }, "readable-stream": { - "version": "3.6.1", - "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.1.tgz", - "integrity": "sha512-+rQmrWMYGA90yenhTYsLWAsLsqVC8osOw6PKE1HDYiO0gdPeKe/xDHNzIAIn4C91YQ6oenEhfYqqc1883qHbjQ==", + "version": "3.6.2", + "resolved": "https://registry.npmjs.org/readable-stream/-/readable-stream-3.6.2.tgz", + "integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==", "dev": true, "requires": { "inherits": "^2.0.3", @@ -9488,15 +9622,22 @@ "util-deprecate": "^1.0.1" } }, + "regenerator-runtime": { + "version": "0.14.1", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz", + "integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==", + "dev": true + }, "regexp.prototype.flags": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.4.3.tgz", - "integrity": "sha512-fjggEOO3slI6Wvgjwflkc4NFRCTZAu5CnNfBd5qOMYhWdn67nJBBu34/TkD++eeFmd8C9r9jfXJ27+nSiRkSUA==", + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.5.2.tgz", + "integrity": "sha512-NcDiDkTLuPR+++OCKB0nWafEmhg/Da8aUPLPMQbK+bxKKCm1/S5he+AqYa4PlMCVBalb4/yxIRub6qkEx5yJbw==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "define-properties": "^1.1.3", - "functions-have-names": "^1.2.2" + "call-bind": "^1.0.6", + "define-properties": "^1.2.1", + "es-errors": "^1.3.0", + "set-function-name": "^2.0.1" } }, "regexpp": { @@ -9518,12 +9659,12 @@ "dev": true }, "resolve": { - "version": "1.22.1", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.1.tgz", - "integrity": "sha512-nBpuuYuY5jFsli/JIs1oldw6fOQCBioohqWZg/2hiaOybXOft4lonv85uDOKXdf8rhyK159cxU5cDcK/NKk8zw==", + "version": "1.22.8", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.8.tgz", + "integrity": "sha512-oKWePCxqpd6FlLvGV1VU0x7bkPmmCNolxzjMf4NczoDnQcIWrAF+cPtZn5i6n+RfD2d9i0tzpKnG6Yk168yIyw==", "dev": true, "requires": { - "is-core-module": "^2.9.0", + "is-core-module": "^2.13.0", "path-parse": "^1.0.7", "supports-preserve-symlinks-flag": "^1.0.0" } @@ -9590,6 +9731,26 @@ "queue-microtask": "^1.2.2" } }, + "safe-array-concat": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.0.tgz", + "integrity": "sha512-ZdQ0Jeb9Ofti4hbt5lX3T2JcAamT9hfzYU1MNB+z/jaEbB6wfFfPIR/zEORmZqobkCCJhSjodobH6WHNmJ97dg==", + "dev": true, + "requires": { + "call-bind": "^1.0.5", + "get-intrinsic": "^1.2.2", + "has-symbols": "^1.0.3", + "isarray": "^2.0.5" + }, + "dependencies": { + "isarray": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.5.tgz", + "integrity": "sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw==", + "dev": true + } + } + }, "safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", @@ -9598,13 +9759,13 @@ "peer": true }, "safe-regex-test": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.0.tgz", - "integrity": "sha512-JBUUzyOgEwXQY1NuPtvcj/qcBDbDmEvWufhlnXZIm75DEHp+afM1r1ujJpJsV/gSM4t59tpDyPi1sd6ZaPFfsA==", + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/safe-regex-test/-/safe-regex-test-1.0.3.tgz", + "integrity": "sha512-CdASjNJPvRa7roO6Ra/gLYBTzYzzPyyBXxIMdGW3USQLyjWEls2RgW5UBTXaQVp+OrpeCK3bLem8smtmheoRuw==", "dev": true, "requires": { - "call-bind": "^1.0.2", - "get-intrinsic": "^1.1.3", + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", "is-regex": "^1.1.4" } }, @@ -9614,6 +9775,32 @@ "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", "dev": true }, + "set-function-length": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/set-function-length/-/set-function-length-1.2.1.tgz", + "integrity": "sha512-j4t6ccc+VsKwYHso+kElc5neZpjtq9EnRICFZtWyBsLojhmeF/ZBd/elqm22WJh/BziDe/SBiOeAt0m2mfLD0g==", + "dev": true, + "requires": { + "define-data-property": "^1.1.2", + "es-errors": "^1.3.0", + "function-bind": "^1.1.2", + "get-intrinsic": "^1.2.3", + "gopd": "^1.0.1", + "has-property-descriptors": "^1.0.1" + } + }, + "set-function-name": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/set-function-name/-/set-function-name-2.0.2.tgz", + "integrity": "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ==", + "dev": true, + "requires": { + "define-data-property": "^1.1.4", + "es-errors": "^1.3.0", + "functions-have-names": "^1.2.3", + "has-property-descriptors": "^1.0.2" + } + }, "shebang-command": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz", @@ -9630,14 +9817,15 @@ "dev": true }, "side-channel": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.4.tgz", - "integrity": "sha512-q5XPytqFEIKHkGdiMIrY10mvLRvnQh42/+GoBlFW3b2LXLE2xxJpZFdm94we0BaoV3RwJyGqg5wS7epxTv0Zvw==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.0.5.tgz", + "integrity": "sha512-QcgiIWV4WV7qWExbN5llt6frQB/lBven9pqliLXfGPB+K9ZYXxDozp0wLkHS24kWCm+6YXH/f0HhnObZnZOBnQ==", "dev": true, "requires": { - "call-bind": "^1.0.0", - "get-intrinsic": "^1.0.2", - "object-inspect": "^1.9.0" + "call-bind": "^1.0.6", + "es-errors": "^1.3.0", + "get-intrinsic": "^1.2.4", + "object-inspect": "^1.13.1" } }, "signal-exit": { @@ -9714,20 +9902,11 @@ "dev": true }, "sprintf-js": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.2.tgz", - "integrity": "sha512-VE0SOVEHCk7Qc8ulkWw3ntAzXuqf7S2lvwQaDLRnUeIEaKNQJzV6BwmLKhOqT61aGhfUMrXeaBk+oDGCzvhcug==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/sprintf-js/-/sprintf-js-1.1.3.tgz", + "integrity": "sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA==", "dev": true }, - "stop-iteration-iterator": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/stop-iteration-iterator/-/stop-iteration-iterator-1.0.0.tgz", - "integrity": "sha512-iCGQj+0l0HOdZ2AEeBADlsRC+vsnDsZsbdSiH1yNSjcfKM7fdpCMfqAL/dwF5BLiw/XhRft/Wax6zQbhq2BcjQ==", - "dev": true, - "requires": { - "internal-slot": "^1.0.4" - } - }, "string_decoder": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz", @@ -9757,41 +9936,53 @@ } }, "string.prototype.matchall": { - "version": "4.0.8", - "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.8.tgz", - "integrity": "sha512-6zOCOcJ+RJAQshcTvXPHoxoQGONa3e/Lqx90wUA+wEzX78sg5Bo+1tQo4N0pohS0erG9qtCqJDjNCQBjeWVxyg==", + "version": "4.0.10", + "resolved": "https://registry.npmjs.org/string.prototype.matchall/-/string.prototype.matchall-4.0.10.tgz", + "integrity": "sha512-rGXbGmOEosIQi6Qva94HUjgPs9vKW+dkG7Y8Q5O2OYkWL6wFaTRZO8zM4mhP94uX55wgyrXzfS2aGtGzUL7EJQ==", "dev": true, "requires": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4", - "get-intrinsic": "^1.1.3", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1", + "get-intrinsic": "^1.2.1", "has-symbols": "^1.0.3", - "internal-slot": "^1.0.3", - "regexp.prototype.flags": "^1.4.3", + "internal-slot": "^1.0.5", + "regexp.prototype.flags": "^1.5.0", + "set-function-name": "^2.0.0", "side-channel": "^1.0.4" } }, + "string.prototype.trim": { + "version": "1.2.8", + "resolved": "https://registry.npmjs.org/string.prototype.trim/-/string.prototype.trim-1.2.8.tgz", + "integrity": "sha512-lfjY4HcixfQXOfaqCvcBuOIapyaroTXhbkfJN3gcB1OtyupngWK4sEET9Knd0cXd28kTUqu/kHoV4HKSJdnjiQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.2", + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" + } + }, "string.prototype.trimend": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.6.tgz", - "integrity": "sha512-JySq+4mrPf9EsDBEDYMOb/lM7XQLulwg5R/m1r0PXEFqrV0qHvl58sdTilSXtKOflCsK2E8jxf+GKC0T07RWwQ==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.7.tgz", + "integrity": "sha512-Ni79DqeB72ZFq1uH/L6zJ+DKZTkOtPIHovb3YZHQViE+HDouuU4mBrLOLDn5Dde3RF8qw5qVETEjhu9locMLvA==", "dev": true, "requires": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" } }, "string.prototype.trimstart": { - "version": "1.0.6", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.6.tgz", - "integrity": "sha512-omqjMDaY92pbn5HOX7f9IccLA+U1tA9GvtU4JrodiXFfYB7jPzzHpRzpglLAjtUV6bB557zwClJezTqnAiYnQA==", + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.7.tgz", + "integrity": "sha512-NGhtDFu3jCEm7B4Fy0DpLewdJQOZcQ0rGbwQ/+stjnrp2i+rlKeCvos9hOIeCmqwratM47OBxY7uFZzjxHXmrg==", "dev": true, "requires": { "call-bind": "^1.0.2", - "define-properties": "^1.1.4", - "es-abstract": "^1.20.4" + "define-properties": "^1.2.0", + "es-abstract": "^1.22.1" } }, "strip-ansi": { @@ -9984,15 +10175,56 @@ "integrity": "sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ==", "dev": true }, + "typed-array-buffer": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-buffer/-/typed-array-buffer-1.0.2.tgz", + "integrity": "sha512-gEymJYKZtKXzzBzM4jqa9w6Q1Jjm7x2d+sh19AdsD4wqnMPDYyvwpsIc2Q/835kHuo3BEQ7CjelGhfTsoBb2MQ==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "es-errors": "^1.3.0", + "is-typed-array": "^1.1.13" + } + }, + "typed-array-byte-length": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/typed-array-byte-length/-/typed-array-byte-length-1.0.1.tgz", + "integrity": "sha512-3iMJ9q0ao7WE9tWcaYKIptkNBuOIcZCCT0d4MRvuuH88fEoEH62IuQe0OtraD3ebQEoTRk8XCBoknUNc1Y67pw==", + "dev": true, + "requires": { + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + } + }, + "typed-array-byte-offset": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/typed-array-byte-offset/-/typed-array-byte-offset-1.0.2.tgz", + "integrity": "sha512-Ous0vodHa56FviZucS2E63zkgtgrACj7omjwd/8lTEMEPFFyjfixMZ1ZXenpgCFBBt4EC1J2XsyVS2gkG0eTFA==", + "dev": true, + "requires": { + "available-typed-arrays": "^1.0.7", + "call-bind": "^1.0.7", + "for-each": "^0.3.3", + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13" + } + }, "typed-array-length": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.4.tgz", - "integrity": "sha512-KjZypGq+I/H7HI5HlOoGHkWUUGq+Q0TPhQurLbyrVrvnKTBgzLhIJ7j6J/XTQOi0d1RjyZ0wdas8bKs2p0x3Ng==", + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/typed-array-length/-/typed-array-length-1.0.5.tgz", + "integrity": "sha512-yMi0PlwuznKHxKmcpoOdeLwxBoVPkqZxd7q2FgMkmD3bNwvF5VW0+UlUQ1k1vmktTu4Yu13Q0RIxEP8+B+wloA==", "dev": true, "requires": { - "call-bind": "^1.0.2", + "call-bind": "^1.0.7", "for-each": "^0.3.3", - "is-typed-array": "^1.1.9" + "gopd": "^1.0.1", + "has-proto": "^1.0.3", + "is-typed-array": "^1.1.13", + "possible-typed-array-names": "^1.0.0" } }, "typescript": { @@ -10024,9 +10256,9 @@ } }, "universalify": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.0.tgz", - "integrity": "sha512-hAZsKq7Yy11Zu1DE0OzWjw7nnLZmJZYTDZZyEFHZdUhV8FkH5MCfoU1XMaxXovpyW5nq5scPqq0ZDP9Zyl04oQ==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-2.0.1.tgz", + "integrity": "sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw==", "dev": true }, "upath": { @@ -10073,9 +10305,9 @@ }, "dependencies": { "semver": { - "version": "7.3.8", - "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.8.tgz", - "integrity": "sha512-NB1ctGL5rlHrPJtFDVIVzTyQylMLu9N9VICA6HSFJo8MCGVTMW6gfpicwKmmK/dAjTOrqu5l63JJOpDSrAis3A==", + "version": "7.6.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.6.0.tgz", + "integrity": "sha512-EnwXhrlwXMk9gKu5/flx5sv/an57AkRplG3hTK68W7FRDN+k+OWBj65M7719OkA82XLBxrcX0KSHj+X5COhOVg==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -10125,30 +10357,17 @@ "is-symbol": "^1.0.3" } }, - "which-collection": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/which-collection/-/which-collection-1.0.1.tgz", - "integrity": "sha512-W8xeTUwaln8i3K/cY1nGXzdnVZlidBcagyNFtBdD5kxnb4TvGKR7FfSIS3mYpwWS1QUCutfKz8IY8RjftB0+1A==", - "dev": true, - "requires": { - "is-map": "^2.0.1", - "is-set": "^2.0.1", - "is-weakmap": "^2.0.1", - "is-weakset": "^2.0.1" - } - }, "which-typed-array": { - "version": "1.1.9", - "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.9.tgz", - "integrity": "sha512-w9c4xkx6mPidwp7180ckYWfMmvxpjlZuIudNtDf4N/tTAUB8VJbX25qZoAsrtGuYNnGw3pa0AXgbGKRB8/EceA==", + "version": "1.1.14", + "resolved": "https://registry.npmjs.org/which-typed-array/-/which-typed-array-1.1.14.tgz", + "integrity": "sha512-VnXFiIW8yNn9kIHN88xvZ4yOWchftKDsRJ8fEPacX/wl1lOvBrhsJ/OeJCXq7B0AaijRuqgzSKalJoPk+D8MPg==", "dev": true, "requires": { - "available-typed-arrays": "^1.0.5", - "call-bind": "^1.0.2", + "available-typed-arrays": "^1.0.6", + "call-bind": "^1.0.5", "for-each": "^0.3.3", "gopd": "^1.0.1", - "has-tostringtag": "^1.0.0", - "is-typed-array": "^1.1.10" + "has-tostringtag": "^1.0.1" } }, "word-wrap": { @@ -10158,9 +10377,9 @@ "dev": true }, "workerpool": { - "version": "6.3.1", - "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.3.1.tgz", - "integrity": "sha512-0x7gJm1rhpn5SPG9NENOxPtbfUZZtK/qOg6gEdSqeDBA3dTeR91RJqSPjccPRCkhNfrnnl/dWxSSj5w9CtdzNA==", + "version": "6.5.1", + "resolved": "https://registry.npmjs.org/workerpool/-/workerpool-6.5.1.tgz", + "integrity": "sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA==", "dev": true }, "wrap-ansi": { @@ -10193,9 +10412,9 @@ "dev": true }, "yargs": { - "version": "17.7.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.1.tgz", - "integrity": "sha512-cwiTb08Xuv5fqF4AovYacTFNxk62th7LKJ6BL9IGUpTJrWoU7/7WdQGTP2SjKf1dUNBGzDd28p/Yfs/GI6JrLw==", + "version": "17.7.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-17.7.2.tgz", + "integrity": "sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==", "dev": true, "requires": { "cliui": "^8.0.1", diff --git a/package.json b/package.json index 2b969537f..3a1942bee 100644 --- a/package.json +++ b/package.json @@ -11,15 +11,15 @@ "@babel/eslint-parser": "7.17.0", "@typescript-eslint/eslint-plugin": "5.12.1", "@typescript-eslint/parser": "5.12.1", - "ember-template-lint": "5.2.0", - "ember-template-lint-plugin-prettier": "4.1.0", + "ember-template-lint": "5.13.0", + "ember-template-lint-plugin-prettier": "5.0.0", "eslint": "8.10.0", "eslint-config-prettier": "8.4.0", "eslint-plugin-ember": "10.5.9", "eslint-plugin-mirego": "^1.0.0", "eslint-plugin-node": "11.1.0", "eslint-plugin-sonarjs": "0.12.0", - "prettier": "2.5.1", + "prettier": "3.2.5", "typescript": "4.7.4" } } diff --git a/priv/repo/migrations/20240227025330_add_copy_on_update_translation_on_versions.exs b/priv/repo/migrations/20240227025330_add_copy_on_update_translation_on_versions.exs new file mode 100644 index 000000000..211f40507 --- /dev/null +++ b/priv/repo/migrations/20240227025330_add_copy_on_update_translation_on_versions.exs @@ -0,0 +1,10 @@ +defmodule Accent.Repo.Migrations.AddCopyOnUpdateTranslationOnVersions do + @moduledoc false + use Ecto.Migration + + def change do + alter table(:versions) do + add(:copy_on_update_translation, :boolean, default: false, null: false) + end + end +end diff --git a/priv/repo/migrations/20240315115937_add_non_nullable_checks.exs b/priv/repo/migrations/20240315115937_add_non_nullable_checks.exs new file mode 100644 index 000000000..f7ebf99ef --- /dev/null +++ b/priv/repo/migrations/20240315115937_add_non_nullable_checks.exs @@ -0,0 +1,86 @@ +defmodule Accent.Repo.Migrations.AddNonNullableChecks do + @moduledoc false + use Ecto.Migration + + def down do + end + + # credo:disable-for-next-line + def up do + drop(constraint(:documents, :documents_project_id_fkey)) + + alter table(:documents) do + modify(:path, :string, null: false) + modify(:format, :string, null: false) + modify(:project_id, references(:projects, type: :uuid), null: false) + end + + alter table(:languages) do + modify(:name, :string, null: false) + modify(:slug, :string, null: false) + modify(:iso_639_1, :string, null: false) + modify(:iso_639_3, :string, null: false) + modify(:locale, :string, null: false) + modify(:android_code, :string, null: false) + modify(:osx_code, :string, null: false) + modify(:osx_locale, :string, null: false) + end + + drop(constraint(:auth_access_tokens, :auth_access_tokens_user_id_fkey)) + + alter table(:auth_access_tokens) do + modify(:token, :string, null: false) + modify(:user_id, references(:users, type: :uuid), null: false) + end + + drop(constraint(:auth_providers, :auth_providers_user_id_fkey)) + + alter table(:auth_providers) do + modify(:name, :string, null: false) + modify(:uid, :string, null: false) + modify(:user_id, references(:users, type: :uuid), null: false) + end + + drop(constraint(:collaborators, :collaborators_project_id_fkey)) + + alter table(:collaborators) do + modify(:role, :string, null: false) + modify(:project_id, references(:projects, type: :uuid), null: false) + end + + drop(constraint(:comments, :comments_user_id_fkey)) + + alter table(:comments) do + modify(:text, :text, null: false) + modify(:user_id, references(:users, type: :uuid), null: false) + end + + alter table(:projects) do + modify(:name, :string, null: false) + modify(:locked_file_operations, :boolean, null: false, default: false) + modify(:sync_lock_version, :integer, null: false, default: 1) + end + + drop(constraint(:revisions, :revisions_project_id_fkey)) + drop(constraint(:revisions, :revisions_language_id_fkey)) + + alter table(:revisions) do + modify(:project_id, references(:projects, type: :uuid), null: false) + modify(:language_id, references(:languages, type: :uuid), null: false) + modify(:master, :boolean, null: false, default: true) + end + + alter table(:operations) do + modify(:rollbacked, :boolean, null: false, default: false) + modify(:batch, :boolean, null: false, default: false) + modify(:action, :string, null: false) + end + + alter table(:translations) do + modify(:key, :text, null: false) + modify(:removed, :boolean, null: false, default: false) + modify(:conflicted, :boolean, null: false, default: false) + modify(:comments_count, :integer, null: false, default: 0) + end + end +end diff --git a/test/auth/user_remote/authenticator_test.exs b/test/auth/user_remote/authenticator_test.exs index 7d2e75cdc..67ab00218 100644 --- a/test/auth/user_remote/authenticator_test.exs +++ b/test/auth/user_remote/authenticator_test.exs @@ -25,17 +25,17 @@ defmodule AccentTest.UserRemote.Authenticator do end test "normalize collaborators with email" do - assigner = Repo.insert!(%User{email: "foo@example.com"}) - language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project", language_id: language.id}) + assigner = Factory.insert(User, email: "foo@example.com") + language = Factory.insert(Language) + project = Factory.insert(Project, language_id: language.id) collaborator = - Repo.insert!(%Collaborator{ + Factory.insert(Collaborator, project_id: project.id, role: "admin", assigner_id: assigner.id, email: "test@example.com" - }) + ) {:ok, _token} = Authenticator.authenticate(%{provider: :dummy, info: %{email: "test@example.com"}}) user = Repo.get_by(User, email: "test@example.com") @@ -45,17 +45,17 @@ defmodule AccentTest.UserRemote.Authenticator do end test "normalize collaborators with uppercased email" do - assigner = Repo.insert!(%User{email: "foo@example.com"}) - language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project", language_id: language.id}) + assigner = Factory.insert(User, email: "foo@example.com") + language = Factory.insert(Language) + project = Factory.insert(Project, language_id: language.id) collaborator = - Repo.insert!(%Collaborator{ + Factory.insert(Collaborator, project_id: project.id, role: "admin", assigner_id: assigner.id, email: "test@example.com" - }) + ) {:ok, _token} = Authenticator.authenticate(%{provider: :dummy, info: %{email: "TeSt@eXamPle.com"}}) user = Repo.get_by(User, email: "test@example.com") diff --git a/test/auth/user_remote/persister_test.exs b/test/auth/user_remote/persister_test.exs index bd1d26dd7..88f2ce64a 100644 --- a/test/auth/user_remote/persister_test.exs +++ b/test/auth/user_remote/persister_test.exs @@ -18,8 +18,8 @@ defmodule AccentTest.UserRemote.Persister do end test "persist with existing user existing provider" do - existing_user = Repo.insert!(%User{email: @user.email}) - Repo.insert!(%AuthProvider{name: @user.provider, uid: @user.uid}) + existing_user = Factory.insert(User, email: @user.email) + Factory.insert(AuthProvider, name: @user.provider, uid: @user.uid) user = Persister.persist(@user) @@ -28,8 +28,8 @@ defmodule AccentTest.UserRemote.Persister do end test "persist with existing user new provider" do - existing_user = Repo.insert!(%User{email: @user.email}) - Repo.insert!(%AuthProvider{name: "dummy", uid: @user.email}) + existing_user = Factory.insert(User, email: @user.email) + Factory.insert(AuthProvider, name: "dummy", uid: @user.email) user = Persister.persist(@user) diff --git a/test/auth/user_remote/token_giver_test.exs b/test/auth/user_remote/token_giver_test.exs index d2701900f..c039b6284 100644 --- a/test/auth/user_remote/token_giver_test.exs +++ b/test/auth/user_remote/token_giver_test.exs @@ -7,15 +7,14 @@ defmodule AccentTest.UserRemote.TokenGiver do alias Accent.User alias Accent.UserRemote.TokenGiver - @user %User{email: "test@test.com"} @token %AccessToken{revoked_at: nil, token: "1234"} test "revoke existing token" do - user = Repo.insert!(@user) + user = Factory.insert(User) token = Repo.insert!(Map.put(@token, :user_id, user.id)) existing_revoked_token = - Repo.insert!(%AccessToken{token: "revoked", revoked_at: NaiveDateTime.utc_now(:second), user_id: user.id}) + Factory.insert(AccessToken, token: "revoked", revoked_at: NaiveDateTime.utc_now(:second), user_id: user.id) TokenGiver.grant_token(user) @@ -27,7 +26,7 @@ defmodule AccentTest.UserRemote.TokenGiver do end test "create token" do - user = Repo.insert!(@user) + user = Factory.insert(User) TokenGiver.grant_token(user) diff --git a/test/badge_generator_test.exs b/test/badge_generator_test.exs index fc3c2e72d..478219218 100644 --- a/test/badge_generator_test.exs +++ b/test/badge_generator_test.exs @@ -13,41 +13,41 @@ defmodule AccentTest.BadgeGenerator do alias Accent.Translation setup do - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project", language_id: french_language.id}) - revision = Repo.insert!(%Revision{language_id: french_language.id, master: true, project_id: project.id}) - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + french_language = Factory.insert(Language) + project = Factory.insert(Project, language_id: french_language.id) + revision = Factory.insert(Revision, language_id: french_language.id, master: true, project_id: project.id) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") {:ok, [project: Repo.preload(project, :revisions), revision: revision, document: document]} end test "percentage_reviewed error", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "a", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "b", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] @@ -59,41 +59,41 @@ defmodule AccentTest.BadgeGenerator do end test "percentage_reviewed warning", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "a", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "b", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "d", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] @@ -105,41 +105,41 @@ defmodule AccentTest.BadgeGenerator do end test "percentage_reviewed success", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "a", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "b", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "d", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] @@ -151,41 +151,41 @@ defmodule AccentTest.BadgeGenerator do end test "translations_count", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "a", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "b", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "d", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] @@ -199,23 +199,23 @@ defmodule AccentTest.BadgeGenerator do end test "conflicts_count", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "d", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] @@ -229,23 +229,23 @@ defmodule AccentTest.BadgeGenerator do end test "reviewed_count", %{project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "c", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "d", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) response = [get: fn _url, _, _ -> {:ok, %{body: ""}} end] diff --git a/test/graphql/helpers/authorization_test.exs b/test/graphql/helpers/authorization_test.exs index 4749f3b08..c21db098c 100644 --- a/test/graphql/helpers/authorization_test.exs +++ b/test/graphql/helpers/authorization_test.exs @@ -15,31 +15,29 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do alias Accent.User alias Accent.Version - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) - version = Repo.insert!(%Version{project_id: project.id, name: "test", tag: "v1.0", user_id: user.id}) - translation = Repo.insert!(%Translation{revision_id: revision.id, key: "test", corrected_text: "bar"}) - collaborator = Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "owner"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") + version = Factory.insert(Version, project_id: project.id, name: "test", tag: "v1.0", user_id: user.id) + translation = Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "bar") + collaborator = Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "owner") integration = - Repo.insert!(%Integration{ + Factory.insert(Integration, project_id: project.id, user_id: user.id, service: "slack", data: %{url: "http://example.com"} - }) + ) translation_comments_subscription = - Repo.insert!(%TranslationCommentsSubscription{translation_id: translation.id, user_id: user.id}) + Factory.insert(TranslationCommentsSubscription, translation_id: translation.id, user_id: user.id) {:ok, [ @@ -127,7 +125,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "unauthorized project root", %{project: project} do - user = Repo.insert!(%User{email: "test+2@test.com"}) + user = Factory.insert(User, email: "test+2@test.com") user = Map.put(user, :permissions, %{}) root = project args = %{} @@ -178,7 +176,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "unauthorized revision root", %{revision: revision} do - user = Repo.insert!(%User{email: "test+2@test.com"}) + user = Factory.insert(User, email: "test+2@test.com") user = Map.put(user, :permissions, %{}) root = revision args = %{} @@ -229,7 +227,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "unauthorized version root", %{version: version} do - user = Repo.insert!(%User{email: "test+2@test.com"}) + user = Factory.insert(User, email: "test+2@test.com") user = Map.put(user, :permissions, %{}) root = version args = %{} @@ -298,7 +296,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "unauthorized translation root", %{translation: translation} do - user = Repo.insert!(%User{email: "test+2@test.com"}) + user = Factory.insert(User, email: "test+2@test.com") user = Map.put(user, :permissions, %{}) root = translation args = %{} @@ -387,7 +385,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "authorized operation revision args", %{user: user, revision: revision, project: project} do - operation = Repo.insert!(%Operation{revision_id: revision.id, user_id: user.id, key: "test", text: "bar"}) + operation = Factory.insert(Operation, revision_id: revision.id, user_id: user.id, key: "test", text: "bar") user = Map.put(user, :permissions, %{project.id => "owner"}) root = nil @@ -401,7 +399,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "authorized operation translation args", %{user: user, translation: translation, project: project} do - operation = Repo.insert!(%Operation{translation_id: translation.id, user_id: user.id, key: "test", text: "bar"}) + operation = Factory.insert(Operation, translation_id: translation.id, user_id: user.id, key: "test", text: "bar") user = Map.put(user, :permissions, %{project.id => "owner"}) root = nil @@ -415,7 +413,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "authorized operation project args", %{user: user, project: project} do - operation = Repo.insert!(%Operation{project_id: project.id, user_id: user.id, key: "test", text: "bar"}) + operation = Factory.insert(Operation, project_id: project.id, user_id: user.id, key: "test", text: "bar") user = Map.put(user, :permissions, %{project.id => "owner"}) root = nil @@ -429,7 +427,7 @@ defmodule AccentTest.GraphQL.Helpers.Authorization do end test "unauthorized operation role", %{user: user, revision: revision, project: project} do - operation = Repo.insert!(%Operation{revision_id: revision.id, user_id: user.id, key: "test", text: "bar"}) + operation = Factory.insert(Operation, revision_id: revision.id, user_id: user.id, key: "test", text: "bar") user = Map.put(user, :permissions, %{project.id => "reviewer"}) root = nil diff --git a/test/graphql/requests/project_integrations_request_test.exs b/test/graphql/requests/project_integrations_request_test.exs index b0183db99..6815fe2c8 100644 --- a/test/graphql/requests/project_integrations_request_test.exs +++ b/test/graphql/requests/project_integrations_request_test.exs @@ -8,21 +8,12 @@ defmodule AccentTest.GraphQL.Requests.ProjectIntegrations do alias Accent.Repo alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - - project = - Repo.insert!(%Project{ - main_color: "#f00", - name: "My project", - last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC") - }) - + user = Factory.insert(User) + project = Factory.insert(Project, last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC")) user = %{user | permissions: %{project.id => "admin"}} - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") create_mutation = """ mutation IntegrationCreate( diff --git a/test/graphql/requests/project_revisions_request_test.exs b/test/graphql/requests/project_revisions_request_test.exs index 871528f6a..999067e4b 100644 --- a/test/graphql/requests/project_revisions_request_test.exs +++ b/test/graphql/requests/project_revisions_request_test.exs @@ -5,35 +5,25 @@ defmodule AccentTest.GraphQL.Requests.ProjectRevisions do alias Accent.Collaborator alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - - project = - Repo.insert!(%Project{ - main_color: "#f00", - name: "My project", - last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC") - }) - + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project, last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC")) user = %{user | permissions: %{project.id => "admin"}} - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: french_language.id, name: "foo", slug: "bar", project_id: project.id, master: true - }) + ) {:ok, [user: user, project: project, revision: revision]} end diff --git a/test/graphql/requests/projects_request_test.exs b/test/graphql/requests/projects_request_test.exs index f4fc66377..3d425bbe7 100644 --- a/test/graphql/requests/projects_request_test.exs +++ b/test/graphql/requests/projects_request_test.exs @@ -5,34 +5,25 @@ defmodule AccentTest.GraphQL.Requests.Projects do alias Accent.Collaborator alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - - project = - Repo.insert!(%Project{ - main_color: "#f00", - name: "My project", - last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC") - }) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project, last_synced_at: DateTime.from_naive!(~N[2017-01-01T00:00:00], "Etc/UTC")) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [user: user, project: project, language: french_language, revision: revision]} end test "list projects", %{user: user, project: project, revision: revision} do - Repo.insert!(%Translation{revision_id: revision.id, key: "A", conflicted: true}) - Repo.insert!(%Translation{revision_id: revision.id, key: "B", conflicted: true}) - Repo.insert!(%Translation{revision_id: revision.id, key: "C", conflicted: false}) + Factory.insert(Translation, revision_id: revision.id, key: "A", conflicted: true) + Factory.insert(Translation, revision_id: revision.id, key: "B", conflicted: true) + Factory.insert(Translation, revision_id: revision.id, key: "C", conflicted: false) {:ok, data} = Absinthe.run( @@ -44,9 +35,6 @@ defmodule AccentTest.GraphQL.Requests.Projects do id name lastSyncedAt - translationsCount - conflictsCount - reviewedCount } } } @@ -61,9 +49,5 @@ defmodule AccentTest.GraphQL.Requests.Projects do assert get_in(data, [:data, "viewer", "projects", "entries", Access.at(0), "lastSyncedAt"]) === "2017-01-01T00:00:00Z" - - assert get_in(data, [:data, "viewer", "projects", "entries", Access.at(0), "translationsCount"]) === 3 - assert get_in(data, [:data, "viewer", "projects", "entries", Access.at(0), "reviewedCount"]) === 1 - assert get_in(data, [:data, "viewer", "projects", "entries", Access.at(0), "conflictsCount"]) === 2 end end diff --git a/test/graphql/resolvers/activity_test.exs b/test/graphql/resolvers/activity_test.exs index 7a638570f..ad5f59cdf 100644 --- a/test/graphql/resolvers/activity_test.exs +++ b/test/graphql/resolvers/activity_test.exs @@ -6,7 +6,6 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do alias Accent.Language alias Accent.Operation alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User @@ -18,25 +17,23 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: language.id, project_id: project.id, master: true) translation = - Repo.insert!(%Translation{revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar"}) + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar") {:ok, [user: user, project: project, revision: revision, translation: translation]} end test "list activities", %{user: user, project: project, translation: translation, revision: revision} do - operation = Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + operation = Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") - Repo.insert!(%Operation{ + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, revision_id: revision.id, @@ -44,7 +41,7 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do text: "foo", action: "update", batch_operation_id: operation.id - }) + ) {:ok, %{entries: entries, meta: meta}} = Resolver.list_operations(operation, %{}, %{}) @@ -57,16 +54,16 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do end test "list project", %{user: user, project: project, translation: translation, revision: revision} do - Repo.insert!(%Operation{ + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, revision_id: revision.id, key: translation.key, text: "foo", action: "update" - }) + ) - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries, meta: meta}} = Resolver.list_project(project, %{}, %{}) assert Enum.count(entries) == 2 @@ -78,9 +75,7 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do end test "list project paginated", %{user: user, project: project} do - for _index <- 1..100 do - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) - end + Factory.seed(Operation, 100, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries, meta: meta}} = Resolver.list_project(project, %{page: 3}, %{}) @@ -93,41 +88,41 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do end test "list project from user", %{user: user, project: project} do - other_user = Repo.insert!(%User{email: "foo@bar.com"}) - Repo.insert!(%Operation{user_id: other_user.id, project_id: project.id, action: "sync"}) - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + other_user = Factory.insert(User, email: "foo@bar.com") + Factory.insert(Operation, user_id: other_user.id, project_id: project.id, action: "sync") + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries}} = Resolver.list_project(project, %{user_id: other_user.id}, %{}) assert Enum.count(entries) == 1 end test "list project from batch", %{user: user, project: project} do - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync", batch: true}) - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync", batch: true) + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries}} = Resolver.list_project(project, %{is_batch: true}, %{}) assert Enum.count(entries) == 1 end test "list project from action", %{user: user, project: project} do - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "delete_document"}) - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "delete_document") + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries}} = Resolver.list_project(project, %{action: "sync"}, %{}) assert Enum.count(entries) == 1 end test "list translation", %{user: user, project: project, translation: translation, revision: revision} do - Repo.insert!(%Operation{ + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, revision_id: revision.id, key: translation.key, text: "foo", action: "update" - }) + ) - Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{entries: entries, meta: meta}} = Resolver.list_translation(translation, %{}, %{}) assert Enum.count(entries) == 1 @@ -139,32 +134,32 @@ defmodule AccentTest.GraphQL.Resolvers.Activity do end test "list translation from user", %{user: user, translation: translation} do - other_user = Repo.insert!(%User{email: "foo@bar.com"}) - Repo.insert!(%Operation{user_id: other_user.id, translation_id: translation.id, action: "update"}) - Repo.insert!(%Operation{user_id: user.id, translation_id: translation.id, action: "update"}) + other_user = Factory.insert(User, email: "foo@bar.com") + Factory.insert(Operation, user_id: other_user.id, translation_id: translation.id, action: "update") + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, action: "update") {:ok, %{entries: entries}} = Resolver.list_translation(translation, %{user_id: other_user.id}, %{}) assert Enum.count(entries) == 1 end test "list translation from batch", %{user: user, translation: translation} do - Repo.insert!(%Operation{user_id: user.id, translation_id: translation.id, action: "sync", batch: true}) - Repo.insert!(%Operation{user_id: user.id, translation_id: translation.id, action: "update"}) + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, action: "sync", batch: true) + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, action: "update") {:ok, %{entries: entries}} = Resolver.list_translation(translation, %{is_batch: true}, %{}) assert Enum.count(entries) == 1 end test "list translation from action", %{user: user, translation: translation} do - Repo.insert!(%Operation{user_id: user.id, translation_id: translation.id, action: "delete_document"}) - Repo.insert!(%Operation{user_id: user.id, translation_id: translation.id, action: "update"}) + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, action: "delete_document") + Factory.insert(Operation, user_id: user.id, translation_id: translation.id, action: "update") {:ok, %{entries: entries}} = Resolver.list_translation(translation, %{action: "update"}, %{}) assert Enum.count(entries) == 1 end test "show project", %{user: user, project: project} do - operation = Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + operation = Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, %{id: id}} = Resolver.show_project(project, %{id: operation.id}, %{}) diff --git a/test/graphql/resolvers/collaborator_test.exs b/test/graphql/resolvers/collaborator_test.exs index 02f8a0492..ef68bf61f 100644 --- a/test/graphql/resolvers/collaborator_test.exs +++ b/test/graphql/resolvers/collaborator_test.exs @@ -13,11 +13,9 @@ defmodule AccentTest.GraphQL.Resolvers.Collaborator do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + project = Factory.insert(Project) {:ok, [user: user, project: project]} end @@ -44,7 +42,7 @@ defmodule AccentTest.GraphQL.Resolvers.Collaborator do test "update", %{project: project, user: user} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - collaborator = Repo.insert!(%Collaborator{email: "test@example.com", role: "reviewer", project_id: project.id}) + collaborator = Factory.insert(Collaborator, email: "test@example.com", role: "reviewer", project_id: project.id) {:ok, result} = Resolver.update(collaborator, %{role: "owner"}, context) @@ -54,7 +52,7 @@ defmodule AccentTest.GraphQL.Resolvers.Collaborator do test "delete", %{project: project, user: user} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - collaborator = Repo.insert!(%Collaborator{email: "test@example.com", role: "reviewer", project_id: project.id}) + collaborator = Factory.insert(Collaborator, email: "test@example.com", role: "reviewer", project_id: project.id) {:ok, result} = Resolver.delete(collaborator, %{}, context) diff --git a/test/graphql/resolvers/comment_test.exs b/test/graphql/resolvers/comment_test.exs index e761c4410..107b483e5 100644 --- a/test/graphql/resolvers/comment_test.exs +++ b/test/graphql/resolvers/comment_test.exs @@ -16,17 +16,15 @@ defmodule AccentTest.GraphQL.Resolvers.Comment do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) translation = - Repo.insert!(%Translation{revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar"}) + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar") {:ok, [user: user, project: project, translation: translation]} end @@ -55,7 +53,7 @@ defmodule AccentTest.GraphQL.Resolvers.Comment do end test "delete", %{translation: translation, user: user} do - comment = Repo.insert!(%Comment{translation_id: translation.id, text: "test", user: user}) + comment = Factory.insert(Comment, translation_id: translation.id, text: "test", user: user) assert get_in(Repo.all(Comment), [Access.all(), Access.key(:id)]) == [comment.id] @@ -66,7 +64,7 @@ defmodule AccentTest.GraphQL.Resolvers.Comment do end test "update", %{translation: translation, user: user} do - comment = Repo.insert!(%Comment{translation_id: translation.id, text: "test", user: user}) + comment = Factory.insert(Comment, translation_id: translation.id, text: "test", user: user) assert get_in(Repo.all(Comment), [Access.all(), Access.key(:id)]) == [comment.id] @@ -77,7 +75,7 @@ defmodule AccentTest.GraphQL.Resolvers.Comment do end test "list project", %{project: project, translation: translation, user: user} do - comment = Repo.insert!(%Comment{translation_id: translation.id, text: "test", user: user}) + comment = Factory.insert(Comment, translation_id: translation.id, text: "test", user: user) {:ok, result} = Resolver.list_project(project, %{}, %{}) @@ -85,7 +83,7 @@ defmodule AccentTest.GraphQL.Resolvers.Comment do end test "list translation", %{translation: translation, user: user} do - comment = Repo.insert!(%Comment{translation_id: translation.id, text: "test", user: user}) + comment = Factory.insert(Comment, translation_id: translation.id, text: "test", user: user) {:ok, result} = Resolver.list_translation(translation, %{}, %{}) diff --git a/test/graphql/resolvers/document_test.exs b/test/graphql/resolvers/document_test.exs index 117e5db6c..94ea69b2f 100644 --- a/test/graphql/resolvers/document_test.exs +++ b/test/graphql/resolvers/document_test.exs @@ -16,34 +16,32 @@ defmodule AccentTest.GraphQL.Resolvers.Document do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) document = - Repo.insert!(%Document{ + Factory.insert(Document, project_id: project.id, path: "test", format: "json", updated_at: DateTime.from_unix!(1_432_560_368_868_569, :microsecond) - }) + ) {:ok, [user: user, project: project, document: document, revision: revision]} end test "delete", %{document: document, revision: revision, user: user} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, document_id: document.id, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} {:ok, result} = Resolver.delete(document, %{}, context) @@ -71,12 +69,12 @@ defmodule AccentTest.GraphQL.Resolvers.Document do test "update with existing path", %{document: document, project: project, user: user} do other_document = - Repo.insert!(%Document{ + Factory.insert(Document, project_id: project.id, path: "test2", format: "json", updated_at: DateTime.add(document.updated_at, 3600, :second) - }) + ) context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} {:ok, result} = Resolver.update(document, %{path: other_document.path}, context) @@ -87,14 +85,14 @@ defmodule AccentTest.GraphQL.Resolvers.Document do end test "show project", %{document: document, project: project, revision: revision} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, document_id: document.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: false - }) + ) {:ok, result} = Resolver.show_project(project, %{id: document.id}, %{}) @@ -106,32 +104,32 @@ defmodule AccentTest.GraphQL.Resolvers.Document do test "list project", %{document: document, project: project, revision: revision} do other_document = - Repo.insert!(%Document{ + Factory.insert(Document, project_id: project.id, path: "test2", format: "json", updated_at: DateTime.add(document.updated_at, 3600, :second) - }) + ) - _empty_document = Repo.insert!(%Document{project_id: project.id, path: "test3", format: "json"}) + _empty_document = Factory.insert(Document, project_id: project.id, path: "test3", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, document_id: document.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: false - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, document_id: other_document.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: true - }) + ) {:ok, result} = Resolver.list_project(project, %{exclude_empty_translations: true}, %{}) @@ -142,17 +140,17 @@ defmodule AccentTest.GraphQL.Resolvers.Document do end test "list project with many deleted documents", %{document: document, project: project, revision: revision} do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, document_id: document.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: false - }) + ) for i <- 1..80 do - Repo.insert!(%Document{project_id: project.id, path: "doc-#{i}", format: "json"}) + Factory.insert(Document, project_id: project.id, path: "doc-#{i}", format: "json") end {:ok, result} = Resolver.list_project(project, %{exclude_empty_translations: true}, %{}) diff --git a/test/graphql/resolvers/integration_test.exs b/test/graphql/resolvers/integration_test.exs index aef6ddf15..43ef90262 100644 --- a/test/graphql/resolvers/integration_test.exs +++ b/test/graphql/resolvers/integration_test.exs @@ -13,11 +13,9 @@ defmodule AccentTest.GraphQL.Resolvers.Integration do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + project = Factory.insert(Project) {:ok, [user: user, project: project]} end @@ -77,7 +75,8 @@ defmodule AccentTest.GraphQL.Resolvers.Integration do assert integration.errors == [ service: - {"is invalid", [validation: :inclusion, enum: ["slack", "github", "discord", "azure_storage_container"]]} + {"is invalid", + [validation: :inclusion, enum: ["slack", "github", "discord", "azure_storage_container", "aws_s3"]]} ] assert Repo.all(Integration) == [] @@ -87,13 +86,13 @@ defmodule AccentTest.GraphQL.Resolvers.Integration do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} integration = - Repo.insert!(%Integration{ + Factory.insert(Integration, project_id: project.id, user_id: user.id, service: "slack", events: ["sync"], - data: %{url: "http://google.ca"} - }) + data: %{id: Ecto.UUID.generate(), url: "http://google.ca"} + ) {:ok, updated_integration} = Resolver.update(integration, %{data: %{url: "http://example.com/update"}}, context) @@ -104,13 +103,13 @@ defmodule AccentTest.GraphQL.Resolvers.Integration do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} integration = - Repo.insert!(%Integration{ + Factory.insert(Integration, project_id: project.id, user_id: user.id, service: "slack", events: ["sync"], - data: %{url: "http://google.ca"} - }) + data: %{id: Ecto.UUID.generate(), url: "http://google.ca"} + ) {:ok, deleted_integration} = Resolver.delete(integration, %{}, context) diff --git a/test/graphql/resolvers/lint_test.exs b/test/graphql/resolvers/lint_test.exs index e07738019..fa1d7c544 100644 --- a/test/graphql/resolvers/lint_test.exs +++ b/test/graphql/resolvers/lint_test.exs @@ -7,7 +7,6 @@ defmodule AccentTest.GraphQL.Resolvers.Lint do alias Accent.Lint.Message alias Accent.Lint.Replacement alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User @@ -17,15 +16,13 @@ defmodule AccentTest.GraphQL.Resolvers.Lint do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) revision = - Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true, slug: "fr"}) + Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true, slug: "fr") context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} @@ -34,23 +31,23 @@ defmodule AccentTest.GraphQL.Resolvers.Lint do test "lint", %{revision: revision, context: context} do master_translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision: revision, conflicted: false, key: "ok2", corrected_text: "bar foo", proposed_text: "bar" - }) + ) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision: revision, master_translation: master_translation, conflicted: false, key: "ok", corrected_text: " bar foo", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.lint_batched_translation(translation, %{}, context) diff --git a/test/graphql/resolvers/operation_test.exs b/test/graphql/resolvers/operation_test.exs index 51ff61127..26711b12d 100644 --- a/test/graphql/resolvers/operation_test.exs +++ b/test/graphql/resolvers/operation_test.exs @@ -12,14 +12,12 @@ defmodule AccentTest.GraphQL.Resolvers.Operation do alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) context = %{context: %{conn: %Plug.Conn{assigns: %{current_user: user}}}} {:ok, [user: user, project: project, revision: revision, context: context]} @@ -27,13 +25,13 @@ defmodule AccentTest.GraphQL.Resolvers.Operation do test "rollback", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "baz", proposed_text: "bar" - }) + ) previous_translation = %PreviousTranslation{ conflicted: false, diff --git a/test/graphql/resolvers/permission_test.exs b/test/graphql/resolvers/permission_test.exs index 5de466204..a6bed8639 100644 --- a/test/graphql/resolvers/permission_test.exs +++ b/test/graphql/resolvers/permission_test.exs @@ -4,7 +4,6 @@ defmodule AccentTest.GraphQL.Resolvers.Permission do alias Accent.GraphQL.Resolvers.Permission, as: Resolver alias Accent.Project - alias Accent.Repo alias Accent.User defmodule PlugConn do @@ -12,11 +11,9 @@ defmodule AccentTest.GraphQL.Resolvers.Permission do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + project = Factory.insert(Project) {:ok, [user: user, project: project]} end diff --git a/test/graphql/resolvers/project_test.exs b/test/graphql/resolvers/project_test.exs index 1d7a6892d..de554e96a 100644 --- a/test/graphql/resolvers/project_test.exs +++ b/test/graphql/resolvers/project_test.exs @@ -2,6 +2,7 @@ defmodule AccentTest.GraphQL.Resolvers.Project do @moduledoc false use Accent.RepoCase, async: true + alias Accent.Collaborator alias Accent.GraphQL.Resolvers.Project, as: Resolver alias Accent.Language alias Accent.Operation @@ -17,14 +18,15 @@ defmodule AccentTest.GraphQL.Resolvers.Project do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = - ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) + ProjectCreator.create( + params: %{main_color: "#f00", name: "My project", language_id: language.id}, + user: user + ) user = %{user | permissions: %{project.id => "owner"}} @@ -37,7 +39,7 @@ defmodule AccentTest.GraphQL.Resolvers.Project do end test "list viewer", %{user: user, project: project} do - Repo.insert!(%Project{main_color: "#f00", name: "Other project"}) + Factory.insert(Project) {:ok, result} = Resolver.list_viewer(user, %{}, %{}) @@ -50,7 +52,7 @@ defmodule AccentTest.GraphQL.Resolvers.Project do end test "list viewer search", %{user: user, language: language} do - Repo.insert!(%Project{main_color: "#f00", name: "Other project"}) + Factory.insert(Project) {:ok, project_two} = ProjectCreator.create( @@ -112,26 +114,19 @@ defmodule AccentTest.GraphQL.Resolvers.Project do assert meta.previous_page == 1 end - test "list viewer ordering", %{user: user, language: language, project: project_one} do - Repo.insert!(%Project{main_color: "#f00", name: "Other project"}) - - {:ok, project_two} = - ProjectCreator.create( - params: %{main_color: "#f00", name: "X - My second project", language_id: language.id}, - user: user - ) + test "list viewer ordering", %{user: user, project: project_one} do + Factory.insert(Project) - {:ok, project_three} = - ProjectCreator.create( - params: %{main_color: "#f00", name: "A - My third project", language_id: language.id}, - user: user - ) + project_two = Factory.insert(Project, last_synced_at: ~U[2020-01-01T00:00:00Z]) + project_three = Factory.insert(Project, last_synced_at: ~U[2022-02-02T00:00:00Z]) + Factory.insert(Collaborator, project_id: project_two.id, user_id: user.id, role: "admin") + Factory.insert(Collaborator, project_id: project_three.id, user_id: user.id, role: "admin") {:ok, result} = Resolver.list_viewer(user, %{}, %{}) assert get_in(result, [:entries, Access.all(), Access.key(:id)]) == [ - project_three.id, project_one.id, + project_three.id, project_two.id ] end @@ -203,7 +198,7 @@ defmodule AccentTest.GraphQL.Resolvers.Project do test "get latest activity", %{user: user, project: project} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - operation = Repo.insert!(%Operation{user_id: user.id, project_id: project.id, action: "sync"}) + operation = Factory.insert(Operation, user_id: user.id, project_id: project.id, action: "sync") {:ok, latest_activity} = Resolver.last_activity(project, %{}, context) @@ -212,7 +207,7 @@ defmodule AccentTest.GraphQL.Resolvers.Project do test "lint_translations", %{user: user, project: project} do [revision] = project.revisions - Repo.insert!(%Translation{revision_id: revision.id, key: "a", proposed_text: " A", corrected_text: " A"}) + Factory.insert(Translation, revision_id: revision.id, key: "a", proposed_text: " A", corrected_text: " A") context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} @@ -229,16 +224,16 @@ defmodule AccentTest.GraphQL.Resolvers.Project do test "lint_translations on current version only", %{user: user, project: project} do [revision] = project.revisions - version = Repo.insert!(%Version{project_id: project.id, name: "foo", tag: "bar", user_id: user.id}) - Repo.insert!(%Translation{revision_id: revision.id, key: "a", proposed_text: " A", corrected_text: " A"}) + version = Factory.insert(Version, project_id: project.id, name: "foo", tag: "bar", user_id: user.id) + Factory.insert(Translation, revision_id: revision.id, key: "a", proposed_text: " A", corrected_text: " A") - Repo.insert!(%Translation{ + Factory.insert(Translation, version_id: version.id, revision_id: revision.id, key: "b", proposed_text: " B", corrected_text: " B" - }) + ) context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} diff --git a/test/graphql/resolvers/revision_test.exs b/test/graphql/resolvers/revision_test.exs index 6a408f494..ff847f956 100644 --- a/test/graphql/resolvers/revision_test.exs +++ b/test/graphql/resolvers/revision_test.exs @@ -5,7 +5,6 @@ defmodule AccentTest.GraphQL.Resolvers.Revision do alias Accent.GraphQL.Resolvers.Revision, as: Resolver alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User @@ -15,23 +14,21 @@ defmodule AccentTest.GraphQL.Resolvers.Revision do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - english_language = Repo.insert!(%Language{name: "english"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + english_language = Factory.insert(Language, name: "english") + project = Factory.insert(Project) - master_revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + master_revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) slave_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: master_revision.id - }) + ) {:ok, [user: user, project: project, master_revision: master_revision, slave_revision: slave_revision]} end @@ -50,7 +47,7 @@ defmodule AccentTest.GraphQL.Resolvers.Revision do test "create", %{project: project, user: user} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - language = Repo.insert!(%Language{name: "spanish"}) + language = Factory.insert(Language, name: "spanish") {:ok, result} = Resolver.create(project, %{language_id: language.id}, context) @@ -75,13 +72,13 @@ defmodule AccentTest.GraphQL.Resolvers.Revision do test "correct all", %{master_revision: revision, user: user} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: true - }) + ) {:ok, result} = Resolver.correct_all(revision, %{}, context) @@ -93,13 +90,13 @@ defmodule AccentTest.GraphQL.Resolvers.Revision do test "uncorrect all", %{master_revision: revision, user: user} do context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", conflicted: false - }) + ) {:ok, result} = Resolver.uncorrect_all(revision, %{}, context) diff --git a/test/graphql/resolvers/translation_comment_subscription_test.exs b/test/graphql/resolvers/translation_comment_subscription_test.exs index 834d6e1c5..c733f9036 100644 --- a/test/graphql/resolvers/translation_comment_subscription_test.exs +++ b/test/graphql/resolvers/translation_comment_subscription_test.exs @@ -20,21 +20,19 @@ defmodule AccentTest.GraphQL.Resolvers.TranslationCommentSubscription do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [user: user, project: project, revision: revision]} end test "create", %{user: user, revision: revision} do translation = - Repo.insert!(%Translation{revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar"}) + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar") context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} @@ -46,10 +44,10 @@ defmodule AccentTest.GraphQL.Resolvers.TranslationCommentSubscription do test "delete", %{user: user, revision: revision} do translation = - Repo.insert!(%Translation{revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar"}) + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar") context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} - subscription = Repo.insert!(%TranslationCommentsSubscription{user_id: user.id, translation_id: translation.id}) + subscription = Factory.insert(TranslationCommentsSubscription, user_id: user.id, translation_id: translation.id) {:ok, result} = Resolver.delete(subscription, %{}, context) diff --git a/test/graphql/resolvers/translation_test.exs b/test/graphql/resolvers/translation_test.exs index 347815723..7dea7e80f 100644 --- a/test/graphql/resolvers/translation_test.exs +++ b/test/graphql/resolvers/translation_test.exs @@ -17,14 +17,12 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) context = %{context: %{conn: %PlugConn{assigns: %{current_user: user}}}} {:ok, [user: user, project: project, revision: revision, context: context]} @@ -42,13 +40,13 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "correct", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.correct(translation, %{text: "Corrected text"}, context) @@ -60,13 +58,13 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "uncorrect", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: false, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.uncorrect(translation, %{text: "baz"}, context) @@ -79,13 +77,13 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "update", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.update(translation, %{text: "Updated text"}, context) @@ -97,13 +95,13 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "show project", %{project: project, revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.show_project(project, %{id: translation.id}, context) @@ -118,13 +116,13 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "show project unknown project", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) {:ok, result} = Resolver.show_project(%Project{id: Ecto.UUID.generate()}, %{id: translation.id}, context) @@ -133,22 +131,22 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "list revision", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "hidden", corrected_text: "bar", proposed_text: "bar", locked: true - }) + ) {:ok, result} = Resolver.list_revision(revision, %{}, context) @@ -157,21 +155,21 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "list revision with query", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "aux", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.list_revision(revision, %{query: "bar"}, context) @@ -179,27 +177,27 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do end test "list revision with document", %{project: project, revision: revision, context: context} do - document = Repo.insert!(%Document{path: "bar", format: "json", project_id: project.id}) - other_document = Repo.insert!(%Document{path: "foo", format: "json", project_id: project.id}) + document = Factory.insert(Document, path: "bar", format: "json", project_id: project.id) + other_document = Factory.insert(Document, path: "foo", format: "json", project_id: project.id) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "foo", proposed_text: "foo", document_id: other_document.id - }) + ) {:ok, result} = Resolver.list_revision(revision, %{document: document.id}, context) @@ -208,22 +206,22 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "list revision with order", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "aaaaaa", corrected_text: "bar", proposed_text: "bar" - }) + ) other_translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "bbbbb", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.list_revision(revision, %{order: "-key"}, context) @@ -232,21 +230,21 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do test "list revision with conflicted", %{revision: revision, context: context} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: false, key: "bar", corrected_text: "bar", proposed_text: "bar" - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "foo", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.list_revision(revision, %{is_conflicted: false}, context) @@ -254,27 +252,27 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do end test "list revision with version", %{project: project, revision: revision, user: user, context: context} do - version = Repo.insert!(%Version{name: "bar", tag: "v1.0", project_id: project.id, user_id: user.id}) - other_version = Repo.insert!(%Version{name: "foo", tag: "v2.0", project_id: project.id, user_id: user.id}) + version = Factory.insert(Version, name: "bar", tag: "v1.0", project_id: project.id, user_id: user.id) + other_version = Factory.insert(Version, name: "foo", tag: "v2.0", project_id: project.id, user_id: user.id) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar", version_id: version.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "foo", proposed_text: "foo", version_id: other_version.id - }) + ) {:ok, result} = Resolver.list_revision(revision, %{version: version.id}, context) @@ -282,33 +280,33 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do end test "related translations", %{project: project, revision: revision, context: context} do - english_language = Repo.insert!(%Language{name: "english"}) + english_language = Factory.insert(Language, name: "english") other_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: revision.id - }) + ) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) other_translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: other_revision.id, conflicted: true, key: "ok", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.related_translations(translation, %{}, context) @@ -316,33 +314,33 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do end test "master translation", %{project: project, revision: revision, context: context} do - english_language = Repo.insert!(%Language{name: "english"}) + english_language = Factory.insert(Language, name: "english") other_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: revision.id - }) + ) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) other_translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: other_revision.id, conflicted: true, key: "ok", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.master_translation(other_translation, %{}, context) @@ -350,32 +348,32 @@ defmodule AccentTest.GraphQL.Resolvers.Translation do end test "master translation as master", %{project: project, revision: revision, context: context} do - english_language = Repo.insert!(%Language{name: "english"}) + english_language = Factory.insert(Language, name: "english") other_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: revision.id - }) + ) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, conflicted: true, key: "ok", corrected_text: "bar", proposed_text: "bar" - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: other_revision.id, conflicted: true, key: "ok", corrected_text: "foo", proposed_text: "foo" - }) + ) {:ok, result} = Resolver.master_translation(translation, %{}, context) diff --git a/test/graphql/resolvers/version_test.exs b/test/graphql/resolvers/version_test.exs index 3b2f68209..9199bad79 100644 --- a/test/graphql/resolvers/version_test.exs +++ b/test/graphql/resolvers/version_test.exs @@ -4,7 +4,6 @@ defmodule AccentTest.GraphQL.Resolvers.Version do alias Accent.GraphQL.Resolvers.Version, as: Resolver alias Accent.Project - alias Accent.Repo alias Accent.User alias Accent.Version @@ -13,13 +12,11 @@ defmodule AccentTest.GraphQL.Resolvers.Version do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + project = Factory.insert(Project) - version = Repo.insert!(%Version{name: "version1", tag: "v1", project_id: project.id, user_id: user.id}) + version = Factory.insert(Version, name: "version1", tag: "v1", project_id: project.id, user_id: user.id) {:ok, [user: user, project: project, version: version]} end diff --git a/test/graphql/resolvers/viewer_test.exs b/test/graphql/resolvers/viewer_test.exs index 39585c0fb..e3714f23b 100644 --- a/test/graphql/resolvers/viewer_test.exs +++ b/test/graphql/resolvers/viewer_test.exs @@ -5,7 +5,6 @@ defmodule AccentTest.GraphQL.Resolvers.Viewer do alias Accent.GraphQL.Resolvers.Viewer, as: Resolver alias Accent.Language alias Accent.ProjectCreator - alias Accent.Repo alias Accent.User defmodule PlugConn do @@ -13,11 +12,9 @@ defmodule AccentTest.GraphQL.Resolvers.Viewer do defstruct [:assigns] end - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) diff --git a/test/hook/hook_test.exs b/test/hook/hook_test.exs index 31a82b2fc..5dbba8e5f 100644 --- a/test/hook/hook_test.exs +++ b/test/hook/hook_test.exs @@ -4,12 +4,11 @@ defmodule AccentTest.Hook do alias Accent.Hook alias Accent.Project - alias Accent.Repo alias Accent.User setup do - project = Repo.insert!(%Project{main_color: "#f00", name: "Test"}) - user = Repo.insert!(%User{fullname: "Test", email: "foo@test.com"}) + project = Factory.insert(Project) + user = Factory.insert(User, fullname: "Test", email: "foo@test.com") payload = %{test: "hook"} context = %Hook.Context{project_id: project.id, user_id: user.id, event: "event", payload: payload} @@ -31,13 +30,4 @@ defmodule AccentTest.Hook do args: %{worker_args | "event" => "sync"} ) end - - test "unsupported event", %{context: context, worker_args: worker_args} do - Hook.outbound(%{context | event: "foobar"}) - - refute_enqueued( - worker: Hook.Outbounds.Mock, - args: %{worker_args | "event" => "foobar"} - ) - end end diff --git a/test/hook/outbounds/discord_test.exs b/test/hook/outbounds/discord_test.exs index f81da9b97..6882be206 100644 --- a/test/hook/outbounds/discord_test.exs +++ b/test/hook/outbounds/discord_test.exs @@ -1,26 +1,25 @@ defmodule AccentTest.Hook.Outbounds.Discord do @moduledoc false - use Accent.RepoCase, async: true + use Accent.RepoCase, async: false import Mock alias Accent.Hook.Outbounds.Discord alias Accent.Integration alias Accent.Project - alias Accent.Repo alias Accent.User setup do - project = Repo.insert!(%Project{main_color: "#f00", name: "Test"}) - user = Repo.insert!(%User{fullname: "Test", email: "foo@test.com"}) + project = Factory.insert(Project) + user = Factory.insert(User, fullname: "Test", email: "foo@test.com") - Repo.insert!(%Integration{ - project: project, - user: user, + Factory.insert(Integration, + project_id: project.id, + user_id: user.id, service: "discord", events: ["sync"], data: %{url: "http://example.com"} - }) + ) [project: project, user: user] end diff --git a/test/hook/outbounds/email_test.exs b/test/hook/outbounds/email_test.exs index b6e954ef7..677095554 100644 --- a/test/hook/outbounds/email_test.exs +++ b/test/hook/outbounds/email_test.exs @@ -17,20 +17,20 @@ defmodule AccentTest.Hook.Outbounds.Email do alias Accent.User setup do - language = Repo.insert!(%Language{name: "Test"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "Test"}) - user = Repo.insert!(%User{fullname: "Test", email: "foo@test.com"}) - revision = Repo.insert!(%Revision{project_id: project.id, language_id: language.id, master: true}) + language = Factory.insert(Language, name: "Test") + project = Factory.insert(Project) + user = Factory.insert(User, fullname: "Test", email: "foo@test.com") + revision = Factory.insert(Revision, project_id: project.id, language_id: language.id, master: true) translation = - Repo.insert!(%Translation{key: "foo", corrected_text: "bar", proposed_text: "bar", revision_id: revision.id}) + Factory.insert(Translation, key: "foo", corrected_text: "bar", proposed_text: "bar", revision_id: revision.id) [project: project, translation: translation, user: user] end test "commenter subscribed", %{project: project, translation: translation, user: user} do - Repo.insert!(%TranslationCommentsSubscription{translation_id: translation.id, user_id: user.id}) - comment = Repo.insert!(%Comment{translation_id: translation.id, user_id: user.id, text: "This is a comment"}) + Factory.insert(TranslationCommentsSubscription, translation_id: translation.id, user_id: user.id) + comment = Factory.insert(Comment, translation_id: translation.id, user_id: user.id, text: "This is a comment") comment = Repo.preload(comment, [:user, translation: [revision: :project]]) payload = %{ @@ -52,9 +52,9 @@ defmodule AccentTest.Hook.Outbounds.Email do end test "comment", %{project: project, translation: translation, user: user} do - Repo.insert!(%TranslationCommentsSubscription{translation_id: translation.id, user_id: user.id}) - commenter = Repo.insert!(%User{fullname: "Commenter", email: "comment@test.com"}) - comment = Repo.insert!(%Comment{translation_id: translation.id, user_id: commenter.id, text: "This is a comment"}) + Factory.insert(TranslationCommentsSubscription, translation_id: translation.id, user_id: user.id) + commenter = Factory.insert(User, fullname: "Commenter", email: "comment@test.com") + comment = Factory.insert(Comment, translation_id: translation.id, user_id: commenter.id, text: "This is a comment") payload = %{ "text" => comment.text, @@ -75,7 +75,7 @@ defmodule AccentTest.Hook.Outbounds.Email do end test "collaborator", %{project: project, user: user} do - collaborator = Repo.insert!(%Collaborator{email: "collab@test.com", project_id: project.id}) + collaborator = Factory.insert(Collaborator, email: "collab@test.com", project_id: project.id) payload = %{ "collaborator" => %{ diff --git a/test/hook/outbounds/slack_test.exs b/test/hook/outbounds/slack_test.exs index 902d45df3..e29884b68 100644 --- a/test/hook/outbounds/slack_test.exs +++ b/test/hook/outbounds/slack_test.exs @@ -7,27 +7,26 @@ end defmodule AccentTest.Hook.Outbounds.Slack do @moduledoc false - use Accent.RepoCase, async: true + use Accent.RepoCase, async: false import Mock alias Accent.Hook.Outbounds.Slack alias Accent.Integration alias Accent.Project - alias Accent.Repo alias Accent.User setup do - project = Repo.insert!(%Project{main_color: "#f00", name: "Test"}) - user = Repo.insert!(%User{fullname: "Test", email: "foo@test.com"}) + project = Factory.insert(Project) + user = Factory.insert(User, fullname: "Test", email: "foo@test.com") - Repo.insert!(%Integration{ - project: project, - user: user, + Factory.insert(Integration, + project_id: project.id, + user_id: user.id, service: "slack", events: ["sync"], data: %{url: "http://example.com"} - }) + ) [project: project, user: user] end diff --git a/test/hook/outbounds/websocket_test.exs b/test/hook/outbounds/websocket_test.exs index 03faf3fa6..de7f09e15 100644 --- a/test/hook/outbounds/websocket_test.exs +++ b/test/hook/outbounds/websocket_test.exs @@ -15,13 +15,13 @@ defmodule AccentTest.Hook.Outbounds.Websocket do alias Accent.UserSocket setup do - language = Repo.insert!(%Language{name: "Test"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "Test"}) - user = Repo.insert!(%User{fullname: "Test", email: "foo@test.com", permissions: %{project.id => "admin"}}) - revision = Repo.insert!(%Revision{project_id: project.id, language_id: language.id, master: true}) + language = Factory.insert(Language, name: "Test") + project = Factory.insert(Project) + user = Factory.insert(User, fullname: "Test", email: "foo@test.com", permissions: %{project.id => "admin"}) + revision = Factory.insert(Revision, project_id: project.id, language_id: language.id, master: true) translation = - Repo.insert!(%Translation{key: "foo", corrected_text: "bar", proposed_text: "bar", revision_id: revision.id}) + Factory.insert(Translation, key: "foo", corrected_text: "bar", proposed_text: "bar", revision_id: revision.id) {:ok, _, socket} = UserSocket @@ -32,8 +32,8 @@ defmodule AccentTest.Hook.Outbounds.Websocket do end test "comment", %{project: project, translation: translation, user: user} do - commenter = Repo.insert!(%User{fullname: "Commenter", email: "comment@test.com"}) - comment = Repo.insert!(%Comment{translation_id: translation.id, user_id: commenter.id, text: "This is a comment"}) + commenter = Factory.insert(User, fullname: "Commenter", email: "comment@test.com") + comment = Factory.insert(Comment, translation_id: translation.id, user_id: commenter.id, text: "This is a comment") comment = Repo.preload(comment, [:user, translation: [revision: :project]]) payload = %{ @@ -91,7 +91,7 @@ defmodule AccentTest.Hook.Outbounds.Websocket do end test "collaborator", %{project: project, user: user} do - collaborator = Repo.insert!(%Collaborator{email: "collab@test.com", project_id: project.id}) + collaborator = Factory.insert(Collaborator, email: "collab@test.com", project_id: project.id) payload = %{ "collaborator" => %{ diff --git a/test/langue/json/exceptions_test.exs b/test/langue/json/exceptions_test.exs index 2a18b2006..23920be6e 100644 --- a/test/langue/json/exceptions_test.exs +++ b/test/langue/json/exceptions_test.exs @@ -2,20 +2,28 @@ defmodule LangueTest.Formatter.Json.Exception do @moduledoc false use ExUnit.Case, async: true + alias Accent.FormatterTestHelper alias Langue.Formatter.Json + alias LangueTest.Formatter.Json.Expectation.DuplicateKey alias LangueTest.Formatter.Json.Expectation.InvalidFloatValue alias LangueTest.Formatter.Json.Expectation.InvalidIntegerValue Code.require_file("expectation_test.exs", __DIR__) test "invalid integer value" do - {expected_parse, result_parse} = Accent.FormatterTestHelper.test_serialize(InvalidIntegerValue, Json) + {expected_parse, result_parse} = FormatterTestHelper.test_serialize(InvalidIntegerValue, Json) assert expected_parse == result_parse end test "invalid float value" do - {expected_parse, result_parse} = Accent.FormatterTestHelper.test_serialize(InvalidFloatValue, Json) + {expected_parse, result_parse} = FormatterTestHelper.test_serialize(InvalidFloatValue, Json) + + assert expected_parse == result_parse + end + + test "duplicate key" do + {expected_parse, result_parse} = FormatterTestHelper.test_parse(DuplicateKey, Json) assert expected_parse == result_parse end diff --git a/test/langue/json/expectation_test.exs b/test/langue/json/expectation_test.exs index 5d039246e..d2a4e3cfa 100644 --- a/test/langue/json/expectation_test.exs +++ b/test/langue/json/expectation_test.exs @@ -126,6 +126,27 @@ defmodule LangueTest.Formatter.Json.Expectation do end end + defmodule DuplicateKey do + @moduledoc false + use Langue.Expectation.Case + + def render do + """ + { + "test": "foo", + "test": "bar", + "test": "baz" + } + """ + end + + def entries do + [ + %Entry{index: 1, key: "test", value: "foo", value_type: "string"} + ] + end + end + defmodule InvalidFloatValue do @moduledoc false use Langue.Expectation.Case diff --git a/test/machine_translations/machine_translations_test.exs b/test/machine_translations/machine_translations_test.exs index 059a84669..86ee22d19 100644 --- a/test/machine_translations/machine_translations_test.exs +++ b/test/machine_translations/machine_translations_test.exs @@ -12,7 +12,7 @@ defmodule AccentTest.MachineTranslations do %{body: body, url: "https://translation.googleapis.com/v3/projects/1234/:translateText"} -> assert Jason.decode!(body) === %{ "contents" => ["Test"], - "mimeType" => "text/html", + "mimeType" => "text/plain", "sourceLanguageCode" => "fr", "targetLanguageCode" => "en" } @@ -35,7 +35,7 @@ defmodule AccentTest.MachineTranslations do %{body: body, url: "https://translation.googleapis.com/v3/projects/1234/:translateText"} -> assert Jason.decode!(body) === %{ "contents" => [~s(Test %{placeholder} bla)], - "mimeType" => "text/html", + "mimeType" => "text/plain", "sourceLanguageCode" => "fr", "targetLanguageCode" => "en" } @@ -73,5 +73,52 @@ defmodule AccentTest.MachineTranslations do {:error, error} = MachineTranslations.translate(entries, source_language, target_language, config) assert error === "Something" end + + test "deepl" do + mock_global(fn + %{body: body, url: "https://api-free.deepl.com/v2/translate"} -> + assert Jason.decode!(body) === %{"source_lang" => "FR", "target_lang" => "EN", "text" => ["Test"]} + + %Tesla.Env{status: 200, body: %{"translations" => [%{"text" => "Translated"}]}} + end) + + entries = [%Langue.Entry{value: "Test", value_type: "string", key: "."}] + source_language = "fr" + target_language = "en" + provider_config = %{"key" => "test"} + config = %{"provider" => "deepl", "config" => provider_config} + + [entry] = MachineTranslations.translate(entries, source_language, target_language, config) + assert entry.value === "Translated" + end + + test "deepl error" do + mock_global(fn %{url: "https://api-free.deepl.com/v2/translate"} -> + %Tesla.Env{status: 400, body: "Something"} + end) + + entries = [%Langue.Entry{value: "Test", value_type: "string", key: "."}] + source_language = "fr" + target_language = "en" + provider_config = %{"key" => "test"} + config = %{"provider" => "deepl", "config" => provider_config} + + {:error, error} = MachineTranslations.translate(entries, source_language, target_language, config) + assert error === "Something" + end + + test "deepl detect source" do + mock_global(fn %{url: "https://api-free.deepl.com/v2/translate"} -> + %Tesla.Env{status: 400, body: "Something"} + end) + + entries = [%Langue.Entry{value: "Test", value_type: "string", key: "."}] + target_language = "en" + provider_config = %{"key" => "test"} + config = %{"provider" => "deepl", "config" => provider_config} + + {:error, error} = MachineTranslations.translate(entries, nil, target_language, config) + assert error === "Something" + end end end diff --git a/test/movement/builders/new_slave_test.exs b/test/movement/builders/new_slave_test.exs index a0af99af2..a3dea1695 100644 --- a/test/movement/builders/new_slave_test.exs +++ b/test/movement/builders/new_slave_test.exs @@ -12,24 +12,22 @@ defmodule AccentTest.Movement.Builders.NewSlave do alias Movement.Builders.NewSlave, as: NewSlaveBuilder alias Movement.Context - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project]} end test "builder fetch translations and process operations", %{revision: revision, project: project, document: document} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -37,7 +35,7 @@ defmodule AccentTest.Movement.Builders.NewSlave do file_comment: "comment", revision_id: revision.id, document_id: document.id - }) + ) context = %Context{} @@ -70,7 +68,7 @@ defmodule AccentTest.Movement.Builders.NewSlave do document: document } do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -78,7 +76,7 @@ defmodule AccentTest.Movement.Builders.NewSlave do file_comment: "comment", revision_id: revision.id, document_id: document.id - }) + ) context = %Context{} @@ -95,7 +93,7 @@ defmodule AccentTest.Movement.Builders.NewSlave do test "with removed translation", %{revision: revision, project: project, document: document} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -103,8 +101,12 @@ defmodule AccentTest.Movement.Builders.NewSlave do file_comment: "comment", revision_id: revision.id, document_id: document.id, + conflicted: false, + conflicted_text: "", + value_type: "string", + translated: true, removed: true - }) + ) context = %Context{} @@ -133,6 +135,7 @@ defmodule AccentTest.Movement.Builders.NewSlave do previous_translation: %PreviousTranslation{ value_type: "string", removed: true, + translated: true, conflicted: false, conflicted_text: "", corrected_text: "A", diff --git a/test/movement/builders/new_version_test.exs b/test/movement/builders/new_version_test.exs index f74f2b0c4..fee275c69 100644 --- a/test/movement/builders/new_version_test.exs +++ b/test/movement/builders/new_version_test.exs @@ -11,24 +11,22 @@ defmodule AccentTest.Movement.Builders.NewVersion do alias Accent.Version alias Movement.Builders.NewVersion, as: NewVersionBuilder - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project, user: user]} end test "builder fetch translations and process operations", %{revision: revision, project: project, document: document} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -38,7 +36,7 @@ defmodule AccentTest.Movement.Builders.NewVersion do locked: true, revision_id: revision.id, document_id: document.id - }) + ) context = %Movement.Context{} @@ -70,10 +68,10 @@ defmodule AccentTest.Movement.Builders.NewVersion do end test "builder with existing version", %{revision: revision, project: project, document: document, user: user} do - version = Repo.insert!(%Version{user_id: user.id, tag: "v3.2", name: "Release", project_id: project.id}) + version = Factory.insert(Version, user_id: user.id, tag: "v3.2", name: "Release", project_id: project.id) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -81,9 +79,9 @@ defmodule AccentTest.Movement.Builders.NewVersion do file_comment: "comment", revision_id: revision.id, document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -93,7 +91,7 @@ defmodule AccentTest.Movement.Builders.NewVersion do version_id: version.id, document_id: document.id, source_translation_id: translation.id - }) + ) context = %Movement.Context{} diff --git a/test/movement/builders/project_sync_test.exs b/test/movement/builders/project_sync_test.exs index df1d2f493..ec0fecd23 100644 --- a/test/movement/builders/project_sync_test.exs +++ b/test/movement/builders/project_sync_test.exs @@ -12,19 +12,17 @@ defmodule AccentTest.Movement.Builders.ProjectSync do alias Movement.Builders.ProjectSync, as: ProjectSyncBuilder alias Movement.Context - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) - other_language = Repo.insert!(%Language{name: "French", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) + other_language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - other_revision = Repo.insert!(%Revision{master: false, project_id: project.id, language_id: other_language.id}) - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + other_revision = Factory.insert(Revision, master: false, project_id: project.id, language_id: other_language.id) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project, other_revision: other_revision]} end @@ -35,9 +33,9 @@ defmodule AccentTest.Movement.Builders.ProjectSync do project: project, other_revision: other_revision } do - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) - Repo.insert!(%Translation{key: "b", proposed_text: "B", revision_id: revision.id, document_id: document.id}) - Repo.insert!(%Translation{key: "a", proposed_text: "C", revision_id: other_revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) + Factory.insert(Translation, key: "b", proposed_text: "B", revision_id: revision.id, document_id: document.id) + Factory.insert(Translation, key: "a", proposed_text: "C", revision_id: other_revision.id, document_id: document.id) entries = [%Langue.Entry{key: "a", value: "B", value_type: "string"}] context = diff --git a/test/movement/builders/revision_correct_all_test.exs b/test/movement/builders/revision_correct_all_test.exs index eb88940b4..a2437d488 100644 --- a/test/movement/builders/revision_correct_all_test.exs +++ b/test/movement/builders/revision_correct_all_test.exs @@ -9,11 +9,9 @@ defmodule AccentTest.Movement.Builders.RevisionCorrectAll do alias Accent.User alias Movement.Builders.RevisionCorrectAll, as: RevisionCorrectAllBuilder - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "French", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) @@ -24,7 +22,7 @@ defmodule AccentTest.Movement.Builders.RevisionCorrectAll do end test "builder fetch translations and correct conflict", %{revision: revision} do - translation = Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id}) + translation = Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id) context = %Movement.Context{} @@ -39,7 +37,7 @@ defmodule AccentTest.Movement.Builders.RevisionCorrectAll do end test "builder fetch translations and ignore corrected translation", %{revision: revision} do - Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: false, revision_id: revision.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: false, revision_id: revision.id) context = %Movement.Context{} diff --git a/test/movement/builders/revision_merge_test.exs b/test/movement/builders/revision_merge_test.exs index 82d6d9009..a7075bcb7 100644 --- a/test/movement/builders/revision_merge_test.exs +++ b/test/movement/builders/revision_merge_test.exs @@ -11,20 +11,18 @@ defmodule AccentTest.Movement.Builders.RevisionMerge do alias Movement.Builders.RevisionMerge, as: RevisionMergeBuilder alias Movement.Context - @user %User{email: "test@test.com"} - test "builder fetch translations and use comparer" do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(revisions: [:language]) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) entries = [%Langue.Entry{key: "a", value: "B", value_type: "string"}] diff --git a/test/movement/builders/revision_sync_test.exs b/test/movement/builders/revision_sync_test.exs index 440668ddc..d6e2bd4b1 100644 --- a/test/movement/builders/revision_sync_test.exs +++ b/test/movement/builders/revision_sync_test.exs @@ -11,24 +11,22 @@ defmodule AccentTest.Movement.Builders.RevisionSync do alias Movement.Builders.RevisionSync, as: RevisionSyncBuilder alias Movement.Context - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document]} end test "builder fetch translations and use comparer", %{revision: revision, document: document} do translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) entries = [%Langue.Entry{key: "a", value: "B", value_type: "string"}] @@ -48,7 +46,7 @@ defmodule AccentTest.Movement.Builders.RevisionSync do test "builder fetch translations and process to remove with empty entries", %{revision: revision, document: document} do translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) context = %Context{entries: []} @@ -66,13 +64,13 @@ defmodule AccentTest.Movement.Builders.RevisionSync do test "builder fetch translations and process to renew with entries", %{revision: revision, document: document} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id, removed: true - }) + ) entries = [%Langue.Entry{key: "a", value: "B", value_type: "string"}] diff --git a/test/movement/builders/revision_uncorrect_all_test.exs b/test/movement/builders/revision_uncorrect_all_test.exs index aad9767f0..851c52ba3 100644 --- a/test/movement/builders/revision_uncorrect_all_test.exs +++ b/test/movement/builders/revision_uncorrect_all_test.exs @@ -9,11 +9,9 @@ defmodule AccentTest.Movement.Builders.RevisionUncorrectAll do alias Accent.User alias Movement.Builders.RevisionUncorrectAll, as: RevisionUncorrectAllBuilder - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) @@ -24,7 +22,7 @@ defmodule AccentTest.Movement.Builders.RevisionUncorrectAll do end test "builder fetch translations and uncorrect conflict", %{revision: revision} do - translation = Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: false, revision_id: revision.id}) + translation = Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: false, revision_id: revision.id) context = %Movement.Context{} @@ -39,7 +37,7 @@ defmodule AccentTest.Movement.Builders.RevisionUncorrectAll do end test "builder fetch translations and ignore conflicted translation", %{revision: revision} do - Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id) context = %Movement.Context{} diff --git a/test/movement/builders/rollback_test.exs b/test/movement/builders/rollback_test.exs index 6e1541aad..fa9a6ee2d 100644 --- a/test/movement/builders/rollback_test.exs +++ b/test/movement/builders/rollback_test.exs @@ -12,23 +12,21 @@ defmodule AccentTest.Movement.Builders.Rollback do alias Accent.User alias Movement.Builders.Rollback, as: RollbackBuilder - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project]} end test "builder process operations for batch", %{project: project} do - operation = Repo.insert!(%Operation{project_id: project.id, batch: true, action: "sync"}) + operation = Factory.insert(Operation, project_id: project.id, batch: true, action: "sync") context = %Movement.Context{} @@ -43,16 +41,17 @@ defmodule AccentTest.Movement.Builders.Rollback do test "builder process operations for translation", %{project: project, revision: revision, document: document} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "A", proposed_text: "TEXT", conflicted_text: "Ex-TEXT", corrected_text: "LOL", + translated: true, removed: false, revision_id: revision.id, value_type: "string", placeholders: [] - }) + ) operation = %Operation{ @@ -88,6 +87,7 @@ defmodule AccentTest.Movement.Builders.Rollback do conflicted_text: translation.conflicted_text, conflicted: translation.conflicted, removed: translation.removed, + translated: translation.translated, placeholders: translation.placeholders } end diff --git a/test/movement/builders/slave_conflict_sync_test.exs b/test/movement/builders/slave_conflict_sync_test.exs index b060c4d65..41120d4de 100644 --- a/test/movement/builders/slave_conflict_sync_test.exs +++ b/test/movement/builders/slave_conflict_sync_test.exs @@ -12,19 +12,17 @@ defmodule AccentTest.Movement.Builders.SlaveConflictSync do alias Movement.Builders.SlaveConflictSync, as: SlaveConflictSyncBuilder alias Movement.Context - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) - other_language = Repo.insert!(%Language{name: "French", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) + other_language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - other_revision = Repo.insert!(%Revision{project_id: project.id, language_id: other_language.id}) - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + other_revision = Factory.insert(Revision, project_id: project.id, language_id: other_language.id) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [project: project, revision: revision, document: document, other_revision: other_revision]} end @@ -35,10 +33,15 @@ defmodule AccentTest.Movement.Builders.SlaveConflictSync do other_revision: other_revision } do translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) other_translation = - Repo.insert!(%Translation{key: "a", proposed_text: "C", revision_id: other_revision.id, document_id: document.id}) + Factory.insert(Translation, + key: "a", + proposed_text: "C", + revision_id: other_revision.id, + document_id: document.id + ) context = %Context{operations: [%{key: "a", action: "conflict_on_proposed"}]} @@ -60,18 +63,18 @@ defmodule AccentTest.Movement.Builders.SlaveConflictSync do document: document, other_revision: other_revision } do - other_document = Repo.insert!(%Document{project_id: project.id, path: "other", format: "json"}) + other_document = Factory.insert(Document, project_id: project.id, path: "other", format: "json") translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) other_translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "C", revision_id: other_revision.id, document_id: other_document.id - }) + ) context = %Context{operations: [%{key: "a", action: "conflict_on_proposed"}]} diff --git a/test/movement/builders/translation_update_test.exs b/test/movement/builders/translation_update_test.exs index 70c44e155..4cb7eec3e 100644 --- a/test/movement/builders/translation_update_test.exs +++ b/test/movement/builders/translation_update_test.exs @@ -2,7 +2,12 @@ defmodule AccentTest.Movement.Builders.TranslationUpdate do @moduledoc false use Accent.RepoCase, async: true + alias Accent.Language + alias Accent.ProjectCreator + alias Accent.Repo alias Accent.Translation + alias Accent.User + alias Accent.Version alias Movement.Builders.TranslationUpdate, as: TranslationUpdateBuilder alias Movement.Context @@ -30,6 +35,67 @@ defmodule AccentTest.Movement.Builders.TranslationUpdate do ] end + test "builder copy on latest version" do + user = Factory.insert(User, email: "test@test.com") + language = Factory.insert(Language) + + {:ok, project} = + ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) + + revision = Repo.one!(Ecto.assoc(project, :revisions)) + + version = + Factory.insert(Version, + name: "1", + tag: "v1", + project_id: project.id, + user_id: user.id, + copy_on_update_translation: true + ) + + source_translation = + Factory.insert(Translation, + key: "a", + proposed_text: "A", + corrected_text: "A", + revision_id: revision.id, + version_id: nil + ) + + translation = + Factory.insert(Translation, + key: "a", + proposed_text: "A", + corrected_text: "A", + version_id: version.id, + revision_id: revision.id, + source_translation_id: source_translation.id + ) + + context = + %Context{} + |> Context.assign(:text, "Updated!") + |> Context.assign(:translation, translation) + |> TranslationUpdateBuilder.build() + + operations = Enum.map(context.operations, &Map.take(&1, [:key, :text, :action, :translation_id])) + + assert operations === [ + %{ + translation_id: translation.id, + key: "a", + text: "Updated!", + action: "update" + }, + %{ + translation_id: source_translation.id, + key: "a", + text: "Updated!", + action: "update" + } + ] + end + test "builder same text translated" do translation = %Translation{ key: "a", diff --git a/test/movement/down_test.exs b/test/movement/down_test.exs index 4165527d0..f52d90134 100644 --- a/test/movement/down_test.exs +++ b/test/movement/down_test.exs @@ -25,21 +25,21 @@ defmodule AccentTest.Migrator.Down do } translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_in_conflict", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: nil, proposed_text: "new proposed text", conflicted_text: "corrected_text", conflicted: true - }) + ) Migrator.down( - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "conflict_on_corrected", translation: translation, previous_translation: PreviousTranslation.from_translation(previous_translation) - }) + ) ) new_translation = Repo.get!(Translation, translation.id) @@ -62,21 +62,21 @@ defmodule AccentTest.Migrator.Down do } translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_in_proposed", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: nil, proposed_text: "new proposed text", conflicted_text: "proposed_text", conflicted: true - }) + ) Migrator.down( - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "conflict_on_proposed", translation: translation, previous_translation: PreviousTranslation.from_translation(previous_translation) - }) + ) ) new_translation = Repo.get!(Translation, translation.id) @@ -89,16 +89,16 @@ defmodule AccentTest.Migrator.Down do test ":new" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_added_down", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: nil, proposed_text: "new text", conflicted_text: nil, conflicted: true - }) + ) - Migrator.down(Repo.insert!(%Operation{action: "new", translation: translation})) + Migrator.down(Factory.insert(Operation, action: "new", translation: translation)) new_translation = Repo.get!(Translation, translation.id) @@ -107,16 +107,16 @@ defmodule AccentTest.Migrator.Down do test ":renew" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_added_down", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: nil, proposed_text: "new text", conflicted_text: nil, conflicted: true - }) + ) - Migrator.down(Repo.insert!(%Operation{action: "renew", translation: translation})) + Migrator.down(Factory.insert(Operation, action: "renew", translation: translation)) new_translation = Repo.get!(Translation, translation.id) @@ -125,18 +125,18 @@ defmodule AccentTest.Migrator.Down do test ":remove" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, value_type: "", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), key: "to_be_added_down", corrected_text: nil, proposed_text: "new text", conflicted_text: nil, conflicted: true, placeholders: [] - }) + ) - Migrator.down(Repo.insert!(%Operation{action: "remove", translation: translation})) + Migrator.down(Factory.insert(Operation, action: "remove", translation: translation)) new_translation = Repo.get!(Translation, translation.id) diff --git a/test/movement/persisters/base_test.exs b/test/movement/persisters/base_test.exs index d7c523b4e..02d9c7add 100644 --- a/test/movement/persisters/base_test.exs +++ b/test/movement/persisters/base_test.exs @@ -15,8 +15,8 @@ defmodule AccentTest.Movement.Persisters.Base do alias Movement.Persisters.Base, as: BasePersister test "don’t overwrite revision" do - revision = Repo.insert!(%Revision{}) - revision_two = Repo.insert!(%Revision{}) + revision = Factory.insert(Revision) + revision_two = Factory.insert(Revision) translation = %Translation{ key: "a", @@ -55,10 +55,10 @@ defmodule AccentTest.Movement.Persisters.Base do end test "persist and execute operations" do - user = Repo.insert!(%User{email: "test@test.com"}) - revision = Repo.insert!(%Revision{}) + user = Factory.insert(User, email: "test@test.com") + revision = Factory.insert(Revision) - translation = Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id}) + translation = Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: true, revision_id: revision.id) operations = [ %Movement.Operation{ @@ -92,10 +92,16 @@ defmodule AccentTest.Movement.Persisters.Base do end test "new operation with removed translation" do - revision = Repo.insert!(%Revision{}) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", conflicted: true, removed: true, revision_id: revision.id}) + Factory.insert(Translation, + key: "a", + proposed_text: "A", + conflicted: true, + removed: true, + revision_id: revision.id + ) operations = [ %Movement.Operation{ @@ -120,10 +126,16 @@ defmodule AccentTest.Movement.Persisters.Base do end test "version operation with source translation" do - revision = Repo.insert!(%Revision{}) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{key: "a", revision_id: revision.id, proposed_text: "A", conflicted: true, removed: true}) + Factory.insert(Translation, + key: "a", + revision_id: revision.id, + proposed_text: "A", + conflicted: true, + removed: true + ) operations = [ %Movement.Operation{ @@ -146,10 +158,16 @@ defmodule AccentTest.Movement.Persisters.Base do end test "version operation add operation on source translation" do - revision = Repo.insert!(%Revision{}) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{key: "a", revision_id: revision.id, proposed_text: "A", conflicted: true, removed: true}) + Factory.insert(Translation, + key: "a", + revision_id: revision.id, + proposed_text: "A", + conflicted: true, + removed: true + ) operations = [ %Movement.Operation{ @@ -177,20 +195,20 @@ defmodule AccentTest.Movement.Persisters.Base do end test "update operation add operation on version source translation" do - user = Repo.insert!(%User{email: "user@example.com"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "project"}) - revision = Repo.insert!(%Revision{project_id: project.id}) - version = Repo.insert!(%Version{name: "foo", tag: "0.1", project: project, user: user}) + user = Factory.insert(User, email: "user@example.com") + project = Factory.insert(Project, main_color: "#f00", name: "project") + revision = Factory.insert(Revision, project_id: project.id) + version = Factory.insert(Version, name: "foo", tag: "0.1", project: project, user: user) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", revision_id: revision.id, proposed_text: "A", conflicted: true, removed: true, version: version - }) + ) operations = [ %Movement.Operation{ diff --git a/test/movement/persisters/new_slave_test.exs b/test/movement/persisters/new_slave_test.exs index 47329bda9..dcdc4fadf 100644 --- a/test/movement/persisters/new_slave_test.exs +++ b/test/movement/persisters/new_slave_test.exs @@ -8,11 +8,9 @@ defmodule AccentTest.Movement.Persisters.NewSlave do alias Accent.User alias Movement.Persisters.NewSlave, as: NewSlavePersister - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) @@ -23,7 +21,7 @@ defmodule AccentTest.Movement.Persisters.NewSlave do end test "create revision success", %{project: project, revision: master_revision} do - new_language = Repo.insert!(%Language{name: "French", slug: Ecto.UUID.generate()}) + new_language = Factory.insert(Language) {:ok, {context, _}} = NewSlavePersister.persist(%Movement.Context{ diff --git a/test/movement/persisters/new_version_test.exs b/test/movement/persisters/new_version_test.exs index 66ebd8fd3..d4c4999d9 100644 --- a/test/movement/persisters/new_version_test.exs +++ b/test/movement/persisters/new_version_test.exs @@ -15,11 +15,9 @@ defmodule AccentTest.Movement.Persisters.NewVersion do alias Movement.Context alias Movement.Persisters.NewVersion, as: NewVersionPersister - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create( @@ -28,7 +26,7 @@ defmodule AccentTest.Movement.Persisters.NewVersion do ) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project, user: user]} end @@ -40,7 +38,7 @@ defmodule AccentTest.Movement.Persisters.NewVersion do document: document } do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", @@ -51,7 +49,7 @@ defmodule AccentTest.Movement.Persisters.NewVersion do conflicted: true, revision_id: revision.id, document_id: document.id - }) + ) operations = [ %Movement.Operation{ diff --git a/test/movement/persisters/project_state_change_worker_test.exs b/test/movement/persisters/project_hook_worker_test.exs similarity index 72% rename from test/movement/persisters/project_state_change_worker_test.exs rename to test/movement/persisters/project_hook_worker_test.exs index f4f1a9772..03c08f21a 100644 --- a/test/movement/persisters/project_state_change_worker_test.exs +++ b/test/movement/persisters/project_hook_worker_test.exs @@ -1,4 +1,4 @@ -defmodule Movement.Persisters.ProjectStateChangeWorkerTest do +defmodule Movement.Persisters.ProjectHookWorkerTest do use Accent.RepoCase, async: true alias Accent.Document @@ -7,17 +7,17 @@ defmodule Movement.Persisters.ProjectStateChangeWorkerTest do alias Accent.Repo alias Accent.Translation alias Accent.User - alias Movement.Persisters.ProjectStateChangeWorker, as: Worker + alias Movement.Persisters.ProjectHookWorker, as: Worker setup do - user = Repo.insert!(%User{email: "test@test.com"}) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User, email: "test@test.com") + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [revision: revision, document: document, project: project, user: user]} end @@ -26,11 +26,9 @@ defmodule Movement.Persisters.ProjectStateChangeWorkerTest do args = %{ "project_id" => project.id, "previous_project_state" => %{ - "project" => %{ - "translations_count" => 0, - "reviewed_count" => 0, - "conflicts_count" => 0 - } + "translations_count" => 0, + "reviewed_count" => 0, + "conflicts_count" => 0 } } @@ -40,24 +38,22 @@ defmodule Movement.Persisters.ProjectStateChangeWorkerTest do end test "new_conflicts", %{user: user, project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: true, corrected_text: "Test", revision_id: revision.id, document_id: document.id - }) + ) args = %{ "project_id" => project.id, "user_id" => user.id, "previous_project_state" => %{ - "project" => %{ - "translations_count" => 0, - "reviewed_count" => 0, - "conflicts_count" => 0 - } + "translations_count" => 0, + "reviewed_count" => 0, + "conflicts_count" => 0 } } @@ -79,24 +75,22 @@ defmodule Movement.Persisters.ProjectStateChangeWorkerTest do end test "complete_review", %{user: user, project: project, revision: revision, document: document} do - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: false, corrected_text: "Test", revision_id: revision.id, document_id: document.id - }) + ) args = %{ "project_id" => project.id, "user_id" => user.id, "previous_project_state" => %{ - "project" => %{ - "translations_count" => 1, - "reviewed_count" => 0, - "conflicts_count" => 1 - } + "translations_count" => 1, + "reviewed_count" => 0, + "conflicts_count" => 1 } } diff --git a/test/movement/persisters/project_sync_test.exs b/test/movement/persisters/project_sync_test.exs index b4f8b9754..d023d5a52 100644 --- a/test/movement/persisters/project_sync_test.exs +++ b/test/movement/persisters/project_sync_test.exs @@ -14,24 +14,22 @@ defmodule AccentTest.Movement.Persisters.ProjectSync do alias Movement.Context alias Movement.Persisters.ProjectSync, as: ProjectSyncPersister - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [project: project, document: document, revision: revision, user: user]} end test "persist operations", %{project: project, revision: revision, document: document, user: user} do translation = - Repo.insert!(%Translation{key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id}) + Factory.insert(Translation, key: "a", proposed_text: "A", revision_id: revision.id, document_id: document.id) operations = [ %Movement.Operation{ diff --git a/test/movement/persisters/rollback_test.exs b/test/movement/persisters/rollback_test.exs index 18f2c4d76..7e0eb77eb 100644 --- a/test/movement/persisters/rollback_test.exs +++ b/test/movement/persisters/rollback_test.exs @@ -15,34 +15,32 @@ defmodule AccentTest.Movement.Persisters.Rollback do alias Movement.Context alias Movement.Persisters.Rollback, as: RollbackPersister - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [project: project, document: document, revision: revision, user: user]} end test "persist operations", %{project: project, revision: revision, document: document, user: user} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", conflicted: false, corrected_text: "Test", revision_id: revision.id, document_id: document.id - }) + ) operation = - Repo.insert!(%Accent.Operation{ + Factory.insert(Operation, text: "B", inserted_at: DateTime.utc_now(), updated_at: DateTime.utc_now(), @@ -52,7 +50,7 @@ defmodule AccentTest.Movement.Persisters.Rollback do translation_id: translation.id, revision_id: translation.revision_id, project_id: project.id - }) + ) operations = [ %Movement.Operation{ @@ -97,26 +95,26 @@ defmodule AccentTest.Movement.Persisters.Rollback do test "rollback batch", %{revision: revision} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", corrected_text: "B", conflicted: true, revision_id: revision.id, revision: revision - }) + ) - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "new", key: "a", text: "B", translation_id: translation.id, revision_id: revision.id - }) + ) - batch_operation = Repo.insert!(%Operation{action: "sync", batch: true, revision_id: revision.id}) + batch_operation = Factory.insert(Operation, action: "sync", batch: true, revision_id: revision.id) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "update", key: "a", text: "UPDATED", @@ -124,7 +122,7 @@ defmodule AccentTest.Movement.Persisters.Rollback do translation_id: translation.id, revision_id: revision.id, batch_operation_id: batch_operation.id - }) + ) rollback_operation = %Movement.Operation{ action: "rollback", @@ -147,24 +145,24 @@ defmodule AccentTest.Movement.Persisters.Rollback do test "rollback rollback does nothing", %{revision: revision} do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", corrected_text: "B", conflicted: true, revision_id: revision.id, revision: revision - }) + ) - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "new", key: "a", text: "B", translation_id: translation.id, revision_id: revision.id - }) + ) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "update", key: "a", text: "UPDATED", @@ -172,17 +170,17 @@ defmodule AccentTest.Movement.Persisters.Rollback do translation_id: translation.id, revision_id: revision.id, rollbacked: true - }) + ) rollback_operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "rollback", key: "a", previous_translation: PreviousTranslation.from_translation(translation), translation_id: translation.id, revision_id: revision.id, rollbacked_operation_id: operation.id - }) + ) rollback_rollback_operation = %Movement.Operation{ action: "rollback", diff --git a/test/movement/up_test.exs b/test/movement/up_test.exs index 0d306884b..7ae570965 100644 --- a/test/movement/up_test.exs +++ b/test/movement/up_test.exs @@ -15,13 +15,13 @@ defmodule AccentTest.Movement.Migrator.Up do end test ":correct" do - user = Repo.insert!(%User{}) - revision = Repo.insert!(%Revision{}) + user = Factory.insert(User) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_corrected", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", corrected_text: nil, proposed_text: "proposed_text", @@ -29,7 +29,7 @@ defmodule AccentTest.Movement.Migrator.Up do conflicted: true, value_type: "string", removed: false - }) + ) Migrator.up(%Operation{ id: Ecto.UUID.generate(), @@ -51,13 +51,13 @@ defmodule AccentTest.Movement.Migrator.Up do end test ":merge_on_proposed_force" do - user = Repo.insert!(%User{}) - revision = Repo.insert!(%Revision{}) + user = Factory.insert(User) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_merged", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", corrected_text: "corrected_text", proposed_text: "proposed_text", @@ -65,7 +65,7 @@ defmodule AccentTest.Movement.Migrator.Up do conflicted_text: nil, conflicted: true, removed: false - }) + ) Migrator.up(%Operation{ id: Ecto.UUID.generate(), @@ -86,19 +86,19 @@ defmodule AccentTest.Movement.Migrator.Up do end test ":merge_on_corrected" do - user = Repo.insert!(%User{}) - revision = Repo.insert!(%Revision{}) + user = Factory.insert(User) + revision = Factory.insert(Revision) translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_merged", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted_text: nil, value_type: "string", conflicted: true - }) + ) Migrator.up(%Operation{ action: "merge_on_corrected", @@ -118,9 +118,9 @@ defmodule AccentTest.Movement.Migrator.Up do test ":uncorrect" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_uncorrected", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", file_index: 1, corrected_text: "new proposed text", @@ -129,7 +129,7 @@ defmodule AccentTest.Movement.Migrator.Up do conflicted: false, removed: false, value_type: "string" - }) + ) Migrator.up(%Operation{ action: "uncorrect_conflict", @@ -149,9 +149,9 @@ defmodule AccentTest.Movement.Migrator.Up do test ":uncorrect with same corrected and proposed" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_uncorrected", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", file_index: 1, corrected_text: "proposed_text", @@ -161,7 +161,7 @@ defmodule AccentTest.Movement.Migrator.Up do conflicted: false, translated: true, removed: false - }) + ) Migrator.up(%Operation{ action: "uncorrect_conflict", @@ -181,16 +181,16 @@ defmodule AccentTest.Movement.Migrator.Up do test ":conflict_on_corrected" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_in_conflict", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", file_index: 1, corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted: false, removed: false - }) + ) Migrator.up(%Operation{ action: "conflict_on_corrected", @@ -210,16 +210,16 @@ defmodule AccentTest.Movement.Migrator.Up do test ":conflict_on_slave" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_in_conflict", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", file_index: 1, corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted: false, removed: false - }) + ) Migrator.up(%Operation{ action: "conflict_on_slave", @@ -238,14 +238,14 @@ defmodule AccentTest.Movement.Migrator.Up do test ":remove" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_removed", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted: false, removed: false - }) + ) Migrator.up(%Operation{ action: "remove", @@ -260,22 +260,23 @@ defmodule AccentTest.Movement.Migrator.Up do test ":renew" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_renewed", - revision: Repo.insert!(%Revision{}), + revision_id: Factory.insert(Revision).id, corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted: false, removed: true - }) + ) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "renew", - translation: translation, + translation_id: translation.id, previous_translation: PreviousTranslation.from_translation(translation) - }) + ) + operation = Repo.preload(operation, [:translation]) Migrator.up(operation) updated_translation = Repo.get(Translation, translation.id) @@ -285,21 +286,23 @@ defmodule AccentTest.Movement.Migrator.Up do test ":rollback" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "to_be_rollbacked", - revision: Repo.insert!(%Revision{}), + revision_id: Factory.insert(Revision).id, corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted: false, removed: true - }) + ) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "rollback", - translation: translation, + translation_id: translation.id, previous_translation: PreviousTranslation.from_translation(%{translation | corrected_text: "previous"}) - }) + ) + + operation = Repo.preload(operation, [:translation]) Migrator.up(operation) @@ -310,10 +313,10 @@ defmodule AccentTest.Movement.Migrator.Up do test ":conflict_on_proposed" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, value_type: "", key: "to_be_conflict_on_proposed", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), file_comment: "", file_index: 1, corrected_text: "corrected_text", @@ -321,7 +324,7 @@ defmodule AccentTest.Movement.Migrator.Up do conflicted: true, removed: false, placeholders: [] - }) + ) Migrator.up(%Operation{ value_type: "", @@ -345,15 +348,15 @@ defmodule AccentTest.Movement.Migrator.Up do test ":updated_proposed" do translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "updated_proposed", - revision: Repo.insert!(%Revision{}), + revision: Factory.insert(Revision), corrected_text: "corrected_text", proposed_text: "proposed_text", conflicted_text: "conflict", conflicted: true, removed: false - }) + ) Migrator.up(%Operation{ action: "update_proposed", diff --git a/test/plugs/assign_current_user_test.exs b/test/plugs/assign_current_user_test.exs index 055964bac..88253d4ef 100644 --- a/test/plugs/assign_current_user_test.exs +++ b/test/plugs/assign_current_user_test.exs @@ -5,12 +5,8 @@ defmodule AccentTest.Plugs.AssignCurrentUser do alias Accent.AccessToken alias Accent.Plugs.AssignCurrentUser - alias Accent.Repo alias Accent.User - @user %User{email: "test@test.com"} - @token %AccessToken{revoked_at: nil, token: "1234"} - defp call_plug(token) do :get |> conn("/foo") @@ -20,11 +16,11 @@ defmodule AccentTest.Plugs.AssignCurrentUser do test "assign current_user" do user = - @user - |> Repo.insert!() + User + |> Factory.insert() |> Map.put(:permissions, %{}) - token = Repo.insert!(Map.put(@token, :user_id, user.id)) + token = Factory.insert(AccessToken, user_id: user.id) assigned_user = token.token diff --git a/test/plugs/movement_context_parser_test.exs b/test/plugs/movement_context_parser_test.exs index 0e44eacd9..b22efb3a5 100644 --- a/test/plugs/movement_context_parser_test.exs +++ b/test/plugs/movement_context_parser_test.exs @@ -30,16 +30,15 @@ defmodule AccentTest.Plugs.MovementContextParser do %Plug.Upload{content_type: "text/plain", filename: "unicode.strings", path: "test/support/invalid_unicode.strings"} end - @user %User{email: "test@test.com"} setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) revision = project |> Repo.preload(:revisions) |> Map.get(:revisions) |> hd() - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "json") {:ok, [project: project, document: document, revision: revision, language: language, user: user]} end diff --git a/test/scopes/translation_test.exs b/test/scopes/translation_test.exs index b846b6837..36b29d6cf 100644 --- a/test/scopes/translation_test.exs +++ b/test/scopes/translation_test.exs @@ -5,21 +5,20 @@ defmodule AccentTest.Scopes.Translation do alias Accent.Document alias Accent.Operation alias Accent.Project - alias Accent.Repo alias Accent.Scopes.Translation, as: Scope alias Accent.Translation doctest Accent.Scopes.Translation setup do - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + project = Factory.insert(Project) {:ok, [project: project]} end describe "parse_added_last_sync/3" do test "existing sync", %{project: project} do - sync = Repo.insert!(%Operation{project: project, action: "sync"}) + sync = Factory.insert(Operation, project_id: project.id, action: "sync") query = Scope.parse_added_last_sync(Translation, true, project.id, nil) @@ -28,9 +27,9 @@ defmodule AccentTest.Scopes.Translation do end test "existing sync with document", %{project: project} do - document = Repo.insert!(%Document{project_id: project.id, path: "my-test", format: "xliff_1_2"}) - sync = Repo.insert!(%Operation{project: project, action: "sync", document: document}) - _other_sync = Repo.insert!(%Operation{project: project, action: "sync"}) + document = Factory.insert(Document, project_id: project.id, path: "my-test", format: "xliff_1_2") + sync = Factory.insert(Operation, project_id: project.id, action: "sync", document_id: document.id) + _other_sync = Factory.insert(Operation, project_id: project.id, action: "sync") query = Scope.parse_added_last_sync(Translation, true, project.id, document.id) @@ -39,9 +38,14 @@ defmodule AccentTest.Scopes.Translation do end test "many sync", %{project: project} do - _ = Repo.insert!(%Operation{project: project, action: "sync", inserted_at: ~U[2018-01-02T00:00:00.000000Z]}) - sync = Repo.insert!(%Operation{project: project, action: "sync", inserted_at: ~U[2018-01-03T00:00:00.000000Z]}) - _ = Repo.insert!(%Operation{project: project, action: "sync", inserted_at: ~U[2018-01-01T00:00:00.000000Z]}) + _ = + Factory.insert(Operation, project_id: project.id, action: "sync", inserted_at: ~U[2018-01-02T00:00:00.000000Z]) + + sync = + Factory.insert(Operation, project_id: project.id, action: "sync", inserted_at: ~U[2018-01-03T00:00:00.000000Z]) + + _ = + Factory.insert(Operation, project_id: project.id, action: "sync", inserted_at: ~U[2018-01-01T00:00:00.000000Z]) query = Scope.parse_added_last_sync(Translation, true, project.id, nil) diff --git a/test/scopes/version_test.exs b/test/scopes/version_test.exs index afc0ce451..7b8044034 100644 --- a/test/scopes/version_test.exs +++ b/test/scopes/version_test.exs @@ -16,12 +16,12 @@ defmodule AccentTest.Scopes.Version do end defp insert_version(tag, project, user) do - Repo.insert!(%Version{name: "foo", tag: tag, project_id: project.id, user_id: user.id}) + Factory.insert(Version, name: "foo", tag: tag, project_id: project.id, user_id: user.id) end setup do - user = Repo.insert!(%User{email: "test@test.com"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User, email: "test@test.com") + project = Factory.insert(Project) {:ok, [user: user, project: project]} end diff --git a/test/services/collaborator_creator_test.exs b/test/services/collaborator_creator_test.exs index 2121782f0..f7528d3ab 100644 --- a/test/services/collaborator_creator_test.exs +++ b/test/services/collaborator_creator_test.exs @@ -5,58 +5,56 @@ defmodule AccentTest.CollaboratorCreator do alias Accent.Collaborator alias Accent.CollaboratorCreator alias Accent.Project - alias Accent.Repo alias Accent.User + @email Faker.Internet.email() + test "create unknown email" do - email = "test@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "admin" {:ok, collaborator} = CollaboratorCreator.create(%{ - "email" => email, + "email" => @email, "assigner_id" => assigner.id, "role" => role, "project_id" => project.id }) - assert collaborator.email === email + assert collaborator.email === @email assert collaborator.assigner_id === assigner.id assert collaborator.role === role end test "create known email" do - email = "test@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - user = Repo.insert!(%User{email: email}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + user = Factory.insert(User, email: @email) + assigner = Factory.insert(User) role = "admin" {:ok, collaborator} = CollaboratorCreator.create(%{ - "email" => email, + "email" => @email, "assigner_id" => assigner.id, "role" => role, "project_id" => project.id }) - assert collaborator.email === email + assert collaborator.email === @email assert collaborator.user_id === user.id assert collaborator.assigner_id === assigner.id assert collaborator.role === role end test "create invalid role" do - email = "test@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "test123" {:error, collaborator} = CollaboratorCreator.create(%{ - "email" => email, + "email" => @email, "assigner_id" => assigner.id, "role" => role, "project_id" => project.id @@ -70,8 +68,8 @@ defmodule AccentTest.CollaboratorCreator do test "create with insensitive email" do email = "TEST@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "admin" {:ok, collaborator} = @@ -86,32 +84,30 @@ defmodule AccentTest.CollaboratorCreator do end test "create with leading and trailing spaces in email" do - email = " test@test.com " - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "admin" {:ok, collaborator} = CollaboratorCreator.create(%{ - "email" => email, + "email" => " #{@email} ", "assigner_id" => assigner.id, "role" => role, "project_id" => project.id }) - assert collaborator.email === "test@test.com" + assert collaborator.email === @email end test "cannot create with already used email for project" do - email = "test@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "admin" - Repo.insert!(%Collaborator{email: email, assigner_id: assigner.id, role: role, project_id: project.id}) + Factory.insert(Collaborator, email: @email, assigner_id: assigner.id, role: role, project_id: project.id) {:error, changeset} = CollaboratorCreator.create(%{ - "email" => email, + "email" => @email, "assigner_id" => assigner.id, "role" => role, "project_id" => project.id diff --git a/test/services/collaborator_normalizer_test.exs b/test/services/collaborator_normalizer_test.exs index 34a82707e..ad4bcba26 100644 --- a/test/services/collaborator_normalizer_test.exs +++ b/test/services/collaborator_normalizer_test.exs @@ -9,15 +9,32 @@ defmodule AccentTest.CollaboratorNormalizer do alias Accent.Repo alias Accent.User alias Accent.UserRemote.CollaboratorNormalizer + alias Faker.Internet test "create with many collaborations" do - project = Repo.insert!(%Project{main_color: "#f00", name: "Ha"}) - project2 = Repo.insert!(%Project{main_color: "#f00", name: "Oh"}) - assigner = Repo.insert!(%User{email: "assigner@test.com"}) + project = Factory.insert(Project) + project2 = Factory.insert(Project) + assigner = Factory.insert(User) collaborators = [ - %Collaborator{role: "admin", project_id: project.id, assigner_id: assigner.id}, - %Collaborator{role: "developer", project_id: project2.id, assigner_id: assigner.id} + struct!( + Collaborator, + Factory.build(Collaborator, + email: Internet.email(), + role: "admin", + project_id: project.id, + assigner_id: assigner.id + ) + ), + struct!( + Collaborator, + Factory.build(Collaborator, + email: Internet.email(), + role: "developer", + project_id: project2.id, + assigner_id: assigner.id + ) + ) ] collaborator_ids = @@ -26,7 +43,7 @@ defmodule AccentTest.CollaboratorNormalizer do |> Enum.map(&Repo.insert!/1) |> Enum.map(&Map.get(&1, :id)) - new_user = Repo.insert!(%User{email: "test@test.com"}) + new_user = Factory.insert(User, email: "test@test.com") %User{} = CollaboratorNormalizer.normalize(new_user) @@ -36,13 +53,29 @@ defmodule AccentTest.CollaboratorNormalizer do end test "create with case insensitive email" do - project = Repo.insert!(%Project{main_color: "#f00", name: "Ha"}) - project2 = Repo.insert!(%Project{main_color: "#f00", name: "Oh"}) - assigner = Repo.insert!(%User{email: "assigner@test.com"}) + project = Factory.insert(Project) + project2 = Factory.insert(Project) + assigner = Factory.insert(User) collaborators = [ - %Collaborator{role: "admin", project_id: project.id, assigner_id: assigner.id}, - %Collaborator{role: "developer", project_id: project2.id, assigner_id: assigner.id} + struct!( + Collaborator, + Factory.build(Collaborator, + email: Internet.email(), + role: "admin", + project_id: project.id, + assigner_id: assigner.id + ) + ), + struct!( + Collaborator, + Factory.build(Collaborator, + email: Internet.email(), + role: "developer", + project_id: project2.id, + assigner_id: assigner.id + ) + ) ] collaborator_ids = @@ -51,7 +84,7 @@ defmodule AccentTest.CollaboratorNormalizer do |> Enum.map(&Repo.insert!/1) |> Enum.map(&Map.get(&1, :id)) - new_user = Repo.insert!(%User{email: "Test@test.com"}) + new_user = Factory.insert(User, email: "Test@test.com") %User{} = CollaboratorNormalizer.normalize(new_user) @@ -61,7 +94,7 @@ defmodule AccentTest.CollaboratorNormalizer do end test "create without collaborations" do - new_user = Repo.insert!(%User{email: "Test@test.com"}) + new_user = Factory.insert(User, email: "Test@test.com") %User{} = CollaboratorNormalizer.normalize(new_user) diff --git a/test/services/collaborator_updater_test.exs b/test/services/collaborator_updater_test.exs index 7bfa90d2e..bc6efde0b 100644 --- a/test/services/collaborator_updater_test.exs +++ b/test/services/collaborator_updater_test.exs @@ -5,20 +5,20 @@ defmodule AccentTest.CollaboratorUpdater do alias Accent.Collaborator alias Accent.CollaboratorUpdater alias Accent.Project - alias Accent.Repo alias Accent.User test "update" do - email = "test@test.com" - project = Repo.insert!(%Project{main_color: "#f00", name: "com"}) - assigner = Repo.insert!(%User{email: "lol@test.com"}) + project = Factory.insert(Project) + assigner = Factory.insert(User) role = "admin" - collaborator = Repo.insert!(%Collaborator{role: role, assigner: assigner, project: project, email: email}) + + collaborator = + Factory.insert(Collaborator, role: role, assigner_id: assigner.id, project_id: project.id, email: assigner.email) {:ok, updated_collaborator} = CollaboratorUpdater.update(collaborator, %{"role" => "reviewer"}) assert updated_collaborator.email === collaborator.email - assert updated_collaborator.assigner_id === collaborator.assigner.id + assert updated_collaborator.assigner_id === collaborator.assigner_id assert updated_collaborator.role === "reviewer" end end diff --git a/test/services/integration_manager_test.exs b/test/services/integration_manager_test.exs index 2f3f18b69..806540e9d 100644 --- a/test/services/integration_manager_test.exs +++ b/test/services/integration_manager_test.exs @@ -1,6 +1,6 @@ defmodule AccentTest.IntegrationManager do @moduledoc false - use Accent.RepoCase, async: true + use Accent.RepoCase, async: false import Mock @@ -17,11 +17,19 @@ defmodule AccentTest.IntegrationManager do describe "execute" do setup do - project = Repo.insert!(%Project{main_color: "red", name: "com"}) - user = Repo.insert!(%User{email: "test@test.com"}) - language = Repo.insert!(%Language{slug: "fr-custom", name: "Fr"}) - revision = Repo.insert!(%Revision{project: project, language: language}) - document = Repo.insert!(%Document{project: project, path: "foo", format: "gettext"}) + project = Factory.insert(Project) + user = Factory.insert(User) + language = Factory.insert(Language, slug: "fr-custom") + + revision = + Factory.insert(Revision, + master: true, + master_revision_id: nil, + project_id: project.id, + language_id: language.id + ) + + document = Factory.insert(Document, project_id: project.id, path: "foo", format: "gettext") {:ok, [project: project, user: user, language: language, revision: revision, document: document]} end @@ -32,30 +40,30 @@ defmodule AccentTest.IntegrationManager do document: document, project: project } do - version = Repo.insert!(%Version{project: project, tag: "1.2.45", name: "vNext", user: user}) + version = Factory.insert(Version, project_id: project.id, tag: "1.2.45", name: "vNext", user_id: user.id) - Repo.insert!(%Translation{ - revision: revision, - document: document, + Factory.insert(Translation, + revision_id: revision.id, + document_id: document.id, key: "key", corrected_text: "value latest" - }) + ) - Repo.insert!(%Translation{ - revision: revision, - version: version, - document: document, + Factory.insert(Translation, + revision_id: revision.id, + version_id: version.id, + document_id: document.id, key: "key", corrected_text: "value v1.2.45" - }) + ) integration = - Repo.insert!(%Integration{ - project: project, - user: user, + Factory.insert(Integration, + project_id: project.id, + user_id: user.id, service: "azure_storage_container", data: %{azure_storage_container_sas: "http://azure.blob.test/container?sas=1234"} - }) + ) with_mock HTTPoison, put: fn url, {:file, file}, headers -> @@ -87,15 +95,20 @@ defmodule AccentTest.IntegrationManager do document: document, project: project } do - Repo.insert!(%Translation{revision: revision, document: document, key: "key", corrected_text: "value"}) + Factory.insert(Translation, + revision_id: revision.id, + document_id: document.id, + key: "key", + corrected_text: "value" + ) integration = - Repo.insert!(%Integration{ - project: project, - user: user, + Factory.insert(Integration, + project_id: project.id, + user_id: user.id, service: "azure_storage_container", data: %{azure_storage_container_sas: "http://azure.blob.test/container?sas=1234"} - }) + ) with_mock HTTPoison, put: fn url, body, headers -> diff --git a/test/services/operation_batcher_test.exs b/test/services/operation_batcher_test.exs index ab2d4ad9b..4e16527fe 100644 --- a/test/services/operation_batcher_test.exs +++ b/test/services/operation_batcher_test.exs @@ -12,44 +12,43 @@ defmodule AccentTest.OperationBatcher do alias Accent.User setup do - user = Repo.insert!(%User{}) - revision = Repo.insert!(%Revision{}) - - translation_one = - Repo.insert!(%Translation{key: "a", conflicted: true, revision_id: revision.id, revision: revision}) - - translation_two = - Repo.insert!(%Translation{key: "b", conflicted: true, revision_id: revision.id, revision: revision}) + user = Factory.insert(User) + revision = Factory.insert(Revision) + translation_one = Factory.insert(Translation, key: "a", conflicted: true, revision_id: revision.id) + translation_two = Factory.insert(Translation, key: "b", conflicted: true, revision_id: revision.id) [user: user, revision: revision, translations: [translation_one, translation_two]] end test "create batch with close operations", %{user: user, revision: revision, translations: [translation_one, _]} do operations = - [ - %Operation{ - action: "correct_conflict", - key: "a", - text: "B", - translation_id: translation_one.id, - user_id: user.id, - revision_id: revision.id, - inserted_at: DateTime.utc_now() - } - ] - |> Enum.map(&Repo.insert!/1) - |> Enum.map(&Map.get(&1, :id)) + Enum.map( + [ + Factory.insert(Operation, + action: "correct_conflict", + key: "a", + text: "B", + translation_id: translation_one.id, + user_id: user.id, + revision_id: revision.id, + project_id: revision.project_id, + inserted_at: DateTime.utc_now() + ) + ], + &Map.get(&1, :id) + ) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "correct_conflict", key: "a", - text: "B", + text: "C", translation_id: translation_one.id, user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, inserted_at: DateTime.utc_now() - }) + ) batch_responses = OperationBatcher.batch(operation) @@ -74,58 +73,63 @@ defmodule AccentTest.OperationBatcher do translations: [translation_one, translation_two] } do batch_operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "batch_correct_conflict", user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, stats: [%{"count" => 2, "action" => "correct_conflict"}], inserted_at: DateTime.utc_now() |> DateTime.to_naive() |> NaiveDateTime.add(-960, :second) |> DateTime.from_naive!("Etc/UTC") - }) + ) operations = - [ - %Operation{ - action: "correct_conflict", - key: "a", - text: "B", - translation_id: translation_one.id, - user_id: user.id, - revision_id: revision.id, - batch_operation_id: batch_operation.id, - inserted_at: - DateTime.utc_now() - |> DateTime.to_naive() - |> NaiveDateTime.add(-960, :second) - |> DateTime.from_naive!("Etc/UTC") - }, - %Operation{ - action: "correct_conflict", - key: "b", - text: "C", - translation_id: translation_two.id, - user_id: user.id, - revision_id: revision.id, - batch_operation_id: batch_operation.id, - inserted_at: DateTime.utc_now() - } - ] - |> Enum.map(&Repo.insert!/1) - |> Enum.map(&Map.get(&1, :id)) + Enum.map( + [ + Factory.insert(Operation, + action: "correct_conflict", + key: "a", + text: "B", + translation_id: translation_one.id, + user_id: user.id, + revision_id: revision.id, + project_id: revision.project_id, + batch_operation_id: batch_operation.id, + inserted_at: + DateTime.utc_now() + |> DateTime.to_naive() + |> NaiveDateTime.add(-960, :second) + |> DateTime.from_naive!("Etc/UTC") + ), + Factory.insert(Operation, + action: "correct_conflict", + key: "b", + text: "C", + translation_id: translation_two.id, + user_id: user.id, + revision_id: revision.id, + project_id: revision.project_id, + batch_operation_id: batch_operation.id, + inserted_at: DateTime.utc_now() + ) + ], + &Map.get(&1, :id) + ) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "correct_conflict", key: "a", text: "B", translation_id: translation_one.id, user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, inserted_at: DateTime.utc_now() - }) + ) batch_responses = Accent.OperationBatcher.batch(operation) @@ -158,6 +162,7 @@ defmodule AccentTest.OperationBatcher do translation_id: translation_one.id, user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, inserted_at: DateTime.utc_now() |> DateTime.to_naive() @@ -171,6 +176,7 @@ defmodule AccentTest.OperationBatcher do translation_id: translation_two.id, user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, inserted_at: DateTime.utc_now() |> DateTime.to_naive() @@ -182,15 +188,16 @@ defmodule AccentTest.OperationBatcher do |> Enum.map(&Map.get(&1, :id)) operation = - Repo.insert!(%Operation{ + Factory.insert(Operation, action: "correct_conflict", key: "a", text: "B", translation_id: translation_one.id, user_id: user.id, revision_id: revision.id, + project_id: revision.project_id, inserted_at: DateTime.utc_now() - }) + ) batch_responses = Accent.OperationBatcher.batch(operation) diff --git a/test/services/project_creator_test.exs b/test/services/project_creator_test.exs index 7f15e29f5..5c7a55a6a 100644 --- a/test/services/project_creator_test.exs +++ b/test/services/project_creator_test.exs @@ -10,8 +10,8 @@ defmodule AccentTest.ProjectCreator do require Ecto.Query test "create with language and user" do - language = Repo.insert!(%Language{name: "french"}) - user = Repo.insert!(%User{email: "lol@test.com"}) + language = Factory.insert(Language) + user = Factory.insert(User) params = %{"main_color" => "#f00", "name" => "OK", "language_id" => language.id} {:ok, project} = ProjectCreator.create(params: params, user: user) @@ -23,8 +23,8 @@ defmodule AccentTest.ProjectCreator do end test "create owner collaborator" do - language = Repo.insert!(%Language{name: "french"}) - user = Repo.insert!(%User{email: "lol@test.com"}) + language = Factory.insert(Language) + user = Factory.insert(User) params = %{"main_color" => "#f00", "name" => "OK", "language_id" => language.id} {:ok, project} = ProjectCreator.create(params: params, user: user) @@ -34,8 +34,8 @@ defmodule AccentTest.ProjectCreator do end test "create bot collaborator" do - language = Repo.insert!(%Language{name: "french"}) - user = Repo.insert!(%User{email: "lol@test.com"}) + language = Factory.insert(Language) + user = Factory.insert(User) params = %{"main_color" => "#f00", "name" => "OK", "language_id" => language.id} {:ok, project} = ProjectCreator.create(params: params, user: user) diff --git a/test/services/project_deleter_test.exs b/test/services/project_deleter_test.exs index 0a6528fe1..50ae1f828 100644 --- a/test/services/project_deleter_test.exs +++ b/test/services/project_deleter_test.exs @@ -3,13 +3,21 @@ defmodule AccentTest.ProjectDeleter do use Accent.RepoCase, async: true alias Accent.Collaborator + alias Accent.Language + alias Accent.Operation alias Accent.Project alias Accent.ProjectDeleter alias Accent.Repo + alias Accent.Revision - test "create with language and user" do - project = Repo.insert!(%Project{main_color: "#f00", name: "french"}) - collaborator = Repo.insert!(%Collaborator{project_id: project.id, role: "reviewer"}) + test "delete collaborators and operations" do + project = Factory.insert(Project) + language = Factory.insert(Language) + revision = Factory.insert(Revision, language_id: language.id, project_id: project.id) + collaborator = Factory.insert(Collaborator, project_id: project.id, role: "reviewer") + + Factory.insert(Operation, project_id: project.id, action: "sync") + Factory.insert(Operation, project_id: project.id, revision_id: revision.id, action: "merge") assert project |> Ecto.assoc(:all_collaborators) @@ -18,6 +26,8 @@ defmodule AccentTest.ProjectDeleter do {:ok, project} = ProjectDeleter.delete(project: project) - assert Repo.all(Ecto.assoc(project, :all_collaborators)) === [] + assert Repo.aggregate(Ecto.assoc(project, :all_collaborators), :count) === 0 + assert Repo.aggregate(Ecto.assoc(project, :operations), :count) === 0 + assert Repo.aggregate(Ecto.assoc(revision, :operations), :count) === 0 end end diff --git a/test/services/revision_deleter_test.exs b/test/services/revision_deleter_test.exs index c808c5d25..901b3ce9d 100644 --- a/test/services/revision_deleter_test.exs +++ b/test/services/revision_deleter_test.exs @@ -11,19 +11,19 @@ defmodule AccentTest.RevisionDeleter do alias Accent.Translation setup do - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) - french_language = Repo.insert!(%Language{name: "french"}) - english_language = Repo.insert!(%Language{name: "english"}) + project = Factory.insert(Project) + french_language = Factory.insert(Language) + english_language = Factory.insert(Language) - master_revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + master_revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) slave_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: master_revision.id - }) + ) {:ok, [master_revision: master_revision, slave_revision: slave_revision]} end @@ -41,7 +41,7 @@ defmodule AccentTest.RevisionDeleter do end test "delete operations", %{slave_revision: revision} do - operation = Repo.insert!(%Operation{action: "new", key: "a", revision_id: revision.id}) + operation = Factory.insert(Operation, action: "new", key: "a", revision_id: revision.id) Accent.Revisions.DeleteWorker.perform(%Oban.Job{args: %{"revision_id" => revision.id}}) @@ -49,7 +49,7 @@ defmodule AccentTest.RevisionDeleter do end test "delete translations", %{slave_revision: revision} do - translation = Repo.insert!(%Translation{key: "a", revision_id: revision.id}) + translation = Factory.insert(Translation, key: "a", revision_id: revision.id) Accent.Revisions.DeleteWorker.perform(%Oban.Job{args: %{"revision_id" => revision.id}}) diff --git a/test/services/revision_master_promoter_test.exs b/test/services/revision_master_promoter_test.exs index edbaac0c6..1473788ef 100644 --- a/test/services/revision_master_promoter_test.exs +++ b/test/services/revision_master_promoter_test.exs @@ -9,19 +9,19 @@ defmodule AccentTest.RevisionMasterPromoter do alias Accent.RevisionManager setup do - french_language = Repo.insert!(%Language{name: "french"}) - english_language = Repo.insert!(%Language{name: "english"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + french_language = Factory.insert(Language) + english_language = Factory.insert(Language) + project = Factory.insert(Project) - master_revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + master_revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) slave_revision = - Repo.insert!(%Revision{ + Factory.insert(Revision, language_id: english_language.id, project_id: project.id, master: false, master_revision_id: master_revision.id - }) + ) {:ok, [master_revision: master_revision, slave_revision: slave_revision]} end diff --git a/test/services/translations_renderer_test.exs b/test/services/translations_renderer_test.exs index cc977e729..e2c87e89c 100644 --- a/test/services/translations_renderer_test.exs +++ b/test/services/translations_renderer_test.exs @@ -10,11 +10,9 @@ defmodule AccentTest.TranslationsRenderer do alias Accent.TranslationsRenderer alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - language = Repo.insert!(%Language{name: "English", slug: Ecto.UUID.generate()}) + user = Factory.insert(User) + language = Factory.insert(Language) {:ok, project} = ProjectCreator.create(params: %{main_color: "#f00", name: "My project", language_id: language.id}, user: user) @@ -30,16 +28,16 @@ defmodule AccentTest.TranslationsRenderer do end test "render json with filename", %{project: project, revision: revision} do - document = Repo.insert!(%Document{project_id: project.id, path: "my-test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "my-test", format: "json") translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "B", corrected_text: "A", revision_id: revision.id, document_id: document.id - }) + ) %{render: render} = TranslationsRenderer.render_translations(%{ @@ -60,28 +58,25 @@ defmodule AccentTest.TranslationsRenderer do end test "render json with runtime error", %{project: project, revision: revision} do - document = Repo.insert!(%Document{project_id: project.id, path: "my-test", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "my-test", format: "json") translations = - Enum.map( - [ - %Translation{ - key: "a.nested.foo", - proposed_text: "B", - corrected_text: "A", - revision_id: revision.id, - document_id: document.id - }, - %Translation{ - key: "a.nested", - proposed_text: "C", - corrected_text: "D", - revision_id: revision.id, - document_id: document.id - } - ], - &Repo.insert!/1 - ) + [ + Factory.insert(Translation, + key: "a.nested.foo", + proposed_text: "B", + corrected_text: "A", + revision_id: revision.id, + document_id: document.id + ), + Factory.insert(Translation, + key: "a.nested", + proposed_text: "C", + corrected_text: "D", + revision_id: revision.id, + document_id: document.id + ) + ] %{render: render} = TranslationsRenderer.render_translations(%{ @@ -97,16 +92,16 @@ defmodule AccentTest.TranslationsRenderer do if Langue.Formatter.Rails.enabled?() do test "render rails with locale", %{project: project, revision: revision} do - document = Repo.insert!(%Document{project_id: project.id, path: "my-test", format: "rails_yml"}) + document = Factory.insert(Document, project_id: project.id, path: "my-test", format: "rails_yml") translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", revision_id: revision.id, document_id: document.id - }) + ) %{render: render} = TranslationsRenderer.render_translations(%{ @@ -128,16 +123,16 @@ defmodule AccentTest.TranslationsRenderer do test "render xliff and revision overrides on source revision", %{project: project, revision: revision} do revision = Repo.update!(Ecto.Changeset.change(revision, %{slug: "testtest"})) - document = Repo.insert!(%Document{project_id: project.id, path: "my-test", format: "xliff_1_2"}) + document = Factory.insert(Document, project_id: project.id, path: "my-test", format: "xliff_1_2") translation = - Repo.insert!(%Translation{ + Factory.insert(Translation, key: "a", proposed_text: "A", corrected_text: "A", revision_id: revision.id, document_id: document.id - }) + ) %{render: render} = TranslationsRenderer.render_translations(%{ diff --git a/test/support/channel_case.ex b/test/support/channel_case.ex index 89e330d08..bfbb4444c 100644 --- a/test/support/channel_case.ex +++ b/test/support/channel_case.ex @@ -12,6 +12,8 @@ defmodule Accent.ChannelCase do # Import conveniences for testing with connections import Phoenix.ChannelTest + alias Accent.Factory + # The default endpoint for testing @endpoint Endpoint diff --git a/test/support/conn_case.ex b/test/support/conn_case.ex index ed4c54ac7..4661b37bc 100644 --- a/test/support/conn_case.ex +++ b/test/support/conn_case.ex @@ -16,8 +16,6 @@ defmodule Accent.ConnCase do use ExUnit.CaseTemplate alias Accent.Endpoint - alias Accent.Repo - alias Ecto.Adapters.SQL.Sandbox alias Phoenix.ConnTest using do @@ -29,18 +27,21 @@ defmodule Accent.ConnCase do import Phoenix.ConnTest import Plug.Conn + alias Accent.Factory + # The default endpoint for testing @endpoint Endpoint end end setup tags do - :ok = Sandbox.checkout(Repo) - - unless tags[:async] do - Sandbox.mode(Repo, {:shared, self()}) - end + setup_sandbox(tags) {:ok, conn: ConnTest.build_conn()} end + + def setup_sandbox(tags) do + pid = Ecto.Adapters.SQL.Sandbox.start_owner!(Accent.Repo, shared: not tags[:async]) + on_exit(fn -> Ecto.Adapters.SQL.Sandbox.stop_owner(pid) end) + end end diff --git a/test/support/factory.ex b/test/support/factory.ex new file mode 100644 index 000000000..d2cfe3d63 --- /dev/null +++ b/test/support/factory.ex @@ -0,0 +1,48 @@ +defmodule Accent.Factory do + @moduledoc false + use Factori, + repo: Accent.Repo, + mappings: [ + Accent.Factory.Mappings.Document, + Accent.Factory.Mappings.Operation, + Accent.Factory.Mappings.Translation, + Accent.Factory.Mappings.Project, + Factori.Mapping.Embed, + Factori.Mapping.Enum, + Factori.Mapping.Faker + ] + + defmodule Mappings do + @moduledoc false + defmodule Project do + @moduledoc false + @behaviour Factori.Mapping + + def match(%{table_name: "projects", name: :locked_file_operations}), do: false + end + + defmodule Document do + @moduledoc false + @behaviour Factori.Mapping + + def match(%{table_name: "documents", name: :format}), do: "json" + def match(%{table_name: "documents", name: :path}), do: "translations" + end + + defmodule Operation do + @moduledoc false + @behaviour Factori.Mapping + + def match(%{table_name: "operations", name: :rollbacked}), do: false + def match(%{table_name: "operations", name: :batch}), do: false + end + + defmodule Translation do + @moduledoc false + @behaviour Factori.Mapping + + def match(%{table_name: "translations", name: :locked}), do: false + def match(%{table_name: "translations", name: :removed}), do: false + end + end +end diff --git a/test/support/mocks.ex b/test/support/mocks.ex index e70aba497..516b67afa 100644 --- a/test/support/mocks.ex +++ b/test/support/mocks.ex @@ -1,7 +1,14 @@ defmodule Accent.Hook.Outbounds.Mock do @moduledoc false + @behaviour Accent.Hook.Events + use Oban.Worker, queue: :hook + @impl Accent.Hook.Events + def registered_events do + :all + end + @impl Oban.Worker def perform(_job) do :ok diff --git a/test/support/repo_case.ex b/test/support/repo_case.ex index 445e6fc60..f955d58d2 100644 --- a/test/support/repo_case.ex +++ b/test/support/repo_case.ex @@ -6,6 +6,8 @@ defmodule Accent.RepoCase do quote do use Oban.Testing, repo: Accent.Repo + alias Accent.Factory + def to_worker_args(struct) do struct |> Jason.encode!() @@ -15,12 +17,13 @@ defmodule Accent.RepoCase do end setup tags do - :ok = Ecto.Adapters.SQL.Sandbox.checkout(Accent.Repo) - - unless tags[:async] do - Ecto.Adapters.SQL.Sandbox.mode(Accent.Repo, {:shared, self()}) - end + setup_sandbox(tags) :ok end + + def setup_sandbox(tags) do + pid = Ecto.Adapters.SQL.Sandbox.start_owner!(Accent.Repo, shared: not tags[:async]) + on_exit(fn -> Ecto.Adapters.SQL.Sandbox.stop_owner(pid) end) + end end diff --git a/test/test_helper.exs b/test/test_helper.exs index aec5171a1..12ed268a4 100644 --- a/test/test_helper.exs +++ b/test/test_helper.exs @@ -1,3 +1,5 @@ +alias Ecto.Adapters.SQL.Sandbox + defmodule Accent.FormatterTestHelper do @moduledoc false def test_parse(variant, parser) do @@ -50,4 +52,9 @@ defmodule Langue.Expectation.Case do end ExUnit.start() -Ecto.Adapters.SQL.Sandbox.mode(Accent.Repo, :manual) + +Sandbox.checkout(Accent.Repo) +Accent.Factory.bootstrap() +Sandbox.checkin(Accent.Repo) + +Sandbox.mode(Accent.Repo, :manual) diff --git a/test/web/channels/project_channel_test.exs b/test/web/channels/project_channel_test.exs index 84e148ca6..e4d504f10 100644 --- a/test/web/channels/project_channel_test.exs +++ b/test/web/channels/project_channel_test.exs @@ -5,15 +5,14 @@ defmodule AccentTest.ProjectChannel do alias Accent.AccessToken alias Accent.Collaborator alias Accent.Project - alias Accent.Repo alias Accent.User alias Accent.UserSocket setup do - user = Repo.insert!(%User{email: "test@test.com"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) + user = Factory.insert(User, email: "test@test.com") + project = Factory.insert(Project) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") socket = socket(UserSocket, "will-autenticated-user", %{}) {:ok, socket} = UserSocket.connect(%{"token" => "Bearer #{access_token.token}"}, socket) @@ -33,8 +32,8 @@ defmodule AccentTest.ProjectChannel do end test "join with unauthorized user", %{project: project} do - user = Repo.insert!(%User{email: "test2@test.com"}) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token-2"}) + user = Factory.insert(User, email: "test2@test.com") + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token-2") socket = socket(UserSocket, "unauthorized-user", %{}) {:ok, socket} = UserSocket.connect(%{"token" => "Bearer #{access_token.token}"}, socket) diff --git a/test/web/channels/user_socket_test.exs b/test/web/channels/user_socket_test.exs index 83d174d0d..1cb377b6f 100644 --- a/test/web/channels/user_socket_test.exs +++ b/test/web/channels/user_socket_test.exs @@ -2,12 +2,11 @@ defmodule AccentTest.UserSocket do use Accent.ChannelCase, async: true alias Accent.AccessToken - alias Accent.Repo alias Accent.User alias Accent.UserSocket setup do - user = Repo.insert!(%User{email: "test@test.com"}) + user = Factory.insert(User, email: "test@test.com") {:ok, user: user} end @@ -20,7 +19,7 @@ defmodule AccentTest.UserSocket do end test "connect with valid token", %{user: user} do - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") socket = socket(UserSocket, "nonautenticated-user", %{}) {:ok, socket} = UserSocket.connect(%{"token" => "Bearer #{access_token.token}"}, socket) diff --git a/test/web/controllers/badge_controller_test.exs b/test/web/controllers/badge_controller_test.exs index c847283da..ccc1710de 100644 --- a/test/web/controllers/badge_controller_test.exs +++ b/test/web/controllers/badge_controller_test.exs @@ -4,7 +4,6 @@ defmodule AccentTest.BadgeController do import Mock alias Accent.Project - alias Accent.Repo defp behave_like_valid_response(response) do assert response.status == 200 @@ -14,7 +13,7 @@ defmodule AccentTest.BadgeController do setup do id = Ecto.UUID.generate() - project = Repo.insert!(%Project{id: id, name: "project", main_color: "#f00"}) + project = Factory.insert(Project, id: id, name: "project", main_color: "#f00") badge_generate_mock = [generate: fn _, _ -> {:ok, ""} end] {:ok, %{project: project, badge_generate_mock: badge_generate_mock}} diff --git a/test/web/controllers/export_controller_test.exs b/test/web/controllers/export_controller_test.exs index b7454a9bf..96364a20c 100644 --- a/test/web/controllers/export_controller_test.exs +++ b/test/web/controllers/export_controller_test.exs @@ -4,34 +4,31 @@ defmodule AccentTest.ExportController do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User alias Accent.Version - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [user: user, project: project, revision: revision, language: french_language]} end test "export inline", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) params = %{ inline_render: true, @@ -53,15 +50,15 @@ defmodule AccentTest.ExportController do end test "export basic", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) params = %{ project_id: project.id, @@ -82,8 +79,8 @@ defmodule AccentTest.ExportController do end test "export unknown language for the project", %{conn: conn, project: project} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) - language = Repo.insert!(%Language{name: "chinese", slug: Ecto.UUID.generate()}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") + language = Factory.insert(Language) params = %{ project_id: project.id, @@ -98,24 +95,24 @@ defmodule AccentTest.ExportController do end test "export document", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) - other_document = Repo.insert!(%Document{project_id: project.id, path: "test3", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") + other_document = Factory.insert(Document, project_id: project.id, path: "test3", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: other_document.id - }) + ) params = %{ project_id: project.id, @@ -143,25 +140,25 @@ defmodule AccentTest.ExportController do end test "export version", %{conn: conn, user: user, project: project, revision: revision, language: language} do - version = Repo.insert!(%Version{project_id: project.id, user_id: user.id, name: "Current", tag: "master"}) - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + version = Factory.insert(Version, project_id: project.id, user_id: user.id, name: "Current", tag: "master") + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id, version_id: version.id - }) + ) params = %{ version: "master", @@ -181,25 +178,25 @@ defmodule AccentTest.ExportController do end test "export without version", %{conn: conn, user: user, project: project, revision: revision, language: language} do - version = Repo.insert!(%Version{project_id: project.id, user_id: user.id, name: "Current", tag: "master"}) - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + version = Factory.insert(Version, project_id: project.id, user_id: user.id, name: "Current", tag: "master") + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id, version_id: version.id - }) + ) params = %{ project_id: project.id, @@ -218,7 +215,7 @@ defmodule AccentTest.ExportController do end test "export with unknown version", %{conn: conn, project: project, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") params = %{ version: "foo", @@ -234,23 +231,23 @@ defmodule AccentTest.ExportController do end test "export with order", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id - }) + ) params = %{ order_by: "key", @@ -271,25 +268,25 @@ defmodule AccentTest.ExportController do end test "export with default order", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id, file_index: 2 - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id, file_index: 1 - }) + ) params = %{ order_by: "", @@ -310,27 +307,29 @@ defmodule AccentTest.ExportController do end if Langue.Formatter.Rails.enabled?() do + alias Accent.Repo + test "export with language overrides", %{conn: conn, project: project, revision: revision} do revision = Repo.update!(Ecto.Changeset.change(revision, %{slug: "testtest"})) - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "rails_yml"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "rails_yml") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id, file_index: 2 - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id, file_index: 1 - }) + ) params = %{ order_by: "", @@ -356,9 +355,9 @@ defmodule AccentTest.ExportController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test", format: "android_xml"}) + document = Factory.insert(Document, project_id: project.id, path: "test", format: "android_xml") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "days.one", corrected_text: "bar", @@ -366,9 +365,9 @@ defmodule AccentTest.ExportController do plural: true, document_id: document.id, file_index: 2 - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "days.other", corrected_text: "foo", @@ -376,7 +375,7 @@ defmodule AccentTest.ExportController do plural: true, document_id: document.id, file_index: 1 - }) + ) params = %{ order_by: "", diff --git a/test/web/controllers/export_jipt_controller_test.exs b/test/web/controllers/export_jipt_controller_test.exs index c80ead15d..2303c2225 100644 --- a/test/web/controllers/export_jipt_controller_test.exs +++ b/test/web/controllers/export_jipt_controller_test.exs @@ -4,33 +4,30 @@ defmodule AccentTest.ExportJIPTController do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + french_language = Factory.insert(Language) + project = Factory.insert(Project) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [user: user, project: project, revision: revision, language: french_language]} end test "export inline", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) params = %{ inline_render: true, @@ -52,15 +49,15 @@ defmodule AccentTest.ExportJIPTController do end test "export basic", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) params = %{ project_id: project.id, @@ -81,24 +78,24 @@ defmodule AccentTest.ExportJIPTController do end test "export document", %{conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) - other_document = Repo.insert!(%Document{project_id: project.id, path: "test3", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") + other_document = Factory.insert(Document, project_id: project.id, path: "test3", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: other_document.id - }) + ) params = %{ project_id: project.id, @@ -127,27 +124,29 @@ defmodule AccentTest.ExportJIPTController do end if Langue.Formatter.Rails.enabled?() do + alias Accent.Repo + test "export with language overrides", %{conn: conn, project: project, revision: revision, language: language} do revision = Repo.update!(Ecto.Changeset.change(revision, %{slug: "testtest"})) - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "rails_yml"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "rails_yml") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "ok", corrected_text: "bar", proposed_text: "bar", document_id: document.id, file_index: 2 - }) + ) - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", corrected_text: "foo", proposed_text: "foo", document_id: document.id, file_index: 1 - }) + ) params = %{ order_by: "", diff --git a/test/web/controllers/format_controller_test.exs b/test/web/controllers/format_controller_test.exs index 18246b828..7639d330e 100644 --- a/test/web/controllers/format_controller_test.exs +++ b/test/web/controllers/format_controller_test.exs @@ -5,17 +5,14 @@ defmodule AccentTest.FormatController do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id) + project = Factory.insert(Project) {:ok, [user: user, project: project, access_token: access_token]} end @@ -74,8 +71,8 @@ defmodule AccentTest.FormatController do end test "format order_by same as export", %{conn: conn, project: project, access_token: access_token} do - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + french_language = Factory.insert(Language) + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) file = %Plug.Upload{ content_type: "application/json", @@ -97,7 +94,7 @@ defmodule AccentTest.FormatController do |> put_req_header("authorization", "Bearer #{access_token.token}") |> post(format_path(conn, :format), body) - document = Repo.insert!(%Document{project_id: project.id, path: "ordering", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "ordering", format: "json") params = %{ order_by: "key", @@ -110,13 +107,13 @@ defmodule AccentTest.FormatController do content = Jason.decode!(File.read!(file.path)) for {key, value} <- content do - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: key, corrected_text: value, proposed_text: value, document_id: document.id - }) + ) end export_response = diff --git a/test/web/controllers/lint_controller_test.exs b/test/web/controllers/lint_controller_test.exs index ac3903093..d6f164ba1 100644 --- a/test/web/controllers/lint_controller_test.exs +++ b/test/web/controllers/lint_controller_test.exs @@ -6,30 +6,27 @@ defmodule AccentTest.LintController do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/lint.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, user: user, project: project, revision: revision, language: french_language]} end test "lint document", %{access_token: access_token, conn: conn, project: project, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") body = %{ file: file(), diff --git a/test/web/controllers/merge_controller_mock_test.exs b/test/web/controllers/merge_controller_mock_test.exs index e264e468a..24938144a 100644 --- a/test/web/controllers/merge_controller_mock_test.exs +++ b/test/web/controllers/merge_controller_mock_test.exs @@ -9,29 +9,26 @@ defmodule AccentTest.MergeControllerMock do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/simple.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, project: project, language: french_language]} end test "sync with failure", %{access_token: access_token, conn: conn, project: project, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") body = %{ file: file(), diff --git a/test/web/controllers/merge_controller_test.exs b/test/web/controllers/merge_controller_test.exs index 290e8eb2c..2e0733571 100644 --- a/test/web/controllers/merge_controller_test.exs +++ b/test/web/controllers/merge_controller_test.exs @@ -14,20 +14,18 @@ defmodule AccentTest.MergeController do alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/simple.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, user: user, project: project, revision: revision, language: french_language]} end @@ -40,16 +38,16 @@ defmodule AccentTest.MergeController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -66,15 +64,7 @@ defmodule AccentTest.MergeController do assert response.status == 200 - assert_enqueued( - worker: Accent.Hook.Outbounds.Mock, - args: %{ - "event" => "add_translations", - "payload" => %{"language_name" => "french", "merge_type" => nil}, - "project_id" => project.id, - "user_id" => user.id - } - ) + assert_enqueued(worker: Movement.Persisters.ProjectHookWorker) merge_on_proposed_operation = Repo.one(from(o in Operation, where: [action: ^"merge_on_proposed"])) merge_operation = Repo.one(from(o in Operation, where: [action: ^"merge"])) @@ -91,16 +81,16 @@ defmodule AccentTest.MergeController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -133,16 +123,16 @@ defmodule AccentTest.MergeController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: false, corrected_text: "initial", proposed_text: "modified", document_id: document.id - }) + ) body = %{ file: file(), @@ -160,15 +150,7 @@ defmodule AccentTest.MergeController do assert response.status == 200 - assert_enqueued( - worker: Accent.Hook.Outbounds.Mock, - args: %{ - "event" => "add_translations", - "payload" => %{"language_name" => "french", "merge_type" => nil}, - "project_id" => project.id, - "user_id" => user.id - } - ) + assert_enqueued(worker: Movement.Persisters.ProjectHookWorker) merge_on_corrected_force_operation = Repo.one(from(o in Operation, where: [action: ^"merge_on_corrected_force"])) merge_operation = Repo.one(from(o in Operation, where: [action: ^"merge"])) @@ -187,16 +169,16 @@ defmodule AccentTest.MergeController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -213,15 +195,7 @@ defmodule AccentTest.MergeController do assert response.status == 200 - assert_enqueued( - worker: Accent.Hook.Outbounds.Mock, - args: %{ - "event" => "add_translations", - "payload" => %{"language_name" => "french", "merge_type" => nil}, - "project_id" => project.id, - "user_id" => user.id - } - ) + assert_enqueued(worker: Movement.Persisters.ProjectHookWorker) merge_on_proposed_operation = Repo.one(from(o in Operation, where: [action: ^"merge_on_proposed"])) merge_operation = Repo.one(from(o in Operation, where: [action: ^"merge"])) diff --git a/test/web/controllers/peek_controller_test.exs b/test/web/controllers/peek_controller_test.exs index f249f4adc..9a05566bd 100644 --- a/test/web/controllers/peek_controller_test.exs +++ b/test/web/controllers/peek_controller_test.exs @@ -6,40 +6,37 @@ defmodule AccentTest.PeekController do alias Accent.Document alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.Translation alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/simple.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - revision = Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + revision = Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, user: user, project: project, revision: revision, language: french_language]} end test "merge", %{access_token: access_token, conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -74,16 +71,16 @@ defmodule AccentTest.PeekController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -118,16 +115,16 @@ defmodule AccentTest.PeekController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -155,16 +152,16 @@ defmodule AccentTest.PeekController do revision: revision, language: language } do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: false, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), @@ -194,16 +191,16 @@ defmodule AccentTest.PeekController do end test "sync", %{access_token: access_token, conn: conn, project: project, revision: revision, language: language} do - document = Repo.insert!(%Document{project_id: project.id, path: "test2", format: "json"}) + document = Factory.insert(Document, project_id: project.id, path: "test2", format: "json") - Repo.insert!(%Translation{ + Factory.insert(Translation, revision_id: revision.id, key: "test", conflicted: true, corrected_text: "initial", proposed_text: "initial", document_id: document.id - }) + ) body = %{ file: file(), diff --git a/test/web/controllers/sync_controller_mock_test.exs b/test/web/controllers/sync_controller_mock_test.exs index b4e6703c8..90bf1cd03 100644 --- a/test/web/controllers/sync_controller_mock_test.exs +++ b/test/web/controllers/sync_controller_mock_test.exs @@ -8,24 +8,21 @@ defmodule AccentTest.SyncControllerMock do alias Accent.Collaborator alias Accent.Language alias Accent.Project - alias Accent.Repo alias Accent.Revision alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/simple.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, project: project, language: french_language]} end diff --git a/test/web/controllers/sync_controller_test.exs b/test/web/controllers/sync_controller_test.exs index c56bdb814..bd700f92d 100644 --- a/test/web/controllers/sync_controller_test.exs +++ b/test/web/controllers/sync_controller_test.exs @@ -13,20 +13,18 @@ defmodule AccentTest.SyncController do alias Accent.Revision alias Accent.User - @user %User{email: "test@test.com"} - def file(filename \\ "simple.json") do %Plug.Upload{content_type: "application/json", filename: filename, path: "test/support/formatter/json/simple.json"} end setup do - user = Repo.insert!(@user) - access_token = Repo.insert!(%AccessToken{user_id: user.id, token: "test-token"}) - french_language = Repo.insert!(%Language{name: "french", slug: Ecto.UUID.generate()}) - project = Repo.insert!(%Project{main_color: "#f00", name: "My project"}) + user = Factory.insert(User) + access_token = Factory.insert(AccessToken, user_id: user.id, token: "test-token") + french_language = Factory.insert(Language) + project = Factory.insert(Project) - Repo.insert!(%Collaborator{project_id: project.id, user_id: user.id, role: "admin"}) - Repo.insert!(%Revision{language_id: french_language.id, project_id: project.id, master: true}) + Factory.insert(Collaborator, project_id: project.id, user_id: user.id, role: "admin") + Factory.insert(Revision, language_id: french_language.id, project_id: project.id, master: true) {:ok, [access_token: access_token, user: user, project: project, language: french_language]} end @@ -52,18 +50,7 @@ defmodule AccentTest.SyncController do assert response.status == 200 - assert_enqueued( - worker: Accent.Hook.Outbounds.Mock, - args: %{ - "event" => "sync", - "payload" => %{ - "batch_operation_stats" => [%{"action" => "new", "count" => 3}], - "document_path" => "simple" - }, - "project_id" => project.id, - "user_id" => user.id - } - ) + assert_enqueued(worker: Movement.Persisters.ProjectHookWorker) assert Enum.map(Repo.all(Document), &Map.get(&1, :path)) == ["simple"] diff --git a/vendor/language_tool/lib/language_tool/backend.ex b/vendor/language_tool/lib/language_tool/backend.ex index 4a581eaa3..e1b28a39f 100644 --- a/vendor/language_tool/lib/language_tool/backend.ex +++ b/vendor/language_tool/lib/language_tool/backend.ex @@ -26,6 +26,7 @@ defmodule LanguageTool.Backend do backend else Logger.warning("LanguageTool could not be started. Install JRE and build the jar in #{jar_file()} to enable it") + nil end end @@ -81,17 +82,29 @@ defmodule LanguageTool.Backend do end defp process_check(process, lang, text) do - lang = sanitize_lang(lang) - Exile.Process.write(process, IO.iodata_to_binary([String.pad_trailing(lang, 7), text, "\n"])) + write_process(process, lang, text) - with {:ok, data} <- Exile.Process.read(process), + with {:ok, data} <- read_process_until_result(process), {:ok, data} <- Jason.decode(data) do data else - _ -> nil + _ -> + nil + end + end + + defp read_process_until_result(process) do + case Exile.Process.read(process) do + {:ok, "\n"} -> read_process_until_result(process) + result -> result end end + defp write_process(process, lang, text) do + lang = sanitize_lang(lang) + Exile.Process.write(process, IO.iodata_to_binary([String.pad_trailing(lang, 7), text, "\n"])) + end + defp sanitize_lang(lang) do if lang === "en" do "en-US" diff --git a/webapp/.ember-cli b/webapp/.ember-cli index aeb9593ce..163515bbb 100644 --- a/webapp/.ember-cli +++ b/webapp/.ember-cli @@ -1,4 +1,4 @@ { "disableAnalytics": false, - "usePods": true + "isTypeScriptProject": true } diff --git a/webapp/.template-lintrc.js b/webapp/.template-lintrc.js index 28a0a4e74..1d872321d 100644 --- a/webapp/.template-lintrc.js +++ b/webapp/.template-lintrc.js @@ -3,6 +3,9 @@ module.exports = { plugins: ["ember-template-lint-plugin-prettier"], extends: ["recommended", "ember-template-lint-plugin-prettier:recommended"], + ignore: [ + 'webapp/app/templates/components/jipt-example' + ], rules: { 'attribute-indentation': {'open-invocation-max-len': 120}, 'block-indentation': 2, @@ -13,6 +16,7 @@ module.exports = { 'link-rel-noopener': true, 'no-abstract-roles': true, 'no-bare-strings': true, + 'block-indentation': false, 'no-potential-path-strings': false, 'no-yield-only': false, 'no-autofocus-attribute': false, @@ -34,6 +38,7 @@ module.exports = { 'no-shadowed-elements': true, 'no-trailing-spaces': true, 'no-triple-curlies': false, + 'no-inline-styles': false, 'no-unused-block-params': true, quotes: false, 'require-valid-alt-text': false, @@ -41,7 +46,7 @@ module.exports = { 'require-button-type': false, 'self-closing-void-elements': false, 'simple-unless': false, - 'style-concatenation': true, + 'style-concatenation': false, 'table-groups': true, 'template-length': [true, {min: 1, max: 200}], } diff --git a/webapp/MODULE_REPORT.md b/webapp/MODULE_REPORT.md new file mode 100644 index 000000000..9cadde035 --- /dev/null +++ b/webapp/MODULE_REPORT.md @@ -0,0 +1,42 @@ +## Module Report +### Unknown Global + +**Global**: `Ember.onerror` + +**Location**: `app/services/raven.js` at line 89 + +```js + enableGlobalErrorCatching() { + if (this.isRavenUsable && !this.globalErrorCatchingInitialized) { + const _oldOnError = Ember.onerror; + + Ember.onerror = (error) => { +``` + +### Unknown Global + +**Global**: `Ember.onerror` + +**Location**: `app/services/raven.js` at line 89 + +```js + enableGlobalErrorCatching() { + if (this.isRavenUsable && !this.globalErrorCatchingInitialized) { + const _oldOnError = Ember.onerror; + + Ember.onerror = (error) => { +``` + +### Unknown Global + +**Global**: `Ember.onerror` + +**Location**: `app/services/raven.js` at line 91 + +```js + const _oldOnError = Ember.onerror; + + Ember.onerror = (error) => { + if (this._ignoreError(error)) { + return; +``` diff --git a/webapp/app/app.js b/webapp/app/app.js index d8e2088b6..e08f51c26 100644 --- a/webapp/app/app.js +++ b/webapp/app/app.js @@ -5,7 +5,6 @@ import config from './config/environment'; export default class App extends Application { modulePrefix = config.modulePrefix; - podModulePrefix = config.podModulePrefix; Resolver = Resolver; } diff --git a/webapp/app/component-helpers/percentage.ts b/webapp/app/component-helpers/percentage.ts index c961cb13e..1edc82026 100644 --- a/webapp/app/component-helpers/percentage.ts +++ b/webapp/app/component-helpers/percentage.ts @@ -2,7 +2,7 @@ export default (count: number, total: number) => { const percentage = (count / total) * 100; if (percentage) { - if (percentage % 1 !== 0) return percentage.toFixed(2); + if (percentage % 1 !== 0) return Number(percentage.toFixed(2)); return percentage; } diff --git a/webapp/app/pods/components/acc-avatar-img/component.ts b/webapp/app/components/acc-avatar-img/component.ts similarity index 100% rename from webapp/app/pods/components/acc-avatar-img/component.ts rename to webapp/app/components/acc-avatar-img/component.ts diff --git a/webapp/app/pods/components/acc-badge/component.ts b/webapp/app/components/acc-badge/component.ts similarity index 100% rename from webapp/app/pods/components/acc-badge/component.ts rename to webapp/app/components/acc-badge/component.ts diff --git a/webapp/app/components/acc-emoji-picker/component.ts b/webapp/app/components/acc-emoji-picker/component.ts new file mode 100644 index 000000000..2b7da0e5d --- /dev/null +++ b/webapp/app/components/acc-emoji-picker/component.ts @@ -0,0 +1,30 @@ +import Component from '@glimmer/component'; +import {action} from '@ember/object'; +import {tracked} from '@glimmer/tracking'; +import {Picker} from 'emoji-picker-element'; + +interface Args { + onPicked: (value: string) => void; +} + +export default class EmojiPicker extends Component { + @tracked + picker?: Picker; + + @action + togglePicker() { + if (this.picker) { + this.picker = undefined; + } else { + this.picker = new Picker({locale: 'fr'}); + this._bindClick(this.picker); + } + } + + _bindClick(picker: Picker) { + picker.addEventListener('emoji-click', (event: CustomEvent) => { + this.args.onPicked(event.detail.unicode); + this.togglePicker(); + }); + } +} diff --git a/webapp/app/pods/components/acc-flash-message/component.ts b/webapp/app/components/acc-flash-message/component.ts similarity index 100% rename from webapp/app/pods/components/acc-flash-message/component.ts rename to webapp/app/components/acc-flash-message/component.ts diff --git a/webapp/app/pods/components/acc-modal/component.ts b/webapp/app/components/acc-modal/component.ts similarity index 100% rename from webapp/app/pods/components/acc-modal/component.ts rename to webapp/app/components/acc-modal/component.ts diff --git a/webapp/app/pods/components/acc-select/component.ts b/webapp/app/components/acc-select/component.ts similarity index 100% rename from webapp/app/pods/components/acc-select/component.ts rename to webapp/app/components/acc-select/component.ts diff --git a/webapp/app/pods/components/activity-item/component.ts b/webapp/app/components/activity-item/component.ts similarity index 98% rename from webapp/app/pods/components/activity-item/component.ts rename to webapp/app/components/activity-item/component.ts index 3f5adc2c5..3c3f8a9b3 100644 --- a/webapp/app/pods/components/activity-item/component.ts +++ b/webapp/app/components/activity-item/component.ts @@ -28,7 +28,7 @@ const ACTIONS_ICON_PATHS = { remove: 'assets/x.svg', new_comment: 'assets/bubble.svg', new_slave: 'assets/language.svg', - document_delete: 'assets/file.svg', + document_delete: 'assets/file.svg' }; const MAXIMUM_COMPACT_BATCH_OPERATION_DOCUMENT_PATHS = 6; @@ -171,7 +171,7 @@ export default class ActivityItem extends Component { 'correct_all', 'batch_correct_conflict', 'batch_update', - 'conflict_on_slave', + 'conflict_on_slave' ]; return ( diff --git a/webapp/app/pods/components/activity-item/stats/component.ts b/webapp/app/components/activity-item/stats/component.ts similarity index 100% rename from webapp/app/pods/components/activity-item/stats/component.ts rename to webapp/app/components/activity-item/stats/component.ts diff --git a/webapp/app/pods/components/application-footer/component.ts b/webapp/app/components/application-footer/component.ts similarity index 100% rename from webapp/app/pods/components/application-footer/component.ts rename to webapp/app/components/application-footer/component.ts diff --git a/webapp/app/pods/components/async-button/component.ts b/webapp/app/components/async-button/component.ts similarity index 100% rename from webapp/app/pods/components/async-button/component.ts rename to webapp/app/components/async-button/component.ts diff --git a/webapp/app/pods/components/azure-push-form/component.ts b/webapp/app/components/azure-push-form/component.ts similarity index 91% rename from webapp/app/pods/components/azure-push-form/component.ts rename to webapp/app/components/azure-push-form/component.ts index ef532f5a2..f71d498a0 100644 --- a/webapp/app/pods/components/azure-push-form/component.ts +++ b/webapp/app/components/azure-push-form/component.ts @@ -7,7 +7,7 @@ interface Args { project: any; onPush: ({ targetVersion, - specificVersion, + specificVersion }: { targetVersion: string; specificVersion: string | null; @@ -19,18 +19,18 @@ export default class AzurePushForm extends Component { { value: 'LATEST', label: - 'components.project_settings.integrations.target_version.options.latest', + 'components.project_settings.integrations.target_version.options.latest' }, { value: 'SPECIFIC', label: - 'components.project_settings.integrations.target_version.options.specific', + 'components.project_settings.integrations.target_version.options.specific' }, { value: 'ALL', label: - 'components.project_settings.integrations.target_version.options.all', - }, + 'components.project_settings.integrations.target_version.options.all' + } ]; @tracked @@ -48,7 +48,7 @@ export default class AzurePushForm extends Component { await this.args.onPush({ targetVersion: this.targetVersion, - specificVersion: this.specificVersion, + specificVersion: this.specificVersion }); this.isSubmitting = false; diff --git a/webapp/app/pods/components/commit-file/component.ts b/webapp/app/components/commit-file/component.ts similarity index 97% rename from webapp/app/pods/components/commit-file/component.ts rename to webapp/app/components/commit-file/component.ts index b88d6cdb0..ba3828c7f 100644 --- a/webapp/app/pods/components/commit-file/component.ts +++ b/webapp/app/components/commit-file/component.ts @@ -19,7 +19,7 @@ const DEFAULT_PROPERTIES = { file: null, fileSource: null, documentPath: null, - documentFormat: 'json', + documentFormat: 'json' }; interface Args { @@ -140,7 +140,7 @@ export default class CommitFile extends Component { return this.args.revisions.map( ({id, language}: {id: string; language: {name: string}}) => ({ label: language.name, - value: id, + value: id }) ); } @@ -152,14 +152,14 @@ export default class CommitFile extends Component { get mappedMergeTypes() { return this.mergeTypes.map((name) => ({ label: name, - value: name, + value: name })); } get mappedSyncTypes() { return this.syncTypes.map((name) => ({ label: name, - value: name, + value: name })); } @@ -167,12 +167,12 @@ export default class CommitFile extends Component { return [ { label: this.intl.t('components.commit_file.no_version_label'), - value: null, + value: null }, ...this.args.versions.map(({id, tag}: {id: string; tag: string}) => ({ label: tag, - value: id, - })), + value: id + })) ]; } @@ -187,7 +187,7 @@ export default class CommitFile extends Component { return this.globalState.documentFormats.map(({slug, name}) => ({ value: slug, - label: name, + label: name })); } @@ -251,7 +251,7 @@ export default class CommitFile extends Component { revision: this.revision, mergeType: this.mergeType.value, syncType: this.syncType.value, - mergeOptions: this.correctOnMerge ? ['correct'] : [], + mergeOptions: this.correctOnMerge ? ['correct'] : [] }); this.onCommitingDone(); @@ -273,7 +273,7 @@ export default class CommitFile extends Component { version: this.version && this.version.tag, mergeType: this.mergeType.value, syncType: this.syncType.value, - mergeOptions: this.correctOnMerge ? ['correct'] : [], + mergeOptions: this.correctOnMerge ? ['correct'] : [] }); this.onPeekingDone(); diff --git a/webapp/app/pods/components/conflicts-filters/component.ts b/webapp/app/components/conflicts-filters/component.ts similarity index 53% rename from webapp/app/pods/components/conflicts-filters/component.ts rename to webapp/app/components/conflicts-filters/component.ts index d66fcf43a..4a6a5886f 100644 --- a/webapp/app/pods/components/conflicts-filters/component.ts +++ b/webapp/app/components/conflicts-filters/component.ts @@ -3,11 +3,9 @@ import {inject as service} from '@ember/service'; import {gt} from '@ember/object/computed'; import Component from '@glimmer/component'; import IntlService from 'ember-intl/services/intl'; -import {PaginationMeta} from 'accent-webapp/pods/components/resource-pagination/component'; +import {PaginationMeta} from 'accent-webapp/components/resource-pagination/component'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 1000; // ms @@ -16,12 +14,16 @@ interface Args { conflicts: any; document: any; documents: any; + relatedRevisions: any; + defaultRelatedRevisions: any[]; + revisions: any; version: any; versions: any; query: any; + withAdvancedFilters: boolean; onChangeDocument: () => void; - onChangeReference: () => void; onChangeVersion: () => void; + onChangeRevisions: () => void; onChangeQuery: (query: string) => void; } @@ -32,20 +34,29 @@ export default class ConflictsFilters extends Component { @gt('args.documents.length', 1) showDocumentsSelect: boolean; + @gt('args.revisions.length', 1) + showRevisionsSelect: boolean; + @gt('args.versions.length', 0) showVersionsSelect: boolean; + get showSomeFilters() { + return this.showDocumentsSelect || this.showVersionsSelect; + } + + @tracked + displayAdvancedFilters = this.args.withAdvancedFilters; + @tracked debouncedQuery = this.args.query; - @restartableTask - *debounceQuery(query: string) { + debounceQuery = restartableTask(async (query: string) => { this.debouncedQuery = query; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.args.onChangeQuery(this.debouncedQuery); - } + }); get mappedDocuments() { if (!this.args.documents) return []; @@ -53,7 +64,7 @@ export default class ConflictsFilters extends Component { const documents = this.args.documents.map( ({id, path}: {id: string; path: string}) => ({ label: path, - value: id, + value: id }) ); @@ -61,12 +72,35 @@ export default class ConflictsFilters extends Component { label: this.intl.t( 'components.conflicts_filters.document_default_option_text' ), - value: '', + value: '' }); return documents; } + get relatedRevisionsValue() { + if (this.args.relatedRevisions.length === 0) { + const revisionIds = this.args.defaultRelatedRevisions.map( + ({id}: any) => id + ); + return this.mappedRevisions.filter(({value}: {value: string}) => + revisionIds.includes(value) + ); + } + + return this.mappedRevisions.filter(({value}: {value: string}) => + this.args.relatedRevisions?.includes(value) + ); + } + + get mappedRevisionsOptions() { + const values = this.relatedRevisionsValue.map(({value}: any) => value); + + return this.mappedRevisions.filter( + ({value}: {value: string}) => !values.includes(value) + ); + } + get documentValue() { return this.mappedDocuments.find( ({value}: {value: string}) => value === this.args.document @@ -77,7 +111,7 @@ export default class ConflictsFilters extends Component { const versions = this.args.versions.map( ({id, tag}: {id: string; tag: string}) => ({ label: tag, - value: id, + value: id }) ); @@ -85,12 +119,26 @@ export default class ConflictsFilters extends Component { label: this.intl.t( 'components.conflicts_filters.version_default_option_text' ), - value: '', + value: '' }); return versions; } + get mappedRevisions() { + return this.args.revisions.map( + (revision: { + id: string; + name: string | null; + slug: string | null; + language: {slug: string; name: string}; + }) => ({ + label: revision.name || revision.language.name, + value: revision.id + }) + ); + } + get versionValue() { return this.mappedVersions.find( ({value}: {value: string}) => value === this.args.version @@ -101,7 +149,12 @@ export default class ConflictsFilters extends Component { setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQuery.perform(target.value); + } + + @action + toggleAdvancedFilters() { + this.displayAdvancedFilters = !this.displayAdvancedFilters; } @action diff --git a/webapp/app/components/conflicts-list/advanced-filters/component.ts b/webapp/app/components/conflicts-list/advanced-filters/component.ts new file mode 100644 index 000000000..1d2a10bd4 --- /dev/null +++ b/webapp/app/components/conflicts-list/advanced-filters/component.ts @@ -0,0 +1,21 @@ +import Component from '@glimmer/component'; + +interface Args { + revisions: any[]; + isTextEmptyFilter: boolean; + isTextNotEmptyFilter: boolean; + isAddedLastSyncFilter: boolean; + isNotTranslatedFilter: boolean; + isCommentedOnFilter: boolean; + onChangeAdvancedFilterBoolean: ( + key: + | 'isTextEmpty' + | 'isTextNotEmpty' + | 'isAddedLastSync' + | 'isCommentedOn' + | 'isNotTranslated', + event: InputEvent + ) => void; +} + +export default class AdvancedFilters extends Component {} diff --git a/webapp/app/components/conflicts-list/component.ts b/webapp/app/components/conflicts-list/component.ts new file mode 100644 index 000000000..d98f1cf58 --- /dev/null +++ b/webapp/app/components/conflicts-list/component.ts @@ -0,0 +1,58 @@ +import {tracked} from '@glimmer/tracking'; +import {action} from '@ember/object'; +import Component from '@glimmer/component'; + +interface Args { + permissions: Record; + project: any; + groupedTranslations: any; + version: any; + versions: any[]; + query: any; + onCorrect: (conflict: any, textInput: string) => Promise; + onCopyTranslation: ( + text: string, + sourceLanguageSlug: string, + targetLanguageSlug: string + ) => void; +} + +export default class ConflictsList extends Component { + @tracked + selectedTranslationId: string | null = null; + + get currentVersion() { + if (!this.args.versions) return; + if (!this.args.version) return; + + return this.args.versions.find( + (version) => version.id === this.args.version + ); + } + + get revisions() { + if (this.args.groupedTranslations.length === 0) return []; + + return this.args.groupedTranslations[0].translations.map( + ({revision}: any) => revision + ); + } + + get mappedRevisions() { + if (this.args.groupedTranslations.length === 0) return []; + + return this.args.groupedTranslations[0].translations.map( + ({revision}: any) => { + return { + name: revision.name || revision.language.name, + slug: revision.slug || revision.language.slug + }; + } + ); + } + + @action + handleFocus(id: string) { + this.selectedTranslationId = id; + } +} diff --git a/webapp/app/components/conflicts-list/group/component.ts b/webapp/app/components/conflicts-list/group/component.ts new file mode 100644 index 000000000..b24276351 --- /dev/null +++ b/webapp/app/components/conflicts-list/group/component.ts @@ -0,0 +1,29 @@ +import {action} from '@ember/object'; +import Component from '@glimmer/component'; +import parsedKeyProperty from 'accent-webapp/computed-macros/parsed-key'; + +interface Args { + selectedTranslationId: string | null; + groupedTranslation: { + key: string; + translations: any[]; + }; + onFocus: (id: string) => void; +} + +export default class ConflictsListGroup extends Component { + translationKey = parsedKeyProperty(this.args.groupedTranslation.key); + + get masterTranslation() { + return this.args.groupedTranslation.translations[0]; + } + + get isFocused() { + return this.masterTranslation.id === this.args.selectedTranslationId; + } + + @action + handleFocus() { + this.args.onFocus(this.masterTranslation.id); + } +} diff --git a/webapp/app/components/conflicts-list/item/component.ts b/webapp/app/components/conflicts-list/item/component.ts new file mode 100644 index 000000000..02ec285da --- /dev/null +++ b/webapp/app/components/conflicts-list/item/component.ts @@ -0,0 +1,192 @@ +import {action} from '@ember/object'; +import {empty} from '@ember/object/computed'; +import Component from '@glimmer/component'; +import parsedKeyProperty from 'accent-webapp/computed-macros/parsed-key'; +import {tracked} from '@glimmer/tracking'; +import {MutationResponse} from 'accent-webapp/services/apollo-mutate'; + +interface Translation { + id: string; + key: string; + conflictedText: string; + correctedText: string; + isConflicted: boolean; + revision: { + name: string | null; + slug: string | null; + rtl: boolean | null; + isMaster: boolean; + language: { + name: string; + slug: string; + rtl: boolean; + }; + }; +} + +interface Args { + permissions: Record; + index: number; + project: any; + prompts: any[]; + translation: Translation; + onFocus: () => void; + onBlur: () => void; + onCorrect: (translation: any, textInput: string) => Promise; + onUpdate: (translation: any, textInput: string) => Promise; + onUncorrect: ( + translation: any, + textInput: string + ) => Promise; +} + +export default class ConflictsListItem extends Component { + @empty('args.translation.conflictedText') + emptyPreviousText: boolean; + + @tracked + textInput = this.args.translation.correctedText; + + @tracked + conflictResolved = false; + + @tracked + isCorrectLoading = false; + + @tracked + isUncorrectLoading = false; + + @tracked + isUpdateLoading = false; + + @tracked + error = false; + + @tracked + inputDisabled = false; + + translationKey = parsedKeyProperty(this.args.translation.key); + textOriginal = this.args.translation.correctedText; + + get showOriginalButton() { + return this.textInput !== this.textOriginal; + } + + get revisionTextDirRtl() { + return this.args.translation.revision.rtl !== null + ? this.args.translation.revision.rtl + : this.args.translation.revision.language.rtl; + } + + get revisionSlug() { + return ( + this.args.translation.revision.slug || + this.args.translation.revision.language.slug + ); + } + + @action + changeTranslationText(text: string) { + this.textInput = text; + } + + @action + setOriginalText() { + this.textInput = this.textOriginal; + } + + @action + onUpdatingText() { + this.inputDisabled = true; + } + + @action + onUpdateText(value: string) { + this.textInput = value; + this.inputDisabled = false; + } + + @action + async correctConflict() { + this.onCorrectLoading(); + + const response = await this.args.onCorrect( + this.args.translation, + this.textInput + ); + + if (response.errors) { + this.onError(); + } else { + this.onCorrectSuccess(); + } + } + + @action + async uncorrectConflict() { + this.onUncorrectLoading(); + + const response = await this.args.onUncorrect( + this.args.translation, + this.textInput + ); + + if (response.errors) { + this.onError(); + } else { + this.onUncorrectSuccess(); + } + } + + @action + async updateConflict() { + this.onUpdateLoading(); + + const response = await this.args.onUpdate( + this.args.translation, + this.textInput + ); + + if (response.errors) { + this.onError(); + } else { + this.onUpdateSuccess(); + } + } + + private onCorrectLoading() { + this.error = false; + this.isCorrectLoading = true; + } + + private onUncorrectLoading() { + this.error = false; + this.isUncorrectLoading = true; + } + + private onUpdateLoading() { + this.error = false; + this.isUpdateLoading = true; + } + + private onError() { + this.error = true; + this.isUpdateLoading = false; + this.isCorrectLoading = false; + this.isUncorrectLoading = false; + } + + private onCorrectSuccess() { + this.conflictResolved = true; + this.isCorrectLoading = false; + } + + private onUncorrectSuccess() { + this.conflictResolved = false; + this.isCorrectLoading = false; + } + + private onUpdateSuccess() { + this.isUpdateLoading = false; + } +} diff --git a/webapp/app/pods/components/dashboard-revisions/component.ts b/webapp/app/components/dashboard-revisions/component.ts similarity index 100% rename from webapp/app/pods/components/dashboard-revisions/component.ts rename to webapp/app/components/dashboard-revisions/component.ts diff --git a/webapp/app/pods/components/dashboard-revisions/item/component.ts b/webapp/app/components/dashboard-revisions/item/component.ts similarity index 94% rename from webapp/app/pods/components/dashboard-revisions/item/component.ts rename to webapp/app/components/dashboard-revisions/item/component.ts index a3749ef1c..2db73b43c 100644 --- a/webapp/app/pods/components/dashboard-revisions/item/component.ts +++ b/webapp/app/components/dashboard-revisions/item/component.ts @@ -58,10 +58,10 @@ export default class DashboardRevisionsItem extends Component { ); } - get reviewsCount() { - const {conflictsCount, translationsCount} = this.args.revision; + get toReviewCount() { + const {reviewedCount, translationsCount} = this.args.revision; - return translationsCount - conflictsCount; + return translationsCount - reviewedCount; } get languageName() { diff --git a/webapp/app/pods/components/date-tag/component.ts b/webapp/app/components/date-tag/component.ts similarity index 94% rename from webapp/app/pods/components/date-tag/component.ts rename to webapp/app/components/date-tag/component.ts index 6df8d1aa1..498ada246 100644 --- a/webapp/app/pods/components/date-tag/component.ts +++ b/webapp/app/components/date-tag/component.ts @@ -1,6 +1,6 @@ import {inject as service} from '@ember/service'; import Component from '@glimmer/component'; -import dateFormat from 'date-fns/format'; +import {format as dateFormat} from 'date-fns'; import IntlService from 'ember-intl/services/intl'; interface Args { diff --git a/webapp/app/pods/components/documents-add-button/component.ts b/webapp/app/components/documents-add-button/component.ts similarity index 100% rename from webapp/app/pods/components/documents-add-button/component.ts rename to webapp/app/components/documents-add-button/component.ts diff --git a/webapp/app/pods/components/documents-list/component.ts b/webapp/app/components/documents-list/component.ts similarity index 100% rename from webapp/app/pods/components/documents-list/component.ts rename to webapp/app/components/documents-list/component.ts diff --git a/webapp/app/pods/components/documents-list/item/component.ts b/webapp/app/components/documents-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/documents-list/item/component.ts rename to webapp/app/components/documents-list/item/component.ts diff --git a/webapp/app/pods/components/documents-machine-translations-button/component.ts b/webapp/app/components/documents-machine-translations-button/component.ts similarity index 100% rename from webapp/app/pods/components/documents-machine-translations-button/component.ts rename to webapp/app/components/documents-machine-translations-button/component.ts diff --git a/webapp/app/pods/components/dummy-login-form/component.ts b/webapp/app/components/dummy-login-form/component.ts similarity index 100% rename from webapp/app/pods/components/dummy-login-form/component.ts rename to webapp/app/components/dummy-login-form/component.ts diff --git a/webapp/app/pods/components/empty-content/component.ts b/webapp/app/components/empty-content/component.ts similarity index 100% rename from webapp/app/pods/components/empty-content/component.ts rename to webapp/app/components/empty-content/component.ts diff --git a/webapp/app/pods/components/error-section/component.ts b/webapp/app/components/error-section/component.ts similarity index 100% rename from webapp/app/pods/components/error-section/component.ts rename to webapp/app/components/error-section/component.ts diff --git a/webapp/app/pods/components/file-export-all/component.ts b/webapp/app/components/file-export-all/component.ts similarity index 94% rename from webapp/app/pods/components/file-export-all/component.ts rename to webapp/app/components/file-export-all/component.ts index 1a98440a1..a8049449c 100644 --- a/webapp/app/pods/components/file-export-all/component.ts +++ b/webapp/app/components/file-export-all/component.ts @@ -41,8 +41,8 @@ export default class FileExport extends Component { filters: { isTextEmptyFilter: this.args.isTextEmptyFilter, isAddedLastSyncFilter: this.args.isAddedLastSyncFilter, - isConflictedFilter: this.args.isConflictedFilter, - }, + isConflictedFilter: this.args.isConflictedFilter + } }); this.content = data; diff --git a/webapp/app/pods/components/file-export/component.ts b/webapp/app/components/file-export/component.ts similarity index 95% rename from webapp/app/pods/components/file-export/component.ts rename to webapp/app/components/file-export/component.ts index 738b494c4..f6b013f3c 100644 --- a/webapp/app/pods/components/file-export/component.ts +++ b/webapp/app/components/file-export/component.ts @@ -43,8 +43,8 @@ export default class FileExport extends Component { filters: { isTextEmptyFilter: this.args.isTextEmptyFilter, isAddedLastSyncFilter: this.args.isAddedLastSyncFilter, - isConflictedFilter: this.args.isConflictedFilter, - }, + isConflictedFilter: this.args.isConflictedFilter + } }); this.content = data; diff --git a/webapp/app/pods/components/file-input/component.ts b/webapp/app/components/file-input/component.ts similarity index 100% rename from webapp/app/pods/components/file-input/component.ts rename to webapp/app/components/file-input/component.ts diff --git a/webapp/app/pods/components/flash-messages-list/component.ts b/webapp/app/components/flash-messages-list/component.ts similarity index 100% rename from webapp/app/pods/components/flash-messages-list/component.ts rename to webapp/app/components/flash-messages-list/component.ts diff --git a/webapp/app/pods/components/highlight-render/component.ts b/webapp/app/components/highlight-render/component.ts similarity index 98% rename from webapp/app/pods/components/highlight-render/component.ts rename to webapp/app/components/highlight-render/component.ts index f77081b8e..76deca1d7 100644 --- a/webapp/app/pods/components/highlight-render/component.ts +++ b/webapp/app/components/highlight-render/component.ts @@ -3,7 +3,7 @@ import {action} from '@ember/object'; import hljs from 'highlight.js'; hljs.configure({ - languages: ['javascript', 'json', 'php', 'xml', 'yaml', 'properties'], + languages: ['javascript', 'json', 'php', 'xml', 'yaml', 'properties'] }); interface Args { diff --git a/webapp/app/pods/components/html-textarea/component.ts b/webapp/app/components/html-textarea/component.ts similarity index 82% rename from webapp/app/pods/components/html-textarea/component.ts rename to webapp/app/components/html-textarea/component.ts index 0daec3834..119310383 100644 --- a/webapp/app/pods/components/html-textarea/component.ts +++ b/webapp/app/components/html-textarea/component.ts @@ -1,9 +1,7 @@ import {action} from '@ember/object'; import Component from '@glimmer/component'; import pell from 'pell'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 1000; // ms @@ -54,12 +52,12 @@ export default class HTMLTextarea extends Component { { icon: '✗', title: 'Clear', - result: clear(pell), - }, + result: clear(pell) + } ], onChange: (text) => { - perform(this.debounceChange, text); - }, + this.debounceChangeTask.perform(text); + } }); this.content = pellInstance.querySelector('.pell-content'); @@ -75,10 +73,9 @@ export default class HTMLTextarea extends Component { } } - @restartableTask - *debounceChange(text: string) { - yield timeout(DEBOUNCE_OFFSET); + debounceChangeTask = restartableTask(async (text: string) => { + await timeout(DEBOUNCE_OFFSET); this.args.onChange(text); - } + }); } diff --git a/webapp/app/pods/components/improve-prompt/component.ts b/webapp/app/components/improve-prompt/component.ts similarity index 83% rename from webapp/app/pods/components/improve-prompt/component.ts rename to webapp/app/components/improve-prompt/component.ts index 3c89e8c03..69cfebc1b 100644 --- a/webapp/app/pods/components/improve-prompt/component.ts +++ b/webapp/app/components/improve-prompt/component.ts @@ -2,8 +2,7 @@ import Component from '@glimmer/component'; import {inject as service} from '@ember/service'; import {action} from '@ember/object'; import {tracked} from '@glimmer/tracking'; -import {dropTask} from 'ember-concurrency-decorators'; -import {taskFor} from 'ember-concurrency-ts'; +import {dropTask} from 'ember-concurrency'; import Apollo from 'accent-webapp/services/apollo'; import improveTextPromptMutation from 'accent-webapp/queries/improve-text-prompt'; @@ -44,7 +43,7 @@ export default class ImprovePrompt extends Component { promptOpened = false; get isSubmitting() { - return taskFor(this.submitTask).isRunning; + return this.submitTask.isRunning; } @action @@ -64,23 +63,22 @@ export default class ImprovePrompt extends Component { this.promptOpened = false; } - @dropTask - *fetchPromptOptions() { + fetchPromptOptions = dropTask(async () => { const variables = {projectId: this.args.project.id}; - const {data} = yield this.apollo.client.query({ + const {data} = await this.apollo.client.query({ query: projectPrompts, fetchPolicy: 'network-only', - variables, + variables }); if (!data.viewer.project.prompts) return; this.promptOptions = data.viewer.project.prompts.map((prompt: Prompt) => ({ label: prompt.name, - value: prompt.id, + value: prompt.id })); this.promptOptionValue = this.promptOptions[0]; - } + }); @action onPromptClose() { @@ -93,8 +91,7 @@ export default class ImprovePrompt extends Component { this.promptOpened = true; } - @dropTask - *submitTask(promptId?: string) { + submitTask = dropTask(async (promptId?: string) => { if (!promptId && !this.promptOptionValue) return; if (!this.promptOpened) this.args.onUpdatingText(); @@ -102,11 +99,11 @@ export default class ImprovePrompt extends Component { const variables = { text: this.args.text, - promptId: promptId || this.promptOptionValue?.value, + promptId: promptId || this.promptOptionValue?.value }; - const {data} = yield this.apollo.client.mutate({ + const {data} = await this.apollo.client.mutate({ mutation: improveTextPromptMutation, - variables, + variables }); if (data.improveTextWithPrompt?.text) { @@ -116,5 +113,5 @@ export default class ImprovePrompt extends Component { this.args.onUpdateText(data.improveTextWithPrompt.text); } } - } + }); } diff --git a/webapp/app/pods/components/inline-machine-translate/component.ts b/webapp/app/components/inline-machine-translate/component.ts similarity index 72% rename from webapp/app/pods/components/inline-machine-translate/component.ts rename to webapp/app/components/inline-machine-translate/component.ts index a2ebb7b64..5f2e7d5eb 100644 --- a/webapp/app/pods/components/inline-machine-translate/component.ts +++ b/webapp/app/components/inline-machine-translate/component.ts @@ -1,7 +1,6 @@ import Component from '@glimmer/component'; import {inject as service} from '@ember/service'; -import {dropTask} from 'ember-concurrency-decorators'; -import {taskFor} from 'ember-concurrency-ts'; +import {dropTask} from 'ember-concurrency'; import Apollo from 'accent-webapp/services/apollo'; import projectTranslateTextQuery from 'accent-webapp/queries/translate-text-project'; @@ -18,26 +17,25 @@ export default class ImprovePrompt extends Component { apollo: Apollo; get isSubmitting() { - return taskFor(this.submitTask).isRunning; + return this.submitTask.isRunning; } - @dropTask - *submitTask(targetLanguageSlug: string) { + submitTask = dropTask(async (targetLanguageSlug: string) => { this.args.onUpdatingText(); const variables = { projectId: this.args.project.id, text: this.args.text, - targetLanguageSlug, + targetLanguageSlug }; - const {data} = yield this.apollo.client.query({ + const {data} = await this.apollo.client.query({ query: projectTranslateTextQuery, - variables, + variables }); if (data.viewer.project.translatedText?.text) { this.args.onUpdateText(data.viewer.project.translatedText?.text); } - } + }); } diff --git a/webapp/app/pods/components/jipt-back-to-translations/component.ts b/webapp/app/components/jipt-back-to-translations/component.ts similarity index 100% rename from webapp/app/pods/components/jipt-back-to-translations/component.ts rename to webapp/app/components/jipt-back-to-translations/component.ts diff --git a/webapp/app/components/jipt-example/component.ts b/webapp/app/components/jipt-example/component.ts new file mode 100644 index 000000000..d5cd1ca2c --- /dev/null +++ b/webapp/app/components/jipt-example/component.ts @@ -0,0 +1,40 @@ +import Component from '@glimmer/component'; + +interface ArgsTranslation { + key: string; + document: { + path: string; + }; +} + +interface Args { + project: { + id: string; + name: string; + revision: { + translations: { + entries: ArgsTranslation[]; + }; + }; + }; +} + +export default class JIPTExample extends Component { + get scriptSrc() { + return `${window.location.origin}/static/jipt/index.js`; + } + + get imageSrc() { + return ''; + } + + get scriptContent() { + return `window.accent=window.accent||function(){(accent.q=accent.q||[]).push(arguments);}; + accent('init',{h:'${window.location.origin}',i:'${this.args.project.id}'});`; + } + + get translationKey() { + const translation = this.args.project.revision.translations.entries[0]; + return `{^${translation.key}@${translation.document.path}}`; + } +} diff --git a/webapp/app/pods/components/jipt-export/component.ts b/webapp/app/components/jipt-export/component.ts similarity index 93% rename from webapp/app/pods/components/jipt-export/component.ts rename to webapp/app/components/jipt-export/component.ts index c0d5634b2..deedae82d 100644 --- a/webapp/app/pods/components/jipt-export/component.ts +++ b/webapp/app/components/jipt-export/component.ts @@ -25,7 +25,7 @@ export default class JIPTExport extends Component { project: this.args.project, document: this.args.document, version: this.args.version, - documentFormat: this.args.documentFormat, + documentFormat: this.args.documentFormat }); this.content = data; diff --git a/webapp/app/pods/components/jipt-header/component.ts b/webapp/app/components/jipt-header/component.ts similarity index 100% rename from webapp/app/pods/components/jipt-header/component.ts rename to webapp/app/components/jipt-header/component.ts diff --git a/webapp/app/pods/components/jipt-translations-filtered-title/component.ts b/webapp/app/components/jipt-translations-filtered-title/component.ts similarity index 100% rename from webapp/app/pods/components/jipt-translations-filtered-title/component.ts rename to webapp/app/components/jipt-translations-filtered-title/component.ts diff --git a/webapp/app/pods/components/jipt-translations-list/component.ts b/webapp/app/components/jipt-translations-list/component.ts similarity index 100% rename from webapp/app/pods/components/jipt-translations-list/component.ts rename to webapp/app/components/jipt-translations-list/component.ts diff --git a/webapp/app/pods/components/jipt-translations-list/item/component.ts b/webapp/app/components/jipt-translations-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/jipt-translations-list/item/component.ts rename to webapp/app/components/jipt-translations-list/item/component.ts diff --git a/webapp/app/pods/components/lint-options/component.ts b/webapp/app/components/lint-options/component.ts similarity index 81% rename from webapp/app/pods/components/lint-options/component.ts rename to webapp/app/components/lint-options/component.ts index dea52f161..cf0508a09 100644 --- a/webapp/app/pods/components/lint-options/component.ts +++ b/webapp/app/components/lint-options/component.ts @@ -1,13 +1,10 @@ import {action} from '@ember/object'; import {inject as service} from '@ember/service'; -import {gt} from '@ember/object/computed'; import Component from '@glimmer/component'; import IntlService from 'ember-intl/services/intl'; import GlobalState from 'accent-webapp/services/global-state'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 1000; // ms @@ -30,23 +27,28 @@ export default class RevisionExportOptions extends Component { @service('global-state') globalState: GlobalState; - @gt('mappedDocuments.length', 1) - showDocuments: boolean; + get showDocuments() { + return this.mappedDocuments.length > 1; + } - @gt('mappedVersions.length', 1) - showVersions: boolean; + get showVersions() { + return this.mappedVersions.length > 1; + } + + get showSomeFilters() { + return this.showDocuments || this.showVersions; + } @tracked debouncedQuery = this.args.query; - @restartableTask - *debounceQuery(query: string) { + debounceQuery = restartableTask(async (query: string) => { this.debouncedQuery = query; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.args.onChangeQuery(this.debouncedQuery); - } + }); get documentValue() { return ( @@ -62,7 +64,7 @@ export default class RevisionExportOptions extends Component { return this.args.documents.map( ({id, path}: {id: string; path: string}) => ({ label: path, - value: id, + value: id }) ); } @@ -81,16 +83,16 @@ export default class RevisionExportOptions extends Component { memo.concat([ { label: tag, - value: tag, - }, + value: tag + } ]), [ { label: this.intl.t( 'components.revision_export_options.default_version' ), - value: '', - }, + value: '' + } ] ); } @@ -113,7 +115,7 @@ export default class RevisionExportOptions extends Component { setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQuery.perform(target.value); } @action diff --git a/webapp/app/components/lint-translations-page/component.ts b/webapp/app/components/lint-translations-page/component.ts new file mode 100644 index 000000000..68f91595d --- /dev/null +++ b/webapp/app/components/lint-translations-page/component.ts @@ -0,0 +1,68 @@ +import {inject as service} from '@ember/service'; +import {tracked} from '@glimmer/tracking'; +import Component from '@glimmer/component'; +import {restartableTask} from 'ember-concurrency'; +import translationUpdateQuery from 'accent-webapp/queries/update-translation'; +import Apollo from 'accent-webapp/services/apollo'; + +interface Args { + project: any; + lintTranslations: any[]; + permissions: Record; +} + +interface Stat { + title: string; + count: number; +} + +export default class LintTranslationsPage extends Component { + @service('apollo') + apollo: Apollo; + + @tracked + fixLintMessageRunningTranslationId: string | null = null; + + get lintTranslationsStatsCount() { + return this.lintTranslationsStats.reduce( + (total, stat) => stat.count + total, + 0 + ); + } + + get lintTranslationsStats() { + const stats = this.args.lintTranslations.reduce((acc, lintTranslation) => { + lintTranslation.messages.forEach((message: {check: string}) => { + if (acc[message.check]) { + acc[message.check]++; + } else { + acc[message.check] = 1; + } + }); + + return acc; + }, {}); + + return Object.entries(stats).map(([title, count]) => ({ + title, + count + })) as Stat[]; + } + + fixLintMessageTask = restartableTask( + async (translation: {id: string}, message: any) => { + this.fixLintMessageRunningTranslationId = translation.id; + + await this.apollo.client.mutate({ + mutation: translationUpdateQuery, + refetchQueries: ['Lint'], + variables: { + text: message.replacement.value, + translationId: translation.id + } + }); + + this.fixLintMessageRunningTranslationId = null; + } + ); +} diff --git a/webapp/app/pods/components/lint-translations-page/item/component.ts b/webapp/app/components/lint-translations-page/item/component.ts similarity index 85% rename from webapp/app/pods/components/lint-translations-page/item/component.ts rename to webapp/app/components/lint-translations-page/item/component.ts index b9ae443f3..545cf6eee 100644 --- a/webapp/app/pods/components/lint-translations-page/item/component.ts +++ b/webapp/app/components/lint-translations-page/item/component.ts @@ -4,10 +4,11 @@ import parsedKeyProperty from 'accent-webapp/computed-macros/parsed-key'; interface Args { lintTranslation: any; project: any; + fix?: (lintTranslation: any, message: any) => void; } const escape = document.createElement('textarea'); -const escapeHTML = (html: string) => { +const escapeHTML = (html: string): string => { escape.textContent = html; return escape.innerHTML; }; @@ -15,6 +16,14 @@ const escapeHTML = (html: string) => { export default class LintTranslationsPageItem extends Component { translationKey = parsedKeyProperty(this.args.lintTranslation.translation.key); + get allReplacable() { + return this.args.lintTranslation.messages.every( + (message: {replacement: object | null}) => { + return Boolean(message.replacement); + } + ); + } + get messages() { const mapSet = new Set(); return this.args.lintTranslation.messages.flatMap((message: any) => { @@ -30,7 +39,7 @@ export default class LintTranslationsPageItem extends Component { get annotatedText() { let offsetTotal = 0; - let text = this.args.lintTranslation.messages + let text = Array.from(this.args.lintTranslation.messages) .sort((a: any, b: any) => a.offset || 0 >= b.offset || 0) .reduce((text: string, message: any) => { if (message.length) { @@ -64,7 +73,7 @@ export default class LintTranslationsPageItem extends Component { } else { return text; } - }, this.args.lintTranslation.messages[0].text); + }, this.args.lintTranslation.messages[0].text) as string; text = escapeHTML(text); text = text.replaceAll('(span data-underline)', ''); diff --git a/webapp/app/pods/components/loading-content/component.ts b/webapp/app/components/loading-content/component.ts similarity index 100% rename from webapp/app/pods/components/loading-content/component.ts rename to webapp/app/components/loading-content/component.ts diff --git a/webapp/app/pods/components/login-forms/component.ts b/webapp/app/components/login-forms/component.ts similarity index 93% rename from webapp/app/pods/components/login-forms/component.ts rename to webapp/app/components/login-forms/component.ts index 2e1c168c1..66f6fb1c4 100644 --- a/webapp/app/pods/components/login-forms/component.ts +++ b/webapp/app/components/login-forms/component.ts @@ -21,6 +21,7 @@ export default class LoginForms extends Component { discordUrl = `${config.API.AUTHENTICATION_PATH}/discord`; microsoftUrl = `${config.API.AUTHENTICATION_PATH}/microsoft`; auth0Url = `${config.API.AUTHENTICATION_PATH}/auth0`; + oidcUrl = `${config.API.AUTHENTICATION_PATH}/oidc`; get version() { return config.version === '__VERSION__' ? 'dev' : config.version; @@ -62,6 +63,10 @@ export default class LoginForms extends Component { return this.providerIds.includes('microsoft'); } + get oidcLoginEnabled() { + return this.providerIds.includes('oidc'); + } + get dummyUrl() { return `${config.API.AUTHENTICATION_PATH}/dummy/callback?email=${this.username}`; } diff --git a/webapp/app/pods/components/machine-translations-document-translate/component.ts b/webapp/app/components/machine-translations-document-translate/component.ts similarity index 92% rename from webapp/app/pods/components/machine-translations-document-translate/component.ts rename to webapp/app/components/machine-translations-document-translate/component.ts index 53851623f..ed860ff76 100644 --- a/webapp/app/pods/components/machine-translations-document-translate/component.ts +++ b/webapp/app/components/machine-translations-document-translate/component.ts @@ -3,11 +3,10 @@ import {inject as service} from '@ember/service'; import {action} from '@ember/object'; import {tracked} from '@glimmer/tracking'; import {htmlSafe} from '@ember/template'; -import {dropTask} from 'ember-concurrency-decorators'; +import {dropTask} from 'ember-concurrency'; import GlobalState from 'accent-webapp/services/global-state'; import LanguageSearcher from 'accent-webapp/services/language-searcher'; import FileSaver from 'accent-webapp/services/file-saver'; -import {taskFor} from 'ember-concurrency-ts'; import Exporter from 'accent-webapp/services/exporter'; interface Project { @@ -40,7 +39,7 @@ interface Args { fromLanguage: string, toLanguage: string, documentFormat: string - ) => void; + ) => Promise; } export default class MachineTranslationsTranslateUploadForm extends Component { @@ -79,7 +78,7 @@ export default class MachineTranslationsTranslateUploadForm extends Component ({ value: slug, - label: name, + label: name })); } @@ -133,7 +132,7 @@ export default class MachineTranslationsTranslateUploadForm extends Component { + await this.renderDocument(); if (this.sameLanguages) return; - yield this.args.onTranslate( + await this.args.onTranslate( this.fromRevision.value, this.toRevision.value, this.documentFormat.value ); - } + }); @action async renderDocument() { @@ -173,7 +171,7 @@ export default class MachineTranslationsTranslateUploadForm extends Component void; + onFileReset: () => Promise; onFileChange: ( file: File, fromLanguage: string, toLanguage: string, documentFormat: string - ) => void; + ) => Promise; } const preventDefault = (event: Event) => event.preventDefault(); @@ -60,7 +59,7 @@ export default class MachineTranslationsTranslateUploadForm extends Component ({ value: slug, - label: name, + label: name })); } @@ -99,11 +98,10 @@ export default class MachineTranslationsTranslateUploadForm extends Component { this.fileContent = null; - yield this.args.onFileReset(); + await this.args.onFileReset(); this.file = files[0]; @@ -120,7 +118,7 @@ export default class MachineTranslationsTranslateUploadForm extends Component { if (!this.file) return; - yield this.args.onFileChange( + await this.args.onFileChange( this.file, this.fromLanguage.value, this.toLanguage.value, this.documentFormat.value ); - } + }); @action exportFile() { if (!this.file || !this.args.translatedFileContent) return; const blob = new Blob([this.args.translatedFileContent as BlobPart], { - type: 'charset=utf-8', + type: 'charset=utf-8' }); this.fileSaver.saveAs(blob, this.file.name); diff --git a/webapp/app/pods/components/operations-peek/component.ts b/webapp/app/components/operations-peek/component.ts similarity index 96% rename from webapp/app/pods/components/operations-peek/component.ts rename to webapp/app/components/operations-peek/component.ts index 2cea413c7..2492f1b00 100644 --- a/webapp/app/pods/components/operations-peek/component.ts +++ b/webapp/app/components/operations-peek/component.ts @@ -14,7 +14,7 @@ export default class RevisionOperations extends Component { return this.args.revisionOperations.map((revisionOperation: any) => { return { label: revisionOperation.language.name, - value: revisionOperation.language.id, + value: revisionOperation.language.id }; }); } diff --git a/webapp/app/pods/components/operations-peek/item/component.ts b/webapp/app/components/operations-peek/item/component.ts similarity index 100% rename from webapp/app/pods/components/operations-peek/item/component.ts rename to webapp/app/components/operations-peek/item/component.ts diff --git a/webapp/app/pods/components/phoenix-channel-listener/component.ts b/webapp/app/components/phoenix-channel-listener/component.ts similarity index 100% rename from webapp/app/pods/components/phoenix-channel-listener/component.ts rename to webapp/app/components/phoenix-channel-listener/component.ts diff --git a/webapp/app/pods/components/project-activities-filter/component.ts b/webapp/app/components/project-activities-filter/component.ts similarity index 94% rename from webapp/app/pods/components/project-activities-filter/component.ts rename to webapp/app/components/project-activities-filter/component.ts index c3092a052..fc6138b49 100644 --- a/webapp/app/pods/components/project-activities-filter/component.ts +++ b/webapp/app/components/project-activities-filter/component.ts @@ -37,7 +37,7 @@ export default class ProjectActivitiesFilter extends Component { 'conflict_on_proposed', 'conflict_on_corrected', 'conflict_on_slave', - 'document_delete', + 'document_delete' ]; get mappedActions() { @@ -45,7 +45,7 @@ export default class ProjectActivitiesFilter extends Component { (key) => { return { value: key, - label: this.intl.t(`${ACTIONS_PREFIX}${key}`), + label: this.intl.t(`${ACTIONS_PREFIX}${key}`) }; } ); @@ -54,7 +54,7 @@ export default class ProjectActivitiesFilter extends Component { label: this.intl.t( 'components.project_activities_filter.actions_default_option_text' ), - value: '', + value: '' }); return actions; @@ -67,14 +67,14 @@ export default class ProjectActivitiesFilter extends Component { .filter((collaborator: any) => !collaborator.isPending) .map(({user: {fullname, id}}: {user: any}) => ({ label: fullname, - value: id, + value: id })); users.unshift({ label: this.intl.t( 'components.project_activities_filter.collaborators_default_option_text' ), - value: '', + value: '' }); return users; @@ -86,7 +86,7 @@ export default class ProjectActivitiesFilter extends Component { const versions = this.args.versions.map( ({tag, id}: {tag: string; id: string}) => ({ label: tag, - value: id, + value: id }) ); @@ -94,7 +94,7 @@ export default class ProjectActivitiesFilter extends Component { label: this.intl.t( 'components.project_activities_filter.versions_default_option_text' ), - value: '', + value: '' }); return versions; diff --git a/webapp/app/pods/components/project-activities-list/component.ts b/webapp/app/components/project-activities-list/component.ts similarity index 100% rename from webapp/app/pods/components/project-activities-list/component.ts rename to webapp/app/components/project-activities-list/component.ts diff --git a/webapp/app/pods/components/project-activity/component.ts b/webapp/app/components/project-activity/component.ts similarity index 98% rename from webapp/app/pods/components/project-activity/component.ts rename to webapp/app/components/project-activity/component.ts index 2678f7cf6..f7216c576 100644 --- a/webapp/app/pods/components/project-activity/component.ts +++ b/webapp/app/components/project-activity/component.ts @@ -23,7 +23,7 @@ const ROLLBACKABLE_ACTIONS = [ 'conflict_on_corrected', 'conflict_on_proposed', 'merge_on_proposed', - 'merge_on_corrected', + 'merge_on_corrected' ]; interface Args { @@ -169,12 +169,12 @@ export default class ProjectActivity extends Component { const variables = { projectId: this.args.project.id, activityId: this.args.activity.id, - page, + page }; const {data} = await this.apollo.client.query({ query: activityActivitiesQuery, - variables, + variables }); const operations = data.viewer.project.activity.operations; diff --git a/webapp/app/pods/components/project-comments-list/component.ts b/webapp/app/components/project-comments-list/component.ts similarity index 95% rename from webapp/app/pods/components/project-comments-list/component.ts rename to webapp/app/components/project-comments-list/component.ts index 3c5739a1b..7123b4664 100644 --- a/webapp/app/pods/components/project-comments-list/component.ts +++ b/webapp/app/components/project-comments-list/component.ts @@ -34,7 +34,7 @@ export default class ProjectCommentsList extends Component { return Object.keys(this.commentsByTranslationId).map((translationId) => { return { items: this.commentsByTranslationId[translationId], - value: this.translationsById[translationId], + value: this.translationsById[translationId] }; }); } diff --git a/webapp/app/pods/components/project-comments-list/item/component.ts b/webapp/app/components/project-comments-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/project-comments-list/item/component.ts rename to webapp/app/components/project-comments-list/item/component.ts diff --git a/webapp/app/pods/components/project-create-form/component.ts b/webapp/app/components/project-create-form/component.ts similarity index 96% rename from webapp/app/pods/components/project-create-form/component.ts rename to webapp/app/components/project-create-form/component.ts index 02a6c1ccb..ffdcae552 100644 --- a/webapp/app/pods/components/project-create-form/component.ts +++ b/webapp/app/components/project-create-form/component.ts @@ -13,7 +13,7 @@ interface Args { languageId, name, mainColor, - logo, + logo }: { languageId: string; name: string; @@ -60,8 +60,8 @@ export default class ProjectCreateForm extends Component { } @action - logoPicked(selection: {emoji: string}) { - this.logo = selection.emoji; + logoPicked(selection: string) { + this.logo = selection; } @action diff --git a/webapp/app/pods/components/project-file-operation/component.ts b/webapp/app/components/project-file-operation/component.ts similarity index 100% rename from webapp/app/pods/components/project-file-operation/component.ts rename to webapp/app/components/project-file-operation/component.ts diff --git a/webapp/app/pods/components/project-header/component.ts b/webapp/app/components/project-header/component.ts similarity index 81% rename from webapp/app/pods/components/project-header/component.ts rename to webapp/app/components/project-header/component.ts index e972db1a2..774eae29b 100644 --- a/webapp/app/pods/components/project-header/component.ts +++ b/webapp/app/components/project-header/component.ts @@ -5,9 +5,7 @@ import Session from 'accent-webapp/services/session'; import RouterService from '@ember/routing/router-service'; import GlobalState from 'accent-webapp/services/global-state'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 500; // ms @@ -45,28 +43,27 @@ export default class ProjectsHeader extends Component { return this.args.project.revisions[0].id; } - @restartableTask - *debounceQuery(query: string) { + debounceQuery = restartableTask(async (query: string) => { this.debouncedQuery = query; if (!this.debouncedQuery) return; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.router.transitionTo( 'logged-in.project.revision.translations', this.args.project.id, this.selectedRevision, { - queryParams: {query}, + queryParams: {query} } ); - } + }); @action setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQuery.perform(target.value); } @action diff --git a/webapp/app/pods/components/project-logo/component.ts b/webapp/app/components/project-logo/component.ts similarity index 100% rename from webapp/app/pods/components/project-logo/component.ts rename to webapp/app/components/project-logo/component.ts diff --git a/webapp/app/pods/components/project-navigation/component.ts b/webapp/app/components/project-navigation/component.ts similarity index 100% rename from webapp/app/pods/components/project-navigation/component.ts rename to webapp/app/components/project-navigation/component.ts diff --git a/webapp/app/pods/components/project-navigation/list/component.ts b/webapp/app/components/project-navigation/list/component.ts similarity index 100% rename from webapp/app/pods/components/project-navigation/list/component.ts rename to webapp/app/components/project-navigation/list/component.ts diff --git a/webapp/app/pods/components/project-settings/api-token/component.ts b/webapp/app/components/project-settings/api-token/component.ts similarity index 89% rename from webapp/app/pods/components/project-settings/api-token/component.ts rename to webapp/app/components/project-settings/api-token/component.ts index 3853d2a85..163a7196b 100644 --- a/webapp/app/pods/components/project-settings/api-token/component.ts +++ b/webapp/app/components/project-settings/api-token/component.ts @@ -3,7 +3,6 @@ import {inject as service} from '@ember/service'; import {action} from '@ember/object'; import {tracked} from '@glimmer/tracking'; import {dropTask} from 'ember-concurrency'; -import {taskFor} from 'ember-concurrency-ts'; import IntlService from 'ember-intl/services/intl'; import {CreateApiTokenResponse} from 'accent-webapp/queries/create-api-token'; @@ -14,7 +13,7 @@ interface Args { name: string; pictureUrl: string | null; permissions: string[]; - }) => void; + }) => Promise; onRevoke: (args: {id: string}) => void; } @@ -43,21 +42,20 @@ export default class APIToken extends Component { } get isSubmitting() { - return taskFor(this.submitTask).isRunning; + return this.submitTask.isRunning; } get isSubmitDisabled() { return !this.apiTokenName || this.apiTokenName.length === 0; } - @dropTask - *submitTask() { + submitTask = dropTask(async () => { if (!this.apiTokenName) return; - const response: CreateApiTokenResponse = yield this.args.onCreate({ + const response: CreateApiTokenResponse = await this.args.onCreate({ name: this.apiTokenName, pictureUrl: this.apiTokenPictureUrl, - permissions: this.apiTokenPermissions, + permissions: this.apiTokenPermissions }); if (response.apiToken) { @@ -66,7 +64,7 @@ export default class APIToken extends Component { this.showPermissionsInput = false; this.unselectAllPermissions(); } - } + }); @action changePermission() { diff --git a/webapp/app/pods/components/project-settings/api-token/item/component.ts b/webapp/app/components/project-settings/api-token/item/component.ts similarity index 77% rename from webapp/app/pods/components/project-settings/api-token/item/component.ts rename to webapp/app/components/project-settings/api-token/item/component.ts index c1cd0e41d..18743dae4 100644 --- a/webapp/app/pods/components/project-settings/api-token/item/component.ts +++ b/webapp/app/components/project-settings/api-token/item/component.ts @@ -3,12 +3,11 @@ import {inject as service} from '@ember/service'; import {action} from '@ember/object'; import {tracked} from '@glimmer/tracking'; import {dropTask} from 'ember-concurrency'; -import {taskFor} from 'ember-concurrency-ts'; import IntlService from 'ember-intl/services/intl'; interface Args { token: any; - onRevoke: (args: {id: string}) => void; + onRevoke: (args: {id: string}) => Promise; } export default class APITokenItem extends Component { @@ -19,11 +18,10 @@ export default class APITokenItem extends Component { showPermissions = false; get isRevoking() { - return taskFor(this.revokeTask).isRunning; + return this.revokeTask.isRunning; } - @dropTask - *revokeTask() { + revokeTask = dropTask(async () => { const message = this.intl.t( 'components.project_settings.api_token.revoke_confirm' ); @@ -33,8 +31,8 @@ export default class APITokenItem extends Component { return; } - yield this.args.onRevoke(this.args.token); - } + await this.args.onRevoke(this.args.token); + }); @action togglePermissions() { diff --git a/webapp/app/pods/components/project-settings/badges/component.ts b/webapp/app/components/project-settings/badges/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/badges/component.ts rename to webapp/app/components/project-settings/badges/component.ts diff --git a/webapp/app/pods/components/project-settings/collaborators/component.ts b/webapp/app/components/project-settings/collaborators/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/component.ts rename to webapp/app/components/project-settings/collaborators/component.ts diff --git a/webapp/app/pods/components/project-settings/collaborators/create-form/component.ts b/webapp/app/components/project-settings/collaborators/create-form/component.ts similarity index 89% rename from webapp/app/pods/components/project-settings/collaborators/create-form/component.ts rename to webapp/app/components/project-settings/collaborators/create-form/component.ts index 27ccd8397..e3b4bd0ff 100644 --- a/webapp/app/pods/components/project-settings/collaborators/create-form/component.ts +++ b/webapp/app/components/project-settings/collaborators/create-form/component.ts @@ -4,7 +4,7 @@ import Component from '@glimmer/component'; import IntlService from 'ember-intl/services/intl'; import GlobalState from 'accent-webapp/services/global-state'; import {tracked} from '@glimmer/tracking'; -import {dropTask} from 'ember-concurrency-decorators'; +import {dropTask} from 'ember-concurrency'; interface Args { project: any; @@ -39,7 +39,7 @@ export default class CreateForm extends Component { get mappedPossibleRoles() { return this.possibleRoles.map((value) => ({ label: this.intl.t(`general.roles.${value}`), - value, + value })); } @@ -47,17 +47,16 @@ export default class CreateForm extends Component { return this.mappedPossibleRoles.find(({value}) => value === this.role); } - @dropTask - *submitTask() { + submitTask = dropTask(async () => { if (!this.email || !this.role) return; this.isCreating = true; - yield this.args.onCreate({email: this.email, role: this.role}); + await this.args.onCreate({email: this.email, role: this.role}); this.email = ''; this.isCreating = false; - } + }); @action setRole({value}: {value: string}) { diff --git a/webapp/app/pods/components/project-settings/collaborators/list/component.ts b/webapp/app/components/project-settings/collaborators/list/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/list/component.ts rename to webapp/app/components/project-settings/collaborators/list/component.ts diff --git a/webapp/app/pods/components/project-settings/collaborators/list/item/component.ts b/webapp/app/components/project-settings/collaborators/list/item/component.ts similarity index 99% rename from webapp/app/pods/components/project-settings/collaborators/list/item/component.ts rename to webapp/app/components/project-settings/collaborators/list/item/component.ts index 2d07fed8d..02de405b2 100644 --- a/webapp/app/pods/components/project-settings/collaborators/list/item/component.ts +++ b/webapp/app/components/project-settings/collaborators/list/item/component.ts @@ -40,7 +40,7 @@ export default class CollaboratorsListItem extends Component { get mappedPossibleRoles() { return this.possibleRoles.map((value) => ({ label: this.intl.t(`general.roles.${value}`), - value, + value })); } diff --git a/webapp/app/pods/components/project-settings/delete-form/component.ts b/webapp/app/components/project-settings/delete-form/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/delete-form/component.ts rename to webapp/app/components/project-settings/delete-form/component.ts diff --git a/webapp/app/pods/components/project-settings/form/component.ts b/webapp/app/components/project-settings/form/component.ts similarity index 94% rename from webapp/app/pods/components/project-settings/form/component.ts rename to webapp/app/components/project-settings/form/component.ts index d821e5f95..66fb16c93 100644 --- a/webapp/app/pods/components/project-settings/form/component.ts +++ b/webapp/app/components/project-settings/form/component.ts @@ -13,7 +13,7 @@ interface Args { isFileOperationsLocked, name, mainColor, - logo, + logo }: { isFileOperationsLocked: boolean; name: string; @@ -48,8 +48,8 @@ export default class ProjectSettingsForm extends Component { isFileOperationsLocked = this.args.project.isFileOperationsLocked; @action - logoPicked(selection: {emoji: string}) { - this.logo = selection.emoji; + logoPicked(selection: string) { + this.logo = selection; } @action @@ -61,7 +61,7 @@ export default class ProjectSettingsForm extends Component { isFileOperationsLocked: this.isFileOperationsLocked, name: this.name, mainColor: this.mainColor, - logo: this.logo, + logo: this.logo }); this.isUpdatingProject = false; @@ -75,7 +75,7 @@ export default class ProjectSettingsForm extends Component { isFileOperationsLocked: this.isFileOperationsLocked, name: this.name, mainColor: this.mainColor, - logo: this.logo, + logo: this.logo }); this.isUpdatingProject = false; diff --git a/webapp/app/pods/components/project-settings/integrations/component.ts b/webapp/app/components/project-settings/integrations/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/component.ts rename to webapp/app/components/project-settings/integrations/component.ts diff --git a/webapp/app/components/project-settings/integrations/form/aws-s3/component.ts b/webapp/app/components/project-settings/integrations/form/aws-s3/component.ts new file mode 100644 index 000000000..13e52a511 --- /dev/null +++ b/webapp/app/components/project-settings/integrations/form/aws-s3/component.ts @@ -0,0 +1,67 @@ +import Component from '@glimmer/component'; +import {action} from '@ember/object'; + +interface Args { + errors: any; + project: any; + bucket: string | null; + pathPrefix: string | null; + onChangeBucket: (value: string) => void; + onChangePathPrefix: (value: string) => void; + onChangeRegion: (value: string) => void; + onChangeAccessKeyId: (value: string) => void; + onChangeSecretAccessKey: (value: string) => void; +} + +export default class AwsS3 extends Component { + get policyContent() { + const bucket = this.args.bucket || '-'; + const pathPrefix = this.args.pathPrefix || '/'; + + return `{ + "Version": "2012-10-17", + "Statement": [ + { + "Effect": "Allow", + "Action": ["s3:PutObject"], + "Resource": "arn:aws:s3:::${bucket}${pathPrefix}*" + } + ] +}`; + } + + @action + changeBucket(event: Event) { + const target = event.target as HTMLInputElement; + + this.args.onChangeBucket(target.value); + } + + @action + changePathPrefix(event: Event) { + const target = event.target as HTMLInputElement; + + this.args.onChangePathPrefix(target.value); + } + + @action + changeRegion(event: Event) { + const target = event.target as HTMLInputElement; + + this.args.onChangeRegion(target.value); + } + + @action + changeAccessKeyId(event: Event) { + const target = event.target as HTMLInputElement; + + this.args.onChangeAccessKeyId(target.value); + } + + @action + changeSecretAccessKey(event: Event) { + const target = event.target as HTMLInputElement; + + this.args.onChangeSecretAccessKey(target.value); + } +} diff --git a/webapp/app/pods/components/project-settings/integrations/form/azure-storage-container/component.ts b/webapp/app/components/project-settings/integrations/form/azure-storage-container/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/form/azure-storage-container/component.ts rename to webapp/app/components/project-settings/integrations/form/azure-storage-container/component.ts diff --git a/webapp/app/pods/components/project-settings/integrations/form/component.ts b/webapp/app/components/project-settings/integrations/form/component.ts similarity index 65% rename from webapp/app/pods/components/project-settings/integrations/form/component.ts rename to webapp/app/components/project-settings/integrations/form/component.ts index c875a720e..0fc10a60c 100644 --- a/webapp/app/pods/components/project-settings/integrations/form/component.ts +++ b/webapp/app/components/project-settings/integrations/form/component.ts @@ -7,9 +7,10 @@ import {tracked} from '@glimmer/tracking'; const LOGOS = { AZURE_STORAGE_CONTAINER: 'assets/services/azure.svg', + AWS_S3: 'assets/services/aws-s3.svg', DISCORD: 'assets/services/discord.svg', GITHUB: 'assets/services/github.svg', - SLACK: 'assets/services/slack.svg', + SLACK: 'assets/services/slack.svg' }; interface Args { @@ -19,7 +20,7 @@ interface Args { service, events, integration, - data: {url, azureStorageContainerSas}, + data: {url, azureStorageContainerSas} }: { service: any; events: any; @@ -27,6 +28,11 @@ interface Args { data: { url: string; azureStorageContainerSas: string; + awsS3Bucket: string; + awsS3PathPrefix: string; + awsS3Region: string; + awsS3AccessKeyId: string; + awsS3SecretAccessKey: string; }; }) => Promise<{errors: any}>; onCancel: () => void; @@ -62,7 +68,22 @@ export default class IntegrationsForm extends Component { @tracked azureStorageContainerSasBaseUrl: string; - services = ['AZURE_STORAGE_CONTAINER', 'SLACK', 'DISCORD']; + @tracked + awsS3Bucket: string; + + @tracked + awsS3PathPrefix: string; + + @tracked + awsS3Region: string; + + @tracked + awsS3AccessKeyId: string; + + @tracked + awsS3SecretAccessKey: string; + + services = ['AWS_S3', 'AZURE_STORAGE_CONTAINER', 'SLACK', 'DISCORD']; @not('url') emptyUrl: boolean; @@ -81,7 +102,7 @@ export default class IntegrationsForm extends Component { return this.services.map((value) => { return { label: this.intl.t(`general.integration_services.${value}`), - value, + value }; }); } @@ -100,8 +121,8 @@ export default class IntegrationsForm extends Component { service: this.args.selectedServiceValue || this.services[0], events: [], data: { - url: this.url, - }, + url: this.url + } }; } @@ -109,6 +130,10 @@ export default class IntegrationsForm extends Component { this.url = this.integration.data.url; this.events = this.integration.events; this.azureStorageContainerSasBaseUrl = this.integration.data.sasBaseUrl; + this.awsS3Bucket = this.integration.data.bucket; + this.awsS3PathPrefix = this.integration.data.pathPrefix; + this.awsS3Region = this.integration.data.region; + this.awsS3AccessKeyId = this.integration.data.accessKeyId; } @action @@ -132,8 +157,33 @@ export default class IntegrationsForm extends Component { } @action - setAzureStorageContainerSas(sas: string) { - this.azureStorageContainerSas = sas; + setAzureStorageContainerSas(value: string) { + this.azureStorageContainerSas = value; + } + + @action + setAwsS3Bucket(value: string) { + this.awsS3Bucket = value; + } + + @action + setAwsS3PathPrefix(value: string) { + this.awsS3PathPrefix = value; + } + + @action + setAwsS3Region(value: string) { + this.awsS3Region = value; + } + + @action + setAwsS3AccessKeyId(value: string) { + this.awsS3AccessKeyId = value; + } + + @action + setAwsS3SecretAccessKey(value: string) { + this.awsS3SecretAccessKey = value; } @action @@ -147,7 +197,12 @@ export default class IntegrationsForm extends Component { data: { url: this.url, azureStorageContainerSas: this.azureStorageContainerSas, - }, + awsS3Bucket: this.awsS3Bucket, + awsS3PathPrefix: this.awsS3PathPrefix, + awsS3Region: this.awsS3Region, + awsS3AccessKeyId: this.awsS3AccessKeyId, + awsS3SecretAccessKey: this.awsS3SecretAccessKey + } }); this.isSubmiting = false; diff --git a/webapp/app/pods/components/project-settings/integrations/form/data-control-checkboxes/component.ts b/webapp/app/components/project-settings/integrations/form/data-control-checkboxes/component.ts similarity index 82% rename from webapp/app/pods/components/project-settings/integrations/form/data-control-checkboxes/component.ts rename to webapp/app/components/project-settings/integrations/form/data-control-checkboxes/component.ts index 569edd7b4..d34b08f5a 100644 --- a/webapp/app/pods/components/project-settings/integrations/form/data-control-checkboxes/component.ts +++ b/webapp/app/components/project-settings/integrations/form/data-control-checkboxes/component.ts @@ -17,18 +17,23 @@ export default class DataControlCheckboxes extends Component { allEvents = [ { value: 'SYNC', - label: 'components.project_settings.integrations.events.options.sync', + label: 'components.project_settings.integrations.events.options.sync' }, { value: 'NEW_CONFLICTS', label: - 'components.project_settings.integrations.events.options.new_conflicts', + 'components.project_settings.integrations.events.options.new_conflicts' }, { value: 'COMPLETE_REVIEW', label: - 'components.project_settings.integrations.events.options.complete_review', + 'components.project_settings.integrations.events.options.complete_review' }, + { + value: 'INTEGRATION_EXECUTE_AZURE_STORAGE_CONTAINER', + label: + 'components.project_settings.integrations.events.options.integration_execute_azure_storage_container' + } ]; @tracked diff --git a/webapp/app/pods/components/project-settings/integrations/form/discord/component.ts b/webapp/app/components/project-settings/integrations/form/discord/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/form/discord/component.ts rename to webapp/app/components/project-settings/integrations/form/discord/component.ts diff --git a/webapp/app/pods/components/project-settings/integrations/form/slack/component.ts b/webapp/app/components/project-settings/integrations/form/slack/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/form/slack/component.ts rename to webapp/app/components/project-settings/integrations/form/slack/component.ts diff --git a/webapp/app/pods/components/project-settings/integrations/list/component.ts b/webapp/app/components/project-settings/integrations/list/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/list/component.ts rename to webapp/app/components/project-settings/integrations/list/component.ts diff --git a/webapp/app/pods/components/project-settings/integrations/list/item/component.ts b/webapp/app/components/project-settings/integrations/list/item/component.ts similarity index 92% rename from webapp/app/pods/components/project-settings/integrations/list/item/component.ts rename to webapp/app/components/project-settings/integrations/list/item/component.ts index fc34b6beb..881737a77 100644 --- a/webapp/app/pods/components/project-settings/integrations/list/item/component.ts +++ b/webapp/app/components/project-settings/integrations/list/item/component.ts @@ -4,12 +4,13 @@ import {tracked} from '@glimmer/tracking'; const LOGOS = { AZURE_STORAGE_CONTAINER: 'assets/services/azure.svg', + AWS_S3: 'assets/services/aws-s3.svg', DISCORD: 'assets/services/discord.svg', GITHUB: 'assets/services/github.svg', - SLACK: 'assets/services/slack.svg', + SLACK: 'assets/services/slack.svg' }; -const EXECUTABLE_SERVICES = ['AZURE_STORAGE_CONTAINER']; +const EXECUTABLE_SERVICES = ['AZURE_STORAGE_CONTAINER', 'AWS_S3']; interface Args { project: any; diff --git a/webapp/app/components/project-settings/integrations/list/item/execute/aws-s3/component.ts b/webapp/app/components/project-settings/integrations/list/item/execute/aws-s3/component.ts new file mode 100644 index 000000000..141ff28f0 --- /dev/null +++ b/webapp/app/components/project-settings/integrations/list/item/execute/aws-s3/component.ts @@ -0,0 +1,110 @@ +import {inject as service} from '@ember/service'; +import {action} from '@ember/object'; +import {readOnly} from '@ember/object/computed'; +import {tracked} from '@glimmer/tracking'; +import Component from '@glimmer/component'; +import IntlService from 'ember-intl/services/intl'; +import FlashMessages from 'ember-cli-flash/services/flash-messages'; +import ApolloMutate from 'accent-webapp/services/apollo-mutate'; +import executeIntegration from 'accent-webapp/queries/execute-integration'; + +const FLASH_MESSAGE_CREATE_SUCCESS = + 'pods.project.edit.flash_messages.integration_execute_success'; +const FLASH_MESSAGE_CREATE_ERROR = + 'pods.project.edit.flash_messages.integration_execute_error'; + +interface Args { + close: () => void; + integration: { + id: string; + }; +} + +export default class IntegrationExecuteAwsS3 extends Component { + @service('intl') + intl: IntlService; + + @service('apollo-mutate') + apolloMutate: ApolloMutate; + + @service('flash-messages') + flashMessages: FlashMessages; + + @readOnly('model.projectModel.project') + project: any; + + @tracked + error = false; + + allTargetVersions = [ + { + value: 'LATEST', + label: + 'components.project_settings.integrations.execute.aws_s3.target_version.options.latest' + }, + { + value: 'SPECIFIC', + label: + 'components.project_settings.integrations.execute.aws_s3.target_version.options.specific' + } + ]; + + @tracked + targetVersion = this.allTargetVersions[0].value; + + @tracked + tag: string | null = null; + + @tracked + isSubmitting = false; + + @action + setTargetVersion(targetVersion: string) { + this.tag = null; + this.targetVersion = targetVersion; + } + + @action + setTag(event: Event) { + const target = event.target as HTMLInputElement; + + this.tag = target.value; + } + + @action + autofocus(input: HTMLInputElement) { + input.focus(); + } + + @action + async submit() { + const confirmMessage = this.intl.t( + 'components.project_settings.integrations.execute.aws_s3.submit_confirm' + ); + /* eslint-disable-next-line no-alert */ + if (!window.confirm(confirmMessage)) { + return; + } + + const response = await this.apolloMutate.mutate({ + mutation: executeIntegration, + refetchQueries: ['ProjectServiceIntegrations'], + variables: { + integrationId: this.args.integration.id, + azureStorageContainer: { + tag: this.tag, + targetVersion: this.targetVersion + } + } + }); + + if (response.errors) { + this.flashMessages.error(this.intl.t(FLASH_MESSAGE_CREATE_ERROR)); + } else { + this.args.close(); + this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CREATE_SUCCESS)); + } + + return response; + } +} diff --git a/webapp/app/pods/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts b/webapp/app/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts similarity index 93% rename from webapp/app/pods/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts rename to webapp/app/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts index 209facb94..90e4d05a2 100644 --- a/webapp/app/pods/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts +++ b/webapp/app/components/project-settings/integrations/list/item/execute/azure-storage-container/component.ts @@ -40,13 +40,13 @@ export default class IntegrationExecuteAzureStorageContainer extends Component void; + onDelete: () => Promise; onSave: ({ provider, enabledActions, usePlatform, - configKey, + configKey }: { provider: string; enabledActions: string[]; @@ -29,8 +28,7 @@ const PROVIDERS = ['google_translate', 'deepl']; /* eslint-disable camelcase */ const LOGOS = { deepl: 'assets/machine_translations_providers/deepl.svg', - google_translate: - 'assets/machine_translations_providers/google_translate.svg', + google_translate: 'assets/machine_translations_providers/google_translate.svg' }; export default class ProjectSettingsMachineTranslations extends Component { @@ -67,18 +65,18 @@ export default class ProjectSettingsMachineTranslations extends Component } get isSubmitting() { - return taskFor(this.submit).isRunning; + return this.submit.isRunning; } get isRemoving() { - return taskFor(this.remove).isRunning; + return this.remove.isRunning; } get mappedProviders() { return PROVIDERS.map((value) => { return { label: this.intl.t(`general.machine_translations_providers.${value}`), - value, + value }; }); } @@ -128,18 +126,16 @@ export default class ProjectSettingsMachineTranslations extends Component this.configKey = (event.target as HTMLInputElement).value; } - @dropTask - *submit() { - yield this.args.onSave({ + submit = dropTask(async () => { + await this.args.onSave({ provider: this.provider, enabledActions: this.enabledActions, usePlatform: this.usePlatform, - configKey: this.configKey, + configKey: this.configKey }); - } + }); - @dropTask - *remove() { - yield this.args.onDelete(); - } + remove = dropTask(async () => { + await this.args.onDelete(); + }); } diff --git a/webapp/app/pods/components/project-settings/manage-languages/component.ts b/webapp/app/components/project-settings/manage-languages/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/component.ts rename to webapp/app/components/project-settings/manage-languages/component.ts diff --git a/webapp/app/pods/components/project-settings/manage-languages/create-form/component.ts b/webapp/app/components/project-settings/manage-languages/create-form/component.ts similarity index 98% rename from webapp/app/pods/components/project-settings/manage-languages/create-form/component.ts rename to webapp/app/components/project-settings/manage-languages/create-form/component.ts index ff8328703..c8810c184 100644 --- a/webapp/app/pods/components/project-settings/manage-languages/create-form/component.ts +++ b/webapp/app/components/project-settings/manage-languages/create-form/component.ts @@ -63,7 +63,7 @@ export default class CreateForm extends Component { await this.args.onCreate(this.language, { machineTranslationsEnabled: this.machineTranslationsEnabled, - defaultNull: this.defaultNull, + defaultNull: this.defaultNull }); this.isLoading = false; diff --git a/webapp/app/pods/components/project-settings/manage-languages/overview/component.ts b/webapp/app/components/project-settings/manage-languages/overview/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/overview/component.ts rename to webapp/app/components/project-settings/manage-languages/overview/component.ts diff --git a/webapp/app/pods/components/project-settings/manage-languages/overview/item/component.ts b/webapp/app/components/project-settings/manage-languages/overview/item/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/overview/item/component.ts rename to webapp/app/components/project-settings/manage-languages/overview/item/component.ts diff --git a/webapp/app/pods/components/project-settings/prompts/config/component.ts b/webapp/app/components/project-settings/prompts/config/component.ts similarity index 66% rename from webapp/app/pods/components/project-settings/prompts/config/component.ts rename to webapp/app/components/project-settings/prompts/config/component.ts index 978e31477..29ce57791 100644 --- a/webapp/app/pods/components/project-settings/prompts/config/component.ts +++ b/webapp/app/components/project-settings/prompts/config/component.ts @@ -4,19 +4,20 @@ import Component from '@glimmer/component'; import {tracked} from '@glimmer/tracking'; import GlobalState from 'accent-webapp/services/global-state'; import FlashMessages from 'ember-cli-flash/services/flash-messages'; -import {dropTask} from 'ember-concurrency-decorators'; -import {taskFor} from 'ember-concurrency-ts'; +import {dropTask} from 'ember-concurrency'; import IntlService from 'ember-intl/services/intl'; interface Args { project: any; - onDelete: () => void; + onDelete: () => Promise; onSave: ({ provider, configKey, + usePlatform }: { provider: string; configKey: string | null; + usePlatform: boolean; }) => Promise; } @@ -24,10 +25,10 @@ const PROVIDERS = ['openai']; /* eslint-disable camelcase */ const LOGOS = { - openai: 'assets/prompts_providers/openai.svg', + openai: 'assets/prompts_providers/openai.svg' }; -export default class ProjectSettingsMachineTranslations extends Component { +export default class ProjectSettingsPromptsConfig extends Component { @service('global-state') globalState: GlobalState; @@ -38,10 +39,10 @@ export default class ProjectSettingsMachineTranslations extends Component intl: IntlService; @tracked - provider = this.args.project.promptsConfig?.provider || 'openai'; + provider = this.args.project.promptConfig?.provider || 'openai'; @tracked - usePlatform = this.args.project.promptsConfig?.usePlatform || false; + usePlatform = this.args.project.promptConfig?.usePlatform || false; @tracked configKey: any; @@ -51,18 +52,18 @@ export default class ProjectSettingsMachineTranslations extends Component } get isSubmitting() { - return taskFor(this.submit).isRunning; + return this.submit.isRunning; } get isRemoving() { - return taskFor(this.remove).isRunning; + return this.remove.isRunning; } get mappedProviders() { return PROVIDERS.map((value) => { return { label: this.intl.t(`general.prompts_providers.${value}`), - value, + value }; }); } @@ -87,16 +88,23 @@ export default class ProjectSettingsMachineTranslations extends Component this.configKey = (event.target as HTMLInputElement).value; } - @dropTask - *submit() { - yield this.args.onSave({ + @action + onUsePlatformChange(event: InputEvent) { + const checked = (event.target as HTMLInputElement).checked; + + if (checked) this.configKey = null; + this.usePlatform = checked; + } + + submit = dropTask(async () => { + await this.args.onSave({ provider: this.provider, configKey: this.configKey, + usePlatform: this.usePlatform }); - } + }); - @dropTask - *remove() { - yield this.args.onDelete(); - } + remove = dropTask(async () => { + await this.args.onDelete(); + }); } diff --git a/webapp/app/pods/components/project-settings/prompts/item/component.ts b/webapp/app/components/project-settings/prompts/item/component.ts similarity index 73% rename from webapp/app/pods/components/project-settings/prompts/item/component.ts rename to webapp/app/components/project-settings/prompts/item/component.ts index 8008b6f90..6ee8d13b4 100644 --- a/webapp/app/pods/components/project-settings/prompts/item/component.ts +++ b/webapp/app/components/project-settings/prompts/item/component.ts @@ -1,20 +1,19 @@ import Component from '@glimmer/component'; -import {dropTask} from 'ember-concurrency-decorators'; +import {dropTask} from 'ember-concurrency'; import IntlService from 'ember-intl/services/intl'; import {inject as service} from '@ember/service'; interface Args { project: any; prompt: any; - onDelete: (id: string) => void; + onDelete: (id: string) => Promise; } export default class ProjectSettingsPromptsItem extends Component { @service('intl') intl: IntlService; - @dropTask - *deletePrompt() { + deletePrompt = dropTask(async () => { const message = this.intl.t( 'components.project_settings.prompts.delete_confirm' ); @@ -24,6 +23,6 @@ export default class ProjectSettingsPromptsItem extends Component { return; } - yield this.args.onDelete(this.args.prompt.id); - } + await this.args.onDelete(this.args.prompt.id); + }); } diff --git a/webapp/app/pods/components/project-settings/title/component.ts b/webapp/app/components/project-settings/title/component.ts similarity index 100% rename from webapp/app/pods/components/project-settings/title/component.ts rename to webapp/app/components/project-settings/title/component.ts diff --git a/webapp/app/pods/components/projects-filters/component.ts b/webapp/app/components/projects-filters/component.ts similarity index 73% rename from webapp/app/pods/components/projects-filters/component.ts rename to webapp/app/components/projects-filters/component.ts index d550e5b40..9558c4254 100644 --- a/webapp/app/pods/components/projects-filters/component.ts +++ b/webapp/app/components/projects-filters/component.ts @@ -2,9 +2,7 @@ import {action} from '@ember/object'; import {inject as service} from '@ember/service'; import Component from '@glimmer/component'; import Session from 'accent-webapp/services/session'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; import {tracked} from '@glimmer/tracking'; const DEBOUNCE_OFFSET = 1000; // ms @@ -22,20 +20,19 @@ export default class ProjectsFilter extends Component { @tracked debouncedQuery: string = this.args.query; - @restartableTask - *debounceQuery(query: string) { + debounceQueryTask = restartableTask(async (query: string) => { this.debouncedQuery = query; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.args.onChangeQuery(query); - } + }); @action setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQueryTask.perform(target.value); } @action diff --git a/webapp/app/pods/components/projects-header/component.ts b/webapp/app/components/projects-header/component.ts similarity index 81% rename from webapp/app/pods/components/projects-header/component.ts rename to webapp/app/components/projects-header/component.ts index e972db1a2..774eae29b 100644 --- a/webapp/app/pods/components/projects-header/component.ts +++ b/webapp/app/components/projects-header/component.ts @@ -5,9 +5,7 @@ import Session from 'accent-webapp/services/session'; import RouterService from '@ember/routing/router-service'; import GlobalState from 'accent-webapp/services/global-state'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 500; // ms @@ -45,28 +43,27 @@ export default class ProjectsHeader extends Component { return this.args.project.revisions[0].id; } - @restartableTask - *debounceQuery(query: string) { + debounceQuery = restartableTask(async (query: string) => { this.debouncedQuery = query; if (!this.debouncedQuery) return; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.router.transitionTo( 'logged-in.project.revision.translations', this.args.project.id, this.selectedRevision, { - queryParams: {query}, + queryParams: {query} } ); - } + }); @action setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQuery.perform(target.value); } @action diff --git a/webapp/app/pods/components/projects-list/component.ts b/webapp/app/components/projects-list/component.ts similarity index 100% rename from webapp/app/pods/components/projects-list/component.ts rename to webapp/app/components/projects-list/component.ts diff --git a/webapp/app/components/projects-list/item/component.ts b/webapp/app/components/projects-list/item/component.ts new file mode 100644 index 000000000..d8ac6a656 --- /dev/null +++ b/webapp/app/components/projects-list/item/component.ts @@ -0,0 +1,11 @@ +import Component from '@glimmer/component'; + +interface Args { + project: any; +} + +export default class ProjectsListItem extends Component { + get colorPrimary() { + return `--color-primary: ${this.args.project.mainColor}`; + } +} diff --git a/webapp/app/pods/components/prompt-create-form/component.ts b/webapp/app/components/prompt-create-form/component.ts similarity index 92% rename from webapp/app/pods/components/prompt-create-form/component.ts rename to webapp/app/components/prompt-create-form/component.ts index 2250281fe..312f59015 100644 --- a/webapp/app/pods/components/prompt-create-form/component.ts +++ b/webapp/app/components/prompt-create-form/component.ts @@ -8,7 +8,7 @@ interface Args { onCreate: ({ content, name, - quickAccess, + quickAccess }: { content: string; name: string; @@ -57,8 +57,8 @@ export default class PromptCreateForm extends Component { } @action - setQuickAccess(selection: {emoji: string}) { - this.quickAccess = selection.emoji; + setQuickAccess(selection: string) { + this.quickAccess = selection; } @action diff --git a/webapp/app/pods/components/prompt-update-form/component.ts b/webapp/app/components/prompt-update-form/component.ts similarity index 92% rename from webapp/app/pods/components/prompt-update-form/component.ts rename to webapp/app/components/prompt-update-form/component.ts index 29f28fbe7..a9f41ad9b 100644 --- a/webapp/app/pods/components/prompt-update-form/component.ts +++ b/webapp/app/components/prompt-update-form/component.ts @@ -9,7 +9,7 @@ interface Args { onUpdate: ({ content, name, - quickAccess, + quickAccess }: { content: string; name: string; @@ -58,8 +58,8 @@ export default class PromptUpdateForm extends Component { } @action - setQuickAccess(selection: {emoji: string}) { - this.quickAccess = selection.emoji; + setQuickAccess(selection: string) { + this.quickAccess = selection; } @action diff --git a/webapp/app/pods/components/recent-project-cache/component.ts b/webapp/app/components/recent-project-cache/component.ts similarity index 70% rename from webapp/app/pods/components/recent-project-cache/component.ts rename to webapp/app/components/recent-project-cache/component.ts index 64ae2a255..67a0a28b2 100644 --- a/webapp/app/pods/components/recent-project-cache/component.ts +++ b/webapp/app/components/recent-project-cache/component.ts @@ -1,7 +1,6 @@ import Component from '@glimmer/component'; import {inject as service} from '@ember/service'; -import {dropTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; +import {timeout, dropTask} from 'ember-concurrency'; import RecentProjects from 'accent-webapp/services/recent-projects'; @@ -15,9 +14,8 @@ export default class RecentProjectCache extends Component { @service('recent-projects') recentProjects: RecentProjects; - @dropTask - *persistRecentProject() { - yield timeout(DEBOUNCE_ADD); + persistRecentProject = dropTask(async () => { + await timeout(DEBOUNCE_ADD); this.recentProjects.add(this.args.project.id); - } + }); } diff --git a/webapp/app/pods/components/recent-projects-list/component.ts b/webapp/app/components/recent-projects-list/component.ts similarity index 100% rename from webapp/app/pods/components/recent-projects-list/component.ts rename to webapp/app/components/recent-projects-list/component.ts diff --git a/webapp/app/pods/components/recent-projects-list/item/component.ts b/webapp/app/components/recent-projects-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/recent-projects-list/item/component.ts rename to webapp/app/components/recent-projects-list/item/component.ts diff --git a/webapp/app/pods/components/related-translations-list/component.ts b/webapp/app/components/related-translations-list/component.ts similarity index 100% rename from webapp/app/pods/components/related-translations-list/component.ts rename to webapp/app/components/related-translations-list/component.ts diff --git a/webapp/app/pods/components/related-translations-list/item/component.ts b/webapp/app/components/related-translations-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/related-translations-list/item/component.ts rename to webapp/app/components/related-translations-list/item/component.ts diff --git a/webapp/app/pods/components/removed-translation-edit/component.ts b/webapp/app/components/removed-translation-edit/component.ts similarity index 100% rename from webapp/app/pods/components/removed-translation-edit/component.ts rename to webapp/app/components/removed-translation-edit/component.ts diff --git a/webapp/app/pods/components/resource-pagination/component.ts b/webapp/app/components/resource-pagination/component.ts similarity index 100% rename from webapp/app/pods/components/resource-pagination/component.ts rename to webapp/app/components/resource-pagination/component.ts diff --git a/webapp/app/pods/components/review-progress-bar/component.ts b/webapp/app/components/review-progress-bar/component.ts similarity index 100% rename from webapp/app/pods/components/review-progress-bar/component.ts rename to webapp/app/components/review-progress-bar/component.ts diff --git a/webapp/app/pods/components/revision-export-options/advanced-filters/component.ts b/webapp/app/components/revision-export-options/advanced-filters/component.ts similarity index 100% rename from webapp/app/pods/components/revision-export-options/advanced-filters/component.ts rename to webapp/app/components/revision-export-options/advanced-filters/component.ts diff --git a/webapp/app/pods/components/revision-export-options/component.ts b/webapp/app/components/revision-export-options/component.ts similarity index 92% rename from webapp/app/pods/components/revision-export-options/component.ts rename to webapp/app/components/revision-export-options/component.ts index 2f7e3b80f..18c392d92 100644 --- a/webapp/app/pods/components/revision-export-options/component.ts +++ b/webapp/app/components/revision-export-options/component.ts @@ -52,14 +52,12 @@ export default class RevisionExportOptions extends Component { return [ { value: '', - label: this.intl.t( - 'components.revision_export_options.orders.original' - ), + label: this.intl.t('components.revision_export_options.orders.original') }, { value: 'key', - label: this.intl.t('components.revision_export_options.orders.az'), - }, + label: this.intl.t('components.revision_export_options.orders.az') + } ]; } @@ -74,7 +72,7 @@ export default class RevisionExportOptions extends Component { return this.globalState.documentFormats.map(({slug, name}) => ({ value: slug, - label: name, + label: name })); } @@ -82,9 +80,9 @@ export default class RevisionExportOptions extends Component { return [ { value: '', - label: this.intl.t('components.revision_export_options.default_format'), + label: this.intl.t('components.revision_export_options.default_format') }, - ...this.formattedDocumentFormats, + ...this.formattedDocumentFormats ]; } @@ -103,14 +101,14 @@ export default class RevisionExportOptions extends Component { ({ id, name, - language, + language }: { id: string; name: string | null; language: any; }) => ({ label: name || language.name, - value: id, + value: id }) ); } @@ -129,7 +127,7 @@ export default class RevisionExportOptions extends Component { return this.args.documents.map( ({id, path}: {id: string; path: string}) => ({ label: path, - value: id, + value: id }) ); } @@ -148,16 +146,16 @@ export default class RevisionExportOptions extends Component { memo.concat([ { label: tag, - value: tag, - }, + value: tag + } ]), [ { label: this.intl.t( 'components.revision_export_options.default_version' ), - value: '', - }, + value: '' + } ] ); } diff --git a/webapp/app/pods/components/revision-selector/component.ts b/webapp/app/components/revision-selector/component.ts similarity index 100% rename from webapp/app/pods/components/revision-selector/component.ts rename to webapp/app/components/revision-selector/component.ts diff --git a/webapp/app/pods/components/revision-update-form/component.ts b/webapp/app/components/revision-update-form/component.ts similarity index 100% rename from webapp/app/pods/components/revision-update-form/component.ts rename to webapp/app/components/revision-update-form/component.ts diff --git a/webapp/app/pods/components/skeleton-ui/content/component.ts b/webapp/app/components/skeleton-ui/content/component.ts similarity index 100% rename from webapp/app/pods/components/skeleton-ui/content/component.ts rename to webapp/app/components/skeleton-ui/content/component.ts diff --git a/webapp/app/pods/components/skeleton-ui/progress-line/component.ts b/webapp/app/components/skeleton-ui/progress-line/component.ts similarity index 100% rename from webapp/app/pods/components/skeleton-ui/progress-line/component.ts rename to webapp/app/components/skeleton-ui/progress-line/component.ts diff --git a/webapp/app/pods/components/skeleton-ui/project-activities-filter/component.ts b/webapp/app/components/skeleton-ui/project-activities-filter/component.ts similarity index 100% rename from webapp/app/pods/components/skeleton-ui/project-activities-filter/component.ts rename to webapp/app/components/skeleton-ui/project-activities-filter/component.ts diff --git a/webapp/app/pods/components/time-ago-in-words-tag/component.ts b/webapp/app/components/time-ago-in-words-tag/component.ts similarity index 95% rename from webapp/app/pods/components/time-ago-in-words-tag/component.ts rename to webapp/app/components/time-ago-in-words-tag/component.ts index bc4d35e1b..383a99721 100644 --- a/webapp/app/pods/components/time-ago-in-words-tag/component.ts +++ b/webapp/app/components/time-ago-in-words-tag/component.ts @@ -1,6 +1,6 @@ import {inject as service} from '@ember/service'; import Component from '@glimmer/component'; -import dateFormat from 'date-fns/format'; +import {format as dateFormat} from 'date-fns'; import IntlService from 'ember-intl/services/intl'; interface Args { diff --git a/webapp/app/pods/components/translation-activities-list/component.ts b/webapp/app/components/translation-activities-list/component.ts similarity index 100% rename from webapp/app/pods/components/translation-activities-list/component.ts rename to webapp/app/components/translation-activities-list/component.ts diff --git a/webapp/app/pods/components/translation-comment-delete/component.ts b/webapp/app/components/translation-comment-delete/component.ts similarity index 100% rename from webapp/app/pods/components/translation-comment-delete/component.ts rename to webapp/app/components/translation-comment-delete/component.ts diff --git a/webapp/app/pods/components/translation-comment-form/component.ts b/webapp/app/components/translation-comment-form/component.ts similarity index 69% rename from webapp/app/pods/components/translation-comment-form/component.ts rename to webapp/app/components/translation-comment-form/component.ts index 4a56d5bcd..00b12f9ad 100644 --- a/webapp/app/pods/components/translation-comment-form/component.ts +++ b/webapp/app/components/translation-comment-form/component.ts @@ -1,10 +1,8 @@ import Component from '@glimmer/component'; import {action} from '@ember/object'; import {tracked} from '@glimmer/tracking'; -import {dropTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; +import {timeout, dropTask} from 'ember-concurrency'; import {MutationResponse} from 'accent-webapp/services/apollo-mutate'; -import {taskFor} from 'ember-concurrency-ts'; interface Args { value?: string; @@ -21,23 +19,22 @@ export default class TranslationCommentForm extends Component { error = false; get isSubmitting() { - return taskFor(this.submitTask).isRunning; + return this.submitTask.isRunning; } - @dropTask - *submitTask(event?: Event): any { + submitTask = dropTask(async (event?: Event) => { this.error = false; event?.preventDefault(); - yield timeout(SUBMIT_DEBOUNCE); - const response = yield this.args.onSubmit(this.text); + await timeout(SUBMIT_DEBOUNCE); + const response = await this.args.onSubmit(this.text); if (response.errors) { this.error = true; } else { this.text = ''; } - } + }); @action setText(event: KeyboardEvent) { diff --git a/webapp/app/pods/components/translation-comments-list/component.ts b/webapp/app/components/translation-comments-list/component.ts similarity index 100% rename from webapp/app/pods/components/translation-comments-list/component.ts rename to webapp/app/components/translation-comments-list/component.ts diff --git a/webapp/app/pods/components/translation-comments-list/item/component.ts b/webapp/app/components/translation-comments-list/item/component.ts similarity index 82% rename from webapp/app/pods/components/translation-comments-list/item/component.ts rename to webapp/app/components/translation-comments-list/item/component.ts index 602671de9..b4be84fb7 100644 --- a/webapp/app/pods/components/translation-comments-list/item/component.ts +++ b/webapp/app/components/translation-comments-list/item/component.ts @@ -5,13 +5,13 @@ import Component from '@glimmer/component'; import MarkdownIt from 'markdown-it'; import {htmlSafe} from '@ember/template'; import Session from 'accent-webapp/services/session'; -import {dropTask} from 'ember-concurrency-decorators'; +import {dropTask} from 'ember-concurrency'; import {tracked} from '@glimmer/tracking'; const markdown = MarkdownIt({ html: false, linkify: true, - typographer: true, + typographer: true }); interface Args { @@ -57,15 +57,13 @@ export default class TranslationsCommentsListItem extends Component { element.querySelector('textarea')?.focus(); } - @dropTask - *deleteComment() { - yield this.args.onDeleteComment(this.args.comment); - } + deleteComment = dropTask(async () => { + await this.args.onDeleteComment(this.args.comment); + }); - @dropTask - *updateComment(text: string) { - yield this.args.onUpdateComment({...this.args.comment, text}); + updateComment = dropTask(async (text: string) => { + await this.args.onUpdateComment({...this.args.comment, text}); this.editComment = false; - } + }); } diff --git a/webapp/app/pods/components/translation-comments-subscriptions/component.ts b/webapp/app/components/translation-comments-subscriptions/component.ts similarity index 100% rename from webapp/app/pods/components/translation-comments-subscriptions/component.ts rename to webapp/app/components/translation-comments-subscriptions/component.ts diff --git a/webapp/app/pods/components/translation-comments-subscriptions/item/component.ts b/webapp/app/components/translation-comments-subscriptions/item/component.ts similarity index 100% rename from webapp/app/pods/components/translation-comments-subscriptions/item/component.ts rename to webapp/app/components/translation-comments-subscriptions/item/component.ts diff --git a/webapp/app/pods/components/translation-conversation/component.ts b/webapp/app/components/translation-conversation/component.ts similarity index 100% rename from webapp/app/pods/components/translation-conversation/component.ts rename to webapp/app/components/translation-conversation/component.ts diff --git a/webapp/app/pods/components/translation-edit/component.ts b/webapp/app/components/translation-edit/component.ts similarity index 96% rename from webapp/app/pods/components/translation-edit/component.ts rename to webapp/app/components/translation-edit/component.ts index bd6b16be2..6ec254f14 100644 --- a/webapp/app/pods/components/translation-edit/component.ts +++ b/webapp/app/components/translation-edit/component.ts @@ -78,9 +78,10 @@ export default class TranslationEdit extends Component { } @action - onUpdateText(value: string) { - this.text = value; + onUpdateText(text: string) { + this.text = text; this.inputDisabled = false; + this.args.onChangeText?.(text); } @action diff --git a/webapp/app/pods/components/translation-edit/form/component.ts b/webapp/app/components/translation-edit/form/component.ts similarity index 77% rename from webapp/app/pods/components/translation-edit/form/component.ts rename to webapp/app/components/translation-edit/form/component.ts index 1e5082bc3..d421d4ad4 100644 --- a/webapp/app/pods/components/translation-edit/form/component.ts +++ b/webapp/app/components/translation-edit/form/component.ts @@ -1,21 +1,20 @@ import {inject as service} from '@ember/service'; import {equal} from '@ember/object/computed'; +import {next} from '@ember/runloop'; import {action} from '@ember/object'; import Component from '@glimmer/component'; import translationLintQuery from 'accent-webapp/queries/lint-translation'; import Apollo from 'accent-webapp/services/apollo'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {perform} from 'ember-concurrency-ts'; -import {timeout, TaskGenerator} from 'ember-concurrency'; +import {timeout, restartableTask} from 'ember-concurrency'; import MarkdownIt from 'markdown-it'; import {htmlSafe} from '@ember/template'; const markdown = MarkdownIt({ html: false, linkify: true, - typographer: true, + typographer: true }); const DEBOUNCE_LINT_MESSAGES = 800; @@ -51,7 +50,7 @@ export default class TranslationEditForm extends Component { @tracked lintTranslation = { translation: {id: this.args.translationId, text: this.args.value}, - messages: this.args.lintMessages, + messages: this.args.lintMessages }; @tracked @@ -107,6 +106,21 @@ export default class TranslationEditForm extends Component { this.args.onKeyUp?.(value); } + @action + handleFocus() { + this.args.onFocus?.(); + } + + @action + handleBlur() { + next(this, () => this.args.onBlur?.()); + } + + @action + handleSubmit() { + this.args.onSubmit(); + } + @action changeText(event: Event) { const target = event.target as HTMLInputElement; @@ -118,32 +132,29 @@ export default class TranslationEditForm extends Component { this.args.onEscape?.(); } - @restartableTask - *onUpdateValue( - _element: HTMLElement, - [value]: string[] - ): TaskGenerator { - yield timeout(DEBOUNCE_LINT_MESSAGES); + onUpdateValue = restartableTask( + async (_element: HTMLElement, [value]: string[]) => { + await timeout(DEBOUNCE_LINT_MESSAGES); - yield perform(this.fetchLintMessagesTask, value); - } + await this.fetchLintMessagesTask.perform(value); + } + ); - @restartableTask - *fetchLintMessagesTask(value: string) { - const {data} = yield this.apollo.client.query({ + fetchLintMessagesTask = restartableTask(async (value: string) => { + const {data} = await this.apollo.client.query({ fetchPolicy: 'network-only', query: translationLintQuery, variables: { text: value, projectId: this.args.projectId, - translationId: this.args.translationId, - }, + translationId: this.args.translationId + } }); this.lintTranslation = Object.assign(this.lintTranslation, { - messages: data.viewer.project.translation.lintMessages, + messages: data.viewer.project.translation.lintMessages }); - } + }); @action replaceText(value: string) { diff --git a/webapp/app/pods/components/translation-edit/helpers/component.ts b/webapp/app/components/translation-edit/helpers/component.ts similarity index 77% rename from webapp/app/pods/components/translation-edit/helpers/component.ts rename to webapp/app/components/translation-edit/helpers/component.ts index a90cdcde4..5a4b38727 100644 --- a/webapp/app/pods/components/translation-edit/helpers/component.ts +++ b/webapp/app/components/translation-edit/helpers/component.ts @@ -6,9 +6,11 @@ interface Args { export default class TranslationEditHelpers extends Component { get machineTranslationLanguages() { + if (!this.args.revisions) return []; + return this.args.revisions.map((revision: any) => ({ name: revision.name || revision.language.name, - slug: revision.slug || revision.language.slug, + slug: revision.slug || revision.language.slug })); } } diff --git a/webapp/app/pods/components/translation-editions-list/component.ts b/webapp/app/components/translation-editions-list/component.ts similarity index 100% rename from webapp/app/pods/components/translation-editions-list/component.ts rename to webapp/app/components/translation-editions-list/component.ts diff --git a/webapp/app/pods/components/translation-editions-list/item/component.ts b/webapp/app/components/translation-editions-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/translation-editions-list/item/component.ts rename to webapp/app/components/translation-editions-list/item/component.ts diff --git a/webapp/app/pods/components/translation-navigation/component.ts b/webapp/app/components/translation-navigation/component.ts similarity index 100% rename from webapp/app/pods/components/translation-navigation/component.ts rename to webapp/app/components/translation-navigation/component.ts diff --git a/webapp/app/pods/components/translation-splash-title/component.ts b/webapp/app/components/translation-splash-title/component.ts similarity index 100% rename from webapp/app/pods/components/translation-splash-title/component.ts rename to webapp/app/components/translation-splash-title/component.ts diff --git a/webapp/app/pods/components/translations-filter/advanced-filters/component.ts b/webapp/app/components/translations-filter/advanced-filters/component.ts similarity index 100% rename from webapp/app/pods/components/translations-filter/advanced-filters/component.ts rename to webapp/app/components/translations-filter/advanced-filters/component.ts diff --git a/webapp/app/pods/components/translations-filter/component.ts b/webapp/app/components/translations-filter/component.ts similarity index 87% rename from webapp/app/pods/components/translations-filter/component.ts rename to webapp/app/components/translations-filter/component.ts index f50055ac3..fe95f82f9 100644 --- a/webapp/app/pods/components/translations-filter/component.ts +++ b/webapp/app/components/translations-filter/component.ts @@ -4,9 +4,7 @@ import {gt} from '@ember/object/computed'; import Component from '@glimmer/component'; import IntlService from 'ember-intl/services/intl'; import {tracked} from '@glimmer/tracking'; -import {restartableTask} from 'ember-concurrency-decorators'; -import {timeout} from 'ember-concurrency'; -import {perform} from 'ember-concurrency-ts'; +import {timeout, restartableTask} from 'ember-concurrency'; const DEBOUNCE_OFFSET = 1000; // ms @@ -25,6 +23,7 @@ interface Args { isCommentedOnFilter: boolean; isConflictedFilter: boolean; isTranslatedFilter: boolean; + jipt?: boolean; onChangeQuery: (query: string) => void; onChangeDocument: () => void; onChangeVersion: () => void; @@ -53,7 +52,7 @@ export default class TranslationsFilter extends Component { const documents = this.args.documents.map( ({id, path}: {id: string; path: string}) => ({ label: path, - value: id, + value: id }) ); @@ -61,7 +60,7 @@ export default class TranslationsFilter extends Component { label: this.intl.t( 'components.translations_filter.document_default_option_text' ), - value: '', + value: '' }); return documents; @@ -77,7 +76,7 @@ export default class TranslationsFilter extends Component { const versions = this.args.versions.map( ({id, tag}: {id: string; tag: string}) => ({ label: tag, - value: id, + value: id }) ); @@ -85,7 +84,7 @@ export default class TranslationsFilter extends Component { label: this.intl.t( 'components.translations_filter.version_default_option_text' ), - value: '', + value: '' }); return versions; @@ -101,17 +100,16 @@ export default class TranslationsFilter extends Component { setDebouncedQuery(event: Event) { const target = event.target as HTMLInputElement; - perform(this.debounceQuery, target.value); + this.debounceQuery.perform(target.value); } - @restartableTask - *debounceQuery(query: string) { + debounceQuery = restartableTask(async (query: string) => { this.debouncedQuery = query; - yield timeout(DEBOUNCE_OFFSET); + await timeout(DEBOUNCE_OFFSET); this.args.onChangeQuery(query); - } + }); @action submitForm(event: Event) { diff --git a/webapp/app/pods/components/translations-list/component.ts b/webapp/app/components/translations-list/component.ts similarity index 100% rename from webapp/app/pods/components/translations-list/component.ts rename to webapp/app/components/translations-list/component.ts diff --git a/webapp/app/pods/components/translations-list/item/component.ts b/webapp/app/components/translations-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/translations-list/item/component.ts rename to webapp/app/components/translations-list/item/component.ts diff --git a/webapp/app/pods/components/version-create-form/component.ts b/webapp/app/components/version-create-form/component.ts similarity index 66% rename from webapp/app/pods/components/version-create-form/component.ts rename to webapp/app/components/version-create-form/component.ts index 4d63b5cda..009e0bf8f 100644 --- a/webapp/app/pods/components/version-create-form/component.ts +++ b/webapp/app/components/version-create-form/component.ts @@ -5,7 +5,7 @@ import {tracked} from '@glimmer/tracking'; interface Args { error: boolean; project: any; - onCreate: ({name, tag}: {name: string; tag: string}) => Promise; + onCreate: (args: object) => Promise; } export default class VersionCreateForm extends Component { @@ -15,6 +15,9 @@ export default class VersionCreateForm extends Component { @tracked tag = ''; + @tracked + copyOnUpdateTranslation = true; + @tracked isCreating = false; @@ -32,14 +35,21 @@ export default class VersionCreateForm extends Component { this.tag = target.value; } + @action + setCopyOnUpdateTranslation(event: Event) { + const target = event.target as HTMLInputElement; + this.copyOnUpdateTranslation = target.checked; + } + @action async submit() { this.isCreating = true; - const tag = this.tag; - const name = this.name; - - await this.args.onCreate({tag, name}); + await this.args.onCreate({ + tag: this.tag, + name: this.name, + copyOnUpdateTranslation: this.copyOnUpdateTranslation + }); this.isCreating = false; } diff --git a/webapp/app/pods/components/version-update-form/component.ts b/webapp/app/components/version-update-form/component.ts similarity index 62% rename from webapp/app/pods/components/version-update-form/component.ts rename to webapp/app/components/version-update-form/component.ts index be65fb31c..26f2f94a1 100644 --- a/webapp/app/pods/components/version-update-form/component.ts +++ b/webapp/app/components/version-update-form/component.ts @@ -6,7 +6,11 @@ interface Args { version: any; error: boolean; project: any; - onUpdate: ({tag, name}: {tag: string; name: string}) => Promise; + onUpdate: (args: { + tag: string; + name: string; + copyOnUpdateTranslation: boolean; + }) => Promise; } export default class VersionUpdateForm extends Component { @@ -16,6 +20,9 @@ export default class VersionUpdateForm extends Component { @tracked tag = this.args.version.tag; + @tracked + copyOnUpdateTranslation = this.args.version.copyOnUpdateTranslation; + @tracked isSubmitting = false; @@ -23,14 +30,21 @@ export default class VersionUpdateForm extends Component { async submit() { this.isSubmitting = true; - const tag = this.tag; - const name = this.name; - - await this.args.onUpdate({tag, name}); + await this.args.onUpdate({ + tag: this.tag, + name: this.name, + copyOnUpdateTranslation: this.copyOnUpdateTranslation + }); if (!this.isDestroyed) this.isSubmitting = false; } + @action + setCopyOnUpdateTranslation(event: Event) { + const target = event.target as HTMLInputElement; + this.copyOnUpdateTranslation = target.checked; + } + @action setName(event: Event) { const target = event.target as HTMLInputElement; diff --git a/webapp/app/pods/components/versions-add-button/component.ts b/webapp/app/components/versions-add-button/component.ts similarity index 100% rename from webapp/app/pods/components/versions-add-button/component.ts rename to webapp/app/components/versions-add-button/component.ts diff --git a/webapp/app/pods/components/versions-list/component.ts b/webapp/app/components/versions-list/component.ts similarity index 100% rename from webapp/app/pods/components/versions-list/component.ts rename to webapp/app/components/versions-list/component.ts diff --git a/webapp/app/pods/components/versions-list/item/component.ts b/webapp/app/components/versions-list/item/component.ts similarity index 100% rename from webapp/app/pods/components/versions-list/item/component.ts rename to webapp/app/components/versions-list/item/component.ts diff --git a/webapp/app/pods/components/welcome-project/component.ts b/webapp/app/components/welcome-project/component.ts similarity index 100% rename from webapp/app/pods/components/welcome-project/component.ts rename to webapp/app/components/welcome-project/component.ts diff --git a/webapp/app/computed-macros/parsed-key.ts b/webapp/app/computed-macros/parsed-key.ts index 0e7d031fd..1045ad0ae 100644 --- a/webapp/app/computed-macros/parsed-key.ts +++ b/webapp/app/computed-macros/parsed-key.ts @@ -11,6 +11,6 @@ export default (key: string): ParsedKey => { return { value: isSplitted ? splittedKey[1] : key, - prefix: isSplitted ? splittedKey[0] : '', + prefix: isSplitted ? splittedKey[0] : '' }; }; diff --git a/webapp/app/config/environment.d.ts b/webapp/app/config/environment.d.ts index db7c72465..69d64852a 100644 --- a/webapp/app/config/environment.d.ts +++ b/webapp/app/config/environment.d.ts @@ -8,7 +8,6 @@ declare const config: { environment: any; modulePrefix: string; - podModulePrefix: string; locationType: string; rootURL: string; version: string; diff --git a/webapp/app/pods/error/controller.ts b/webapp/app/controllers/error.ts similarity index 100% rename from webapp/app/pods/error/controller.ts rename to webapp/app/controllers/error.ts diff --git a/webapp/app/pods/logged-in/controller.ts b/webapp/app/controllers/logged-in.ts similarity index 100% rename from webapp/app/pods/logged-in/controller.ts rename to webapp/app/controllers/logged-in.ts diff --git a/webapp/app/pods/logged-in/jipt/controller.ts b/webapp/app/controllers/logged-in/jipt.ts similarity index 91% rename from webapp/app/pods/logged-in/jipt/controller.ts rename to webapp/app/controllers/logged-in/jipt.ts index 3bca76bdb..6fa31c40c 100644 --- a/webapp/app/pods/logged-in/jipt/controller.ts +++ b/webapp/app/controllers/logged-in/jipt.ts @@ -6,11 +6,15 @@ import {readOnly} from '@ember/object/computed'; import Color from 'color'; import RouterService from '@ember/routing/router-service'; import {tracked} from '@glimmer/tracking'; +import GlobalState from 'accent-webapp/services/global-state'; export default class JIPTController extends Controller { @service('router') router: RouterService; + @service('global-state') + globalState: GlobalState; + queryParams = ['revisionId']; @tracked @@ -28,7 +32,7 @@ export default class JIPTController extends Controller { @readOnly('model.project.revisions') revisions: any; - constructor(...args: any) { + constructor(...args: any[]) { super(...args); window.addEventListener( @@ -42,7 +46,7 @@ export default class JIPTController extends Controller { } if (payload.data.jipt && payload.data.selectIds) { this.router.transitionTo('logged-in.jipt.index', { - queryParams: {translationIds: payload.data.selectIds}, + queryParams: {translationIds: payload.data.selectIds} }); } }, diff --git a/webapp/app/pods/logged-in/jipt/index/controller.ts b/webapp/app/controllers/logged-in/jipt/index.ts similarity index 89% rename from webapp/app/pods/logged-in/jipt/index/controller.ts rename to webapp/app/controllers/logged-in/jipt/index.ts index e22ae959a..d1dd9d317 100644 --- a/webapp/app/pods/logged-in/jipt/index/controller.ts +++ b/webapp/app/controllers/logged-in/jipt/index.ts @@ -53,15 +53,15 @@ export default class IndexController extends Controller { } @action - changeVersion(versionId: string) { + changeVersion(element: HTMLSelectElement) { this.page = 1; - this.version = versionId; + this.version = element.value; } @action - changeDocument(documentId: string) { + changeDocument(element: HTMLSelectElement) { this.page = 1; - this.document = documentId; + this.document = element.value; } @action diff --git a/webapp/app/pods/logged-in/jipt/translation/controller.ts b/webapp/app/controllers/logged-in/jipt/translation.ts similarity index 100% rename from webapp/app/pods/logged-in/jipt/translation/controller.ts rename to webapp/app/controllers/logged-in/jipt/translation.ts diff --git a/webapp/app/pods/logged-in/jipt/translation/index/controller.ts b/webapp/app/controllers/logged-in/jipt/translation/index.ts similarity index 97% rename from webapp/app/pods/logged-in/jipt/translation/index/controller.ts rename to webapp/app/controllers/logged-in/jipt/translation/index.ts index d2a5cd855..8d53ccf41 100644 --- a/webapp/app/pods/logged-in/jipt/translation/index/controller.ts +++ b/webapp/app/controllers/logged-in/jipt/translation/index.ts @@ -62,8 +62,8 @@ export default class IndexController extends Controller { mutation: translationCorrectQuery, variables: { translationId: conflict.id, - text, - }, + text + } }); if (response.errors) { @@ -81,14 +81,15 @@ export default class IndexController extends Controller { } @action - async uncorrectConflict() { + async uncorrectConflict(text: string) { const conflict = this.model.translation; const response = await this.apolloMutate.mutate({ mutation: translationUncorrectQuery, variables: { translationId: conflict.id, - }, + text + } }); if (response.errors) { @@ -113,8 +114,8 @@ export default class IndexController extends Controller { mutation: translationUpdateQuery, variables: { translationId: translation.id, - text, - }, + text + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/controller.ts b/webapp/app/controllers/logged-in/project.ts similarity index 100% rename from webapp/app/pods/logged-in/project/controller.ts rename to webapp/app/controllers/logged-in/project.ts diff --git a/webapp/app/pods/logged-in/project/activities/controller.ts b/webapp/app/controllers/logged-in/project/activities.ts similarity index 98% rename from webapp/app/pods/logged-in/project/activities/controller.ts rename to webapp/app/controllers/logged-in/project/activities.ts index c7262246c..d320d1da8 100644 --- a/webapp/app/pods/logged-in/project/activities/controller.ts +++ b/webapp/app/controllers/logged-in/project/activities.ts @@ -26,7 +26,7 @@ export default class ActivitiesController extends Controller { 'actionFilter', 'userFilter', 'page', - 'versionFilter', + 'versionFilter' ]; @tracked diff --git a/webapp/app/pods/logged-in/project/activity/controller.ts b/webapp/app/controllers/logged-in/project/activity.ts similarity index 96% rename from webapp/app/pods/logged-in/project/activity/controller.ts rename to webapp/app/controllers/logged-in/project/activity.ts index 828716a76..e4dc236c1 100644 --- a/webapp/app/pods/logged-in/project/activity/controller.ts +++ b/webapp/app/controllers/logged-in/project/activity.ts @@ -41,8 +41,8 @@ export default class ActivityController extends Controller { const response = await this.apolloMutate.mutate({ mutation: operationRollbackQuery, variables: { - operationId: this.model.activity.id, - }, + operationId: this.model.activity.id + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/collaborators/controller.ts b/webapp/app/controllers/logged-in/project/collaborators.ts similarity index 95% rename from webapp/app/pods/logged-in/project/collaborators/controller.ts rename to webapp/app/controllers/logged-in/project/collaborators.ts index e9023e93f..5963dcc81 100644 --- a/webapp/app/pods/logged-in/project/collaborators/controller.ts +++ b/webapp/app/controllers/logged-in/project/collaborators.ts @@ -58,8 +58,8 @@ export default class CollaboratorsController extends Controller { variables: { projectId: project.id, email, - role, - }, + role + } }); } @@ -71,8 +71,8 @@ export default class CollaboratorsController extends Controller { errorMessage: FLASH_MESSAGE_COLLABORATOR_UPDATE_ERROR, variables: { collaboratorId: collaborator.id, - role, - }, + role + } }); } @@ -83,8 +83,8 @@ export default class CollaboratorsController extends Controller { successMessage: FLASH_MESSAGE_COLLABORATOR_REMOVE_SUCCESS, errorMessage: FLASH_MESSAGE_COLLABORATOR_REMOVE_ERROR, variables: { - collaboratorId: collaborator.id, - }, + collaboratorId: collaborator.id + } }); } @@ -92,7 +92,7 @@ export default class CollaboratorsController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -102,7 +102,7 @@ export default class CollaboratorsController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['ProjectCollaborators'], + refetchQueries: ['ProjectCollaborators'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/comments/controller.ts b/webapp/app/controllers/logged-in/project/comments.ts similarity index 96% rename from webapp/app/pods/logged-in/project/comments/controller.ts rename to webapp/app/controllers/logged-in/project/comments.ts index 27034135a..52554462a 100644 --- a/webapp/app/pods/logged-in/project/comments/controller.ts +++ b/webapp/app/controllers/logged-in/project/comments.ts @@ -49,8 +49,8 @@ export default class CommentsController extends Controller { refetchQueries: ['ProjectComments'], variables: { commentId: comment.id, - text: comment.text, - }, + text: comment.text + } }); } @@ -60,8 +60,8 @@ export default class CommentsController extends Controller { mutation: commentDeleteQuery, refetchQueries: ['ProjectComments'], variables: { - commentId: comment.id, - }, + commentId: comment.id + } }); if (response.errors) { diff --git a/webapp/app/controllers/logged-in/project/conflicts.ts b/webapp/app/controllers/logged-in/project/conflicts.ts new file mode 100644 index 000000000..ae352e67e --- /dev/null +++ b/webapp/app/controllers/logged-in/project/conflicts.ts @@ -0,0 +1,212 @@ +import {action} from '@ember/object'; +import {inject as service} from '@ember/service'; +import {readOnly, equal, and} from '@ember/object/computed'; +import Controller from '@ember/controller'; +import translationCorrectQuery from 'accent-webapp/queries/correct-translation'; +import translationUncorrectQuery from 'accent-webapp/queries/uncorrect-translation'; +import translationUpdateQuery from 'accent-webapp/queries/update-translation'; +import IntlService from 'ember-intl/services/intl'; +import FlashMessages from 'ember-cli-flash/services/flash-messages'; +import Apollo from 'accent-webapp/services/apollo'; +import ApolloMutate from 'accent-webapp/services/apollo-mutate'; +import GlobalState from 'accent-webapp/services/global-state'; +import {tracked} from '@glimmer/tracking'; + +const FLASH_MESSAGE_CORRECT_SUCCESS = + 'pods.project.conflicts.flash_messages.correct_success'; +const FLASH_MESSAGE_CORRECT_ERROR = + 'pods.project.conflicts.flash_messages.correct_error'; +const FLASH_MESSAGE_UNCORRECT_SUCCESS = + 'pods.project.conflicts.flash_messages.uncorrect_success'; +const FLASH_MESSAGE_UNCORRECT_ERROR = + 'pods.project.conflicts.flash_messages.uncorrect_error'; +const FLASH_MESSAGE_UPDATE_SUCCESS = + 'pods.project.conflicts.flash_messages.update_success'; +const FLASH_MESSAGE_UPDATE_ERROR = + 'pods.project.conflicts.flash_messages.update_error'; + +export default class ConflictsController extends Controller { + @tracked + model: any; + + @service('intl') + intl: IntlService; + + @service('flash-messages') + flashMessages: FlashMessages; + + @service('apollo') + apollo: Apollo; + + @service('apollo-mutate') + apolloMutate: ApolloMutate; + + @service('global-state') + globalState: GlobalState; + + queryParams = [ + 'page', + 'query', + 'document', + 'version', + 'relatedRevisions', + 'isTextEmpty', + 'isTextNotEmpty', + 'isAddedLastSync', + 'isCommentedOn', + 'isTranslatedFilter' + ]; + + @tracked + query = ''; + + @tracked + document: string | null = null; + + @tracked + version: string | null = null; + + @tracked + relatedRevisions: string[] = []; + + @tracked + isTextEmpty: 'true' | null = null; + + @tracked + isTextNotEmpty: 'true' | null = null; + + @tracked + isAddedLastSync: 'true' | null = null; + + @tracked + isTranslated: 'true' | null = null; + + @tracked + isCommentedOn: 'true' | null = null; + + @tracked + page = 1; + + @readOnly('globalState.permissions') + permissions: any; + + @equal('model.groupedTranslations.entries', undefined) + emptyEntries: boolean; + + @and('emptyEntries', 'model.loading') + showLoading: boolean; + + @and('emptyEntries', 'model.loading') + showSkeleton: boolean; + + get withAdvancedFilters() { + return [ + this.isTextEmpty, + this.isTextNotEmpty, + this.isAddedLastSync, + this.isCommentedOn, + this.isTranslated + ].filter((filter) => filter === 'true').length; + } + + @action + changeAdvancedFilterBoolean( + key: + | 'isTextEmpty' + | 'isTextNotEmpty' + | 'isAddedLastSync' + | 'isCommentedOn' + | 'isTranslated', + event: InputEvent + ) { + this[key] = (event.target as HTMLInputElement).checked ? 'true' : null; + } + + @action + async correctConflict(conflict: any, text: string) { + const response = await this.apolloMutate.mutate({ + mutation: translationCorrectQuery, + variables: { + translationId: conflict.id, + text + } + }); + + if (response.errors) { + this.flashMessages.error(this.intl.t(FLASH_MESSAGE_CORRECT_ERROR)); + } else { + this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CORRECT_SUCCESS)); + } + + return response; + } + + @action + async uncorrectConflict(conflict: any, text: string) { + const response = await this.apolloMutate.mutate({ + mutation: translationUncorrectQuery, + variables: { + translationId: conflict.id, + text + } + }); + + if (response.errors) { + this.flashMessages.error(this.intl.t(FLASH_MESSAGE_UNCORRECT_ERROR)); + } else { + this.flashMessages.success(this.intl.t(FLASH_MESSAGE_UNCORRECT_SUCCESS)); + } + + return response; + } + + @action + async updateConflict(conflict: any, text: string) { + const response = await this.apolloMutate.mutate({ + mutation: translationUpdateQuery, + variables: { + translationId: conflict.id, + text + } + }); + + if (response.errors) { + this.flashMessages.error(this.intl.t(FLASH_MESSAGE_UPDATE_ERROR)); + } else { + this.flashMessages.success(this.intl.t(FLASH_MESSAGE_UPDATE_SUCCESS)); + } + + return response; + } + + @action + changeQuery(query: string) { + this.page = 1; + this.query = query; + } + + @action + changeDocument(select: HTMLSelectElement) { + this.page = 1; + this.document = select.value ? select.value : null; + } + + @action + changeVersion(select: HTMLSelectElement) { + this.page = 1; + this.version = select.value ? select.value : null; + } + + @action + changeRelatedRevisions(choices: Array<{value: string}>) { + this.page = 1; + this.relatedRevisions = choices.map(({value}) => value); + } + + @action + selectPage(page: number) { + window.scrollTo(0, 0); + + this.page = page; + } +} diff --git a/webapp/app/pods/logged-in/project/edit/api-token/controller.ts b/webapp/app/controllers/logged-in/project/edit/api-token.ts similarity index 95% rename from webapp/app/pods/logged-in/project/edit/api-token/controller.ts rename to webapp/app/controllers/logged-in/project/edit/api-token.ts index 4851d8f53..468012627 100644 --- a/webapp/app/pods/logged-in/project/edit/api-token/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/api-token.ts @@ -51,7 +51,7 @@ export default class APITokenController extends Controller { async createApiToken({ name, pictureUrl, - permissions, + permissions }: { name: string; pictureUrl: string | null; @@ -67,8 +67,8 @@ export default class APITokenController extends Controller { projectId: project.id, pictureUrl, name, - permissions, - }, + permissions + } }); } @@ -79,8 +79,8 @@ export default class APITokenController extends Controller { successMessage: FLASH_MESSAGE_API_TOKEN_REVOKE_SUCCESS, errorMessage: FLASH_MESSAGE_API_TOKEN_REVOKE_ERROR, variables: { - id: apiToken.id, - }, + id: apiToken.id + } }); } @@ -88,7 +88,7 @@ export default class APITokenController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -98,7 +98,7 @@ export default class APITokenController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['ProjectApiToken'], + refetchQueries: ['ProjectApiToken'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/badges/controller.ts b/webapp/app/controllers/logged-in/project/edit/badges.ts similarity index 100% rename from webapp/app/pods/logged-in/project/edit/badges/controller.ts rename to webapp/app/controllers/logged-in/project/edit/badges.ts diff --git a/webapp/app/pods/logged-in/project/edit/index/controller.ts b/webapp/app/controllers/logged-in/project/edit/index.ts similarity index 90% rename from webapp/app/pods/logged-in/project/edit/index/controller.ts rename to webapp/app/controllers/logged-in/project/edit/index.ts index 33322302f..8fec65d9a 100644 --- a/webapp/app/pods/logged-in/project/edit/index/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/index.ts @@ -9,6 +9,7 @@ import IntlService from 'ember-intl/services/intl'; import FlashMessages from 'ember-cli-flash/services/flash-messages'; import ApolloMutate from 'accent-webapp/services/apollo-mutate'; import GlobalState from 'accent-webapp/services/global-state'; +import RouterService from '@ember/routing/router-service'; const FLASH_MESSAGE_PREFIX = 'pods.project.edit.flash_messages.'; const FLASH_MESSAGE_PROJECT_SUCCESS = `${FLASH_MESSAGE_PREFIX}update_success`; @@ -29,6 +30,9 @@ export default class ProjectEditIndexController extends Controller { @service('global-state') globalState: GlobalState; + @service('router') + router: RouterService; + @readOnly('model.project') project: any; @@ -48,8 +52,8 @@ export default class ProjectEditIndexController extends Controller { const response = await this.apolloMutate.mutate({ mutation: projectDeleteQuery, variables: { - projectId: project.id, - }, + projectId: project.id + } }); if (response.errors) { @@ -59,7 +63,7 @@ export default class ProjectEditIndexController extends Controller { this.intl.t(FLASH_MESSAGE_DELETE_PROJECT_SUCCESS) ); - this.transitionToRoute('logged-in.projects'); + this.router.transitionTo('logged-in.projects'); } return response; @@ -75,8 +79,8 @@ export default class ProjectEditIndexController extends Controller { errorMessage: FLASH_MESSAGE_PROJECT_ERROR, variables: { projectId: project.id, - ...projectAttributes, - }, + ...projectAttributes + } }); } @@ -84,7 +88,7 @@ export default class ProjectEditIndexController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -94,7 +98,7 @@ export default class ProjectEditIndexController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['ProjectEdit'], + refetchQueries: ['ProjectEdit'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/jipt/controller.ts b/webapp/app/controllers/logged-in/project/edit/jipt.ts similarity index 100% rename from webapp/app/pods/logged-in/project/edit/jipt/controller.ts rename to webapp/app/controllers/logged-in/project/edit/jipt.ts diff --git a/webapp/app/pods/logged-in/project/edit/machine-translations/controller.ts b/webapp/app/controllers/logged-in/project/edit/machine-translations.ts similarity index 94% rename from webapp/app/pods/logged-in/project/edit/machine-translations/controller.ts rename to webapp/app/controllers/logged-in/project/edit/machine-translations.ts index 0ccf96d64..b6fa03b00 100644 --- a/webapp/app/pods/logged-in/project/edit/machine-translations/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/machine-translations.ts @@ -8,7 +8,7 @@ import FlashMessages from 'ember-cli-flash/services/flash-messages'; import ApolloMutate from 'accent-webapp/services/apollo-mutate'; import machineTranslationsConfigSaveQuery, { - SaveProjectMachineTranslationsConfigVariables, + SaveProjectMachineTranslationsConfigVariables } from 'accent-webapp/queries/save-project-machine-translations-config'; import machineTranslationsConfigDeleteQuery from 'accent-webapp/queries/delete-project-machine-translations-config'; @@ -51,7 +51,7 @@ export default class MachineTranslationController extends Controller { mutation: machineTranslationsConfigSaveQuery, successMessage: FLASH_MESSAGE_CONFIG_SUCCESS, errorMessage: FLASH_MESSAGE_CONFIG_ERROR, - variables: {...config, projectId: this.project.id}, + variables: {...config, projectId: this.project.id} }); } @@ -61,7 +61,7 @@ export default class MachineTranslationController extends Controller { mutation: machineTranslationsConfigDeleteQuery, successMessage: FLASH_MESSAGE_CONFIG_REMOVE_SUCCESS, errorMessage: FLASH_MESSAGE_CONFIG_REMOVE_ERROR, - variables: {projectId: this.project.id}, + variables: {projectId: this.project.id} }); } @@ -69,7 +69,7 @@ export default class MachineTranslationController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -79,7 +79,7 @@ export default class MachineTranslationController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['Project', 'ProjectMachineTranslationsConfig'], + refetchQueries: ['Project', 'ProjectMachineTranslationsConfig'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/prompts/controller.ts b/webapp/app/controllers/logged-in/project/edit/prompts.ts similarity index 79% rename from webapp/app/pods/logged-in/project/edit/prompts/controller.ts rename to webapp/app/controllers/logged-in/project/edit/prompts.ts index 1e9d52496..9779b6663 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/prompts.ts @@ -8,14 +8,12 @@ import FlashMessages from 'ember-cli-flash/services/flash-messages'; import ApolloMutate from 'accent-webapp/services/apollo-mutate'; import promptConfigSaveQuery, { - SaveProjectPromptConfigVariables, + SaveProjectPromptConfigVariables } from 'accent-webapp/queries/save-project-prompt-config'; import promptConfigDeleteQuery from 'accent-webapp/queries/delete-project-prompt-config'; import promptDeleteQuery from 'accent-webapp/queries/delete-project-prompt'; -import projectPromptConfigQuery, { - ProjectPromptConfigResponse, -} from 'accent-webapp/queries/project-prompt-config'; -import {InMemoryCache} from 'apollo-boost'; +import projectPromptConfigQuery from 'accent-webapp/queries/project-prompt-config'; +import {InMemoryCache} from '@apollo/client/cache'; const FLASH_MESSAGE_PREFIX = 'pods.project.edit.flash_messages.'; const FLASH_MESSAGE_CONFIG_SUCCESS = `${FLASH_MESSAGE_PREFIX}prompts_config_success`; @@ -59,7 +57,7 @@ export default class PromptsController extends Controller { mutation: promptConfigSaveQuery, successMessage: FLASH_MESSAGE_CONFIG_SUCCESS, errorMessage: FLASH_MESSAGE_CONFIG_ERROR, - variables: {...config, projectId: this.project.id}, + variables: {...config, projectId: this.project.id} }); } @@ -69,7 +67,7 @@ export default class PromptsController extends Controller { mutation: promptConfigDeleteQuery, successMessage: FLASH_MESSAGE_CONFIG_REMOVE_SUCCESS, errorMessage: FLASH_MESSAGE_CONFIG_REMOVE_ERROR, - variables: {projectId: this.project.id}, + variables: {projectId: this.project.id} }); } @@ -79,7 +77,7 @@ export default class PromptsController extends Controller { mutation: promptDeleteQuery, successMessage: FLASH_MESSAGE_PROMPT_REMOVE_SUCCESS, errorMessage: FLASH_MESSAGE_PROMPT_REMOVE_ERROR, - variables: {promptId}, + variables: {promptId} }); } @@ -87,7 +85,7 @@ export default class PromptsController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -97,7 +95,7 @@ export default class PromptsController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['Project', 'ProjectPromptConfig'], + refetchQueries: ['Project', 'ProjectPromptConfig'] }); if (response.errors) { @@ -113,7 +111,7 @@ export default class PromptsController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -125,22 +123,26 @@ export default class PromptsController extends Controller { variables, refetchQueries: ['Project'], update: (cache: InMemoryCache) => { - const data = cache.readQuery({ - query: projectPromptConfigQuery, - variables: {projectId: this.project.id}, - }) as ProjectPromptConfigResponse; - - const prompts = data.viewer.project.prompts.filter( - (prompt) => prompt.id !== variables.promptId + cache.updateQuery( + { + query: projectPromptConfigQuery, + variables: {projectId: this.project.id} + }, + (data) => { + return { + viewer: { + ...data.viewer, + project: { + ...data.viewer.project, + prompts: data.viewer.project.prompts.filter( + (prompt: any) => prompt.id !== variables.promptId + ) + } + } + }; + } ); - data.viewer.project.prompts = prompts; - - cache.writeQuery({ - query: projectPromptConfigQuery, - variables: {projectId: this.project.id}, - data, - }); - }, + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/prompts/edit/controller.ts b/webapp/app/controllers/logged-in/project/edit/prompts/edit.ts similarity index 97% rename from webapp/app/pods/logged-in/project/edit/prompts/edit/controller.ts rename to webapp/app/controllers/logged-in/project/edit/prompts/edit.ts index fd72a58c0..195ea2207 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/edit/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/prompts/edit.ts @@ -50,7 +50,7 @@ export default class PromptsEditController extends Controller { async update({ content, name, - quickAccess, + quickAccess }: { name: string | null; content: string; @@ -60,7 +60,7 @@ export default class PromptsEditController extends Controller { mutation: promptUpdateQuery, successMessage: FLASH_MESSAGE_PROMPT_UPDATE_SUCCESS, errorMessage: FLASH_MESSAGE_PROMPT_UPDATE_ERROR, - variables: {id: this.prompt.id, content, name, quickAccess}, + variables: {id: this.prompt.id, content, name, quickAccess} }); } @@ -68,7 +68,7 @@ export default class PromptsEditController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -79,7 +79,7 @@ export default class PromptsEditController extends Controller { const response = await this.apolloMutate.mutate({ mutation, - variables, + variables }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/prompts/new/controller.ts b/webapp/app/controllers/logged-in/project/edit/prompts/new.ts similarity index 77% rename from webapp/app/pods/logged-in/project/edit/prompts/new/controller.ts rename to webapp/app/controllers/logged-in/project/edit/prompts/new.ts index 3b98546af..4531d13cc 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/new/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/prompts/new.ts @@ -10,12 +10,10 @@ import ApolloMutate from 'accent-webapp/services/apollo-mutate'; import RouterService from '@ember/routing/router-service'; import promptCreateQuery, { - CreatePromptResponse, + CreatePromptResponse } from 'accent-webapp/queries/create-project-prompt'; -import projectPromptConfigQuery, { - ProjectPromptConfigResponse, -} from 'accent-webapp/queries/project-prompt-config'; -import {InMemoryCache} from 'apollo-boost'; +import projectPromptConfigQuery from 'accent-webapp/queries/project-prompt-config'; +import {InMemoryCache} from '@apollo/client/cache'; const FLASH_MESSAGE_PREFIX = 'pods.project.edit.flash_messages.'; const FLASH_MESSAGE_PROMPT_CREATE_SUCCESS = `${FLASH_MESSAGE_PREFIX}prompts_create_success`; @@ -50,7 +48,7 @@ export default class PromptsNewController extends Controller { async create({ content, name, - quickAccess, + quickAccess }: { name: string | null; content: string; @@ -60,7 +58,7 @@ export default class PromptsNewController extends Controller { mutation: promptCreateQuery, successMessage: FLASH_MESSAGE_PROMPT_CREATE_SUCCESS, errorMessage: FLASH_MESSAGE_PROMPT_CREATE_ERROR, - variables: {id: this.project.id, content, name, quickAccess}, + variables: {id: this.project.id, content, name, quickAccess} }); } @@ -68,7 +66,7 @@ export default class PromptsNewController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -84,24 +82,29 @@ export default class PromptsNewController extends Controller { update: ( cache: InMemoryCache, { - data: {createProjectPrompt}, + data: {createProjectPrompt} }: {data: {createProjectPrompt: CreatePromptResponse}} ) => { - const data = cache.readQuery({ - query: projectPromptConfigQuery, - variables: {projectId: this.project.id}, - }) as ProjectPromptConfigResponse; - const prompts = data.viewer.project.prompts.concat([ - createProjectPrompt.prompt, - ]); - data.viewer.project.prompts = prompts; - - cache.writeQuery({ - query: projectPromptConfigQuery, - variables: {projectId: this.project.id}, - data, - }); - }, + cache.updateQuery( + { + query: projectPromptConfigQuery, + variables: {projectId: this.project.id} + }, + (data) => { + return { + viewer: { + ...data.viewer, + project: { + ...data.viewer.project, + prompts: data.viewer.project.prompts.concat([ + createProjectPrompt.prompt + ]) + } + } + }; + } + ); + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/edit/service-integrations/controller.ts b/webapp/app/controllers/logged-in/project/edit/service-integrations.ts similarity index 94% rename from webapp/app/pods/logged-in/project/edit/service-integrations/controller.ts rename to webapp/app/controllers/logged-in/project/edit/service-integrations.ts index 862a8918a..314988256 100644 --- a/webapp/app/pods/logged-in/project/edit/service-integrations/controller.ts +++ b/webapp/app/controllers/logged-in/project/edit/service-integrations.ts @@ -48,7 +48,7 @@ export default class ServiceIntegrationsController extends Controller { async createIntegration({ data, events, - service, + service }: { data: any; events: any; @@ -64,8 +64,8 @@ export default class ServiceIntegrationsController extends Controller { projectId: project.id, data, events, - service, - }, + service + } }); } @@ -74,7 +74,7 @@ export default class ServiceIntegrationsController extends Controller { integration, data, events, - service, + service }: { integration: any; data: any; @@ -89,8 +89,8 @@ export default class ServiceIntegrationsController extends Controller { integrationId: integration.id, data, events, - service, - }, + service + } }); } @@ -101,8 +101,8 @@ export default class ServiceIntegrationsController extends Controller { successMessage: FLASH_MESSAGE_INTEGRATION_REMOVE_SUCCESS, errorMessage: FLASH_MESSAGE_INTEGRATION_REMOVE_ERROR, variables: { - integrationId: integration.id, - }, + integrationId: integration.id + } }); } @@ -110,7 +110,7 @@ export default class ServiceIntegrationsController extends Controller { mutation, variables, successMessage, - errorMessage, + errorMessage }: { mutation: any; variables: any; @@ -120,7 +120,7 @@ export default class ServiceIntegrationsController extends Controller { const response = await this.apolloMutate.mutate({ mutation, variables, - refetchQueries: ['ProjectServiceIntegrations'], + refetchQueries: ['ProjectServiceIntegrations'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/files/controller.ts b/webapp/app/controllers/logged-in/project/files.ts similarity index 97% rename from webapp/app/pods/logged-in/project/files/controller.ts rename to webapp/app/controllers/logged-in/project/files.ts index 575223c3c..e5e3f1dda 100644 --- a/webapp/app/pods/logged-in/project/files/controller.ts +++ b/webapp/app/controllers/logged-in/project/files.ts @@ -52,8 +52,8 @@ export default class FilesController extends Controller { const response = await this.apolloMutate.mutate({ mutation: documentDeleteQuery, variables: { - documentId: documentEntity.id, - }, + documentId: documentEntity.id + } }); if (response.errors) { @@ -70,8 +70,8 @@ export default class FilesController extends Controller { mutation: documentUpdateQuery, variables: { documentId: documentEntity.id, - path, - }, + path + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/files/add-translations/controller.ts b/webapp/app/controllers/logged-in/project/files/add-translations.ts similarity index 97% rename from webapp/app/pods/logged-in/project/files/add-translations/controller.ts rename to webapp/app/controllers/logged-in/project/files/add-translations.ts index 27f5f52a8..b5c23ec29 100644 --- a/webapp/app/pods/logged-in/project/files/add-translations/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/add-translations.ts @@ -90,7 +90,7 @@ export default class AddTranslationsController extends Controller { revision, version, mergeType, - mergeOptions, + mergeOptions }: { fileSource: any; documentFormat: string; @@ -111,7 +111,7 @@ export default class AddTranslationsController extends Controller { documentPath, documentFormat, mergeType, - mergeOptions, + mergeOptions }); this.revisionOperations = revisionOperations; @@ -123,7 +123,7 @@ export default class AddTranslationsController extends Controller { revision, documentFormat, mergeType, - mergeOptions, + mergeOptions }: { fileSource: any; revision: any; @@ -143,7 +143,7 @@ export default class AddTranslationsController extends Controller { documentPath, documentFormat, mergeType, - mergeOptions, + mergeOptions }); this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CREATE_SUCCESS)); diff --git a/webapp/app/pods/logged-in/project/files/export-all/controller.ts b/webapp/app/controllers/logged-in/project/files/export-all.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/export-all/controller.ts rename to webapp/app/controllers/logged-in/project/files/export-all.ts index 6b9b581f8..2c823bd8c 100644 --- a/webapp/app/pods/logged-in/project/files/export-all/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/export-all.ts @@ -31,7 +31,7 @@ export default class ExportAllController extends Controller { 'versionFilter', 'isTextEmpty', 'isAddedLastSync', - 'isConflictedFilter', + 'isConflictedFilter' ]; @tracked @@ -135,7 +135,7 @@ export default class ExportAllController extends Controller { @action exportFile() { const blob = new Blob([this.fileRender], { - type: 'charset=utf-8', + type: 'charset=utf-8' }); const filename = `${this.document.path}.${this.fileExtension}`; diff --git a/webapp/app/pods/logged-in/project/files/export/controller.ts b/webapp/app/controllers/logged-in/project/files/export.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/export/controller.ts rename to webapp/app/controllers/logged-in/project/files/export.ts index df326c49d..56c2b8c8e 100644 --- a/webapp/app/pods/logged-in/project/files/export/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/export.ts @@ -31,7 +31,7 @@ export default class ExportController extends Controller { 'versionFilter', 'isTextEmpty', 'isAddedLastSync', - 'isConflictedFilter', + 'isConflictedFilter' ]; @tracked @@ -144,7 +144,7 @@ export default class ExportController extends Controller { @action exportFile() { const blob = new Blob([this.fileRender], { - type: 'charset=utf-8', + type: 'charset=utf-8' }); const filename = `${this.document.path}.${this.fileExtension}`; diff --git a/webapp/app/pods/logged-in/project/files/jipt/controller.ts b/webapp/app/controllers/logged-in/project/files/jipt.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/jipt/controller.ts rename to webapp/app/controllers/logged-in/project/files/jipt.ts index bbd75a8cf..575b9ddad 100644 --- a/webapp/app/pods/logged-in/project/files/jipt/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/jipt.ts @@ -90,7 +90,7 @@ export default class JIPTController extends Controller { @action exportFile() { const blob = new Blob([this.fileRender], { - type: 'charset=utf-8', + type: 'charset=utf-8' }); const filename = `${this.document.path}.${this.fileExtension}`; diff --git a/webapp/app/pods/logged-in/project/files/machine-translations/controller.ts b/webapp/app/controllers/logged-in/project/files/machine-translations.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/machine-translations/controller.ts rename to webapp/app/controllers/logged-in/project/files/machine-translations.ts index 78b7ef77c..292335c90 100644 --- a/webapp/app/pods/logged-in/project/files/machine-translations/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/machine-translations.ts @@ -69,7 +69,7 @@ export default class MachineTranslationsController extends Controller { documentId: this.model.fileId, fromLanguage, toLanguage, - documentFormat, + documentFormat }); this.translatedFileContent = content; } catch (error) { diff --git a/webapp/app/pods/logged-in/project/files/new-machine-translations/controller.ts b/webapp/app/controllers/logged-in/project/files/new-machine-translations.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/new-machine-translations/controller.ts rename to webapp/app/controllers/logged-in/project/files/new-machine-translations.ts index 45478ed69..51d169ae6 100644 --- a/webapp/app/pods/logged-in/project/files/new-machine-translations/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/new-machine-translations.ts @@ -52,7 +52,7 @@ export default class NewMachineTranslationsController extends Controller { file, fromLanguage, toLanguage, - documentFormat, + documentFormat }); this.translatedFileContent = content; } catch (error) { diff --git a/webapp/app/pods/logged-in/project/files/new-sync/controller.ts b/webapp/app/controllers/logged-in/project/files/new-sync.ts similarity index 97% rename from webapp/app/pods/logged-in/project/files/new-sync/controller.ts rename to webapp/app/controllers/logged-in/project/files/new-sync.ts index c0ef72674..0a10f2353 100644 --- a/webapp/app/pods/logged-in/project/files/new-sync/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/new-sync.ts @@ -73,7 +73,7 @@ export default class NewSyncController extends Controller { documentPath, revision, version, - syncType, + syncType }: { fileSource: any; documentFormat: any; @@ -94,7 +94,7 @@ export default class NewSyncController extends Controller { file, documentPath, documentFormat, - syncType, + syncType }); this.revisionOperations = revisionOperations; @@ -107,7 +107,7 @@ export default class NewSyncController extends Controller { documentPath, revision, version, - syncType, + syncType }: { fileSource: any; documentFormat: any; @@ -127,7 +127,7 @@ export default class NewSyncController extends Controller { version, documentPath, documentFormat, - syncType, + syncType }); this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CREATE_SUCCESS)); diff --git a/webapp/app/pods/logged-in/project/files/sync/controller.ts b/webapp/app/controllers/logged-in/project/files/sync.ts similarity index 98% rename from webapp/app/pods/logged-in/project/files/sync/controller.ts rename to webapp/app/controllers/logged-in/project/files/sync.ts index e2121b5a0..b45dcc559 100644 --- a/webapp/app/pods/logged-in/project/files/sync/controller.ts +++ b/webapp/app/controllers/logged-in/project/files/sync.ts @@ -88,7 +88,7 @@ export default class SyncController extends Controller { documentFormat, revision, version, - syncType, + syncType }: { fileSource: any; documentFormat: any; @@ -108,7 +108,7 @@ export default class SyncController extends Controller { file, documentPath, documentFormat, - syncType, + syncType }); this.revisionOperations = revisionOperations; @@ -120,7 +120,7 @@ export default class SyncController extends Controller { documentFormat, revision, version, - syncType, + syncType }: { fileSource: any; documentFormat: any; @@ -140,7 +140,7 @@ export default class SyncController extends Controller { file, documentPath, documentFormat, - syncType, + syncType }); this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CREATE_SUCCESS)); diff --git a/webapp/app/pods/logged-in/project/index/controller.ts b/webapp/app/controllers/logged-in/project/index.ts similarity index 97% rename from webapp/app/pods/logged-in/project/index/controller.ts rename to webapp/app/controllers/logged-in/project/index.ts index 389b8706a..0b63ed8a1 100644 --- a/webapp/app/pods/logged-in/project/index/controller.ts +++ b/webapp/app/controllers/logged-in/project/index.ts @@ -59,7 +59,7 @@ export default class ProjectIndexController extends Controller { const response = await this.apolloMutate.mutate({ mutation: correctAllRevisionQuery, variables: {revisionId: revision.id}, - refetchQueries: ['Dashboard'], + refetchQueries: ['Dashboard'] }); if (response.errors) { @@ -78,7 +78,7 @@ export default class ProjectIndexController extends Controller { const response = await this.apolloMutate.mutate({ mutation: uncorrectAllRevisionQuery, variables: {revisionId: revision.id}, - refetchQueries: ['Dashboard'], + refetchQueries: ['Dashboard'] }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/manage-languages/controller.ts b/webapp/app/controllers/logged-in/project/manage-languages.ts similarity index 97% rename from webapp/app/pods/logged-in/project/manage-languages/controller.ts rename to webapp/app/controllers/logged-in/project/manage-languages.ts index 0897047b5..966a5ff6b 100644 --- a/webapp/app/pods/logged-in/project/manage-languages/controller.ts +++ b/webapp/app/controllers/logged-in/project/manage-languages.ts @@ -72,8 +72,8 @@ export default class ManageLanguagesController extends Controller { const response = await this.apolloMutate.mutate({ mutation: revisionDeleteQuery, variables: { - revisionId: revision.id, - }, + revisionId: revision.id + } }); if (response.errors) { @@ -94,8 +94,8 @@ export default class ManageLanguagesController extends Controller { const response = await this.apolloMutate.mutate({ mutation: revisionMasterPromoteQuery, variables: { - revisionId: revision.id, - }, + revisionId: revision.id + } }); if (response.errors) { @@ -126,8 +126,8 @@ export default class ManageLanguagesController extends Controller { projectId: project.id, languageId, defaultNull: options.defaultNull, - machineTranslationsEnabled: options.machineTranslationsEnabled, - }, + machineTranslationsEnabled: options.machineTranslationsEnabled + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/manage-languages/edit/controller.ts b/webapp/app/controllers/logged-in/project/manage-languages/edit.ts similarity index 97% rename from webapp/app/pods/logged-in/project/manage-languages/edit/controller.ts rename to webapp/app/controllers/logged-in/project/manage-languages/edit.ts index 3beb37535..adcc68d78 100644 --- a/webapp/app/pods/logged-in/project/manage-languages/edit/controller.ts +++ b/webapp/app/controllers/logged-in/project/manage-languages/edit.ts @@ -52,8 +52,8 @@ export default class ManageLanguagesEditController extends Controller { mutation: revisionUpdateQuery, variables: { revisionId: this.revision.id, - ...revisionAttributes, - }, + ...revisionAttributes + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/revision/controller.ts b/webapp/app/controllers/logged-in/project/revision.ts similarity index 100% rename from webapp/app/pods/logged-in/project/revision/controller.ts rename to webapp/app/controllers/logged-in/project/revision.ts diff --git a/webapp/app/pods/logged-in/project/revision/lint-translations/controller.ts b/webapp/app/controllers/logged-in/project/revision/lint-translations.ts similarity index 100% rename from webapp/app/pods/logged-in/project/revision/lint-translations/controller.ts rename to webapp/app/controllers/logged-in/project/revision/lint-translations.ts diff --git a/webapp/app/pods/logged-in/project/revision/translations/controller.ts b/webapp/app/controllers/logged-in/project/revision/translations.ts similarity index 97% rename from webapp/app/pods/logged-in/project/revision/translations/controller.ts rename to webapp/app/controllers/logged-in/project/revision/translations.ts index fea9a4b6d..ba3ec9fed 100644 --- a/webapp/app/pods/logged-in/project/revision/translations/controller.ts +++ b/webapp/app/controllers/logged-in/project/revision/translations.ts @@ -37,7 +37,7 @@ export default class TranslationsController extends Controller { 'isAddedLastSync', 'isCommentedOn', 'isConflictedFilter', - 'isTranslatedFilter', + 'isTranslatedFilter' ]; @tracked @@ -95,7 +95,7 @@ export default class TranslationsController extends Controller { this.isAddedLastSync, this.isCommentedOn, this.isConflicted, - this.isTranslated, + this.isTranslated ].filter((filter) => filter === 'true').length; } @@ -144,8 +144,8 @@ export default class TranslationsController extends Controller { mutation: translationUpdateQuery, variables: { translationId: translation.id, - text, - }, + text + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/translation/controller.ts b/webapp/app/controllers/logged-in/project/translation.ts similarity index 100% rename from webapp/app/pods/logged-in/project/translation/controller.ts rename to webapp/app/controllers/logged-in/project/translation.ts diff --git a/webapp/app/pods/logged-in/project/translation/activities/controller.ts b/webapp/app/controllers/logged-in/project/translation/activities.ts similarity index 100% rename from webapp/app/pods/logged-in/project/translation/activities/controller.ts rename to webapp/app/controllers/logged-in/project/translation/activities.ts diff --git a/webapp/app/pods/logged-in/project/translation/comments/controller.ts b/webapp/app/controllers/logged-in/project/translation/comments.ts similarity index 94% rename from webapp/app/pods/logged-in/project/translation/comments/controller.ts rename to webapp/app/controllers/logged-in/project/translation/comments.ts index a0412794d..751ed09c7 100644 --- a/webapp/app/pods/logged-in/project/translation/comments/controller.ts +++ b/webapp/app/controllers/logged-in/project/translation/comments.ts @@ -58,8 +58,8 @@ export default class CommentsController extends Controller { refetchQueries: ['TranslationComments'], variables: { translationId: translation.id, - text, - }, + text + } }); } @@ -70,8 +70,8 @@ export default class CommentsController extends Controller { refetchQueries: ['TranslationComments'], variables: { commentId: comment.id, - text: comment.text, - }, + text: comment.text + } }); } @@ -81,8 +81,8 @@ export default class CommentsController extends Controller { mutation: commentDeleteQuery, refetchQueries: ['TranslationComments'], variables: { - commentId: comment.id, - }, + commentId: comment.id + } }); if (response.errors) { @@ -103,8 +103,8 @@ export default class CommentsController extends Controller { refetchQueries: ['TranslationComments'], variables: { translationId: translation.id, - userId: user.id, - }, + userId: user.id + } }); } @@ -114,8 +114,8 @@ export default class CommentsController extends Controller { mutation: translationCommentsSubscriptionDeleteQuery, refetchQueries: ['TranslationComments'], variables: { - translationCommentsSubscripitionId: subscription.id, - }, + translationCommentsSubscripitionId: subscription.id + } }); } diff --git a/webapp/app/pods/logged-in/project/translation/editions/controller.ts b/webapp/app/controllers/logged-in/project/translation/editions.ts similarity index 95% rename from webapp/app/pods/logged-in/project/translation/editions/controller.ts rename to webapp/app/controllers/logged-in/project/translation/editions.ts index f2eb24589..0347d67c4 100644 --- a/webapp/app/pods/logged-in/project/translation/editions/controller.ts +++ b/webapp/app/controllers/logged-in/project/translation/editions.ts @@ -29,7 +29,7 @@ export default class TranslationEditionsController extends Controller { @readOnly('globalState.permissions') permissions: any; - @readOnly('model.revisionModel.project.revisions') + @readOnly('model.translationModel.revisions') revisions: any; @equal('model.translations', undefined) @@ -50,8 +50,8 @@ export default class TranslationEditionsController extends Controller { mutation: translationUpdateQuery, variables: { translationId: translation.id, - text, - }, + text + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/translation/index/controller.ts b/webapp/app/controllers/logged-in/project/translation/index.ts similarity index 97% rename from webapp/app/pods/logged-in/project/translation/index/controller.ts rename to webapp/app/controllers/logged-in/project/translation/index.ts index 33e84a4cb..62a1df965 100644 --- a/webapp/app/pods/logged-in/project/translation/index/controller.ts +++ b/webapp/app/controllers/logged-in/project/translation/index.ts @@ -53,8 +53,8 @@ export default class IndexController extends Controller { mutation: translationCorrectQuery, variables: { translationId: conflict.id, - text, - }, + text + } }); if (response.errors) { @@ -72,8 +72,8 @@ export default class IndexController extends Controller { mutation: translationUncorrectQuery, variables: { translationId: conflict.id, - text, - }, + text + } }); if (response.errors) { @@ -91,8 +91,8 @@ export default class IndexController extends Controller { mutation: translationUpdateQuery, variables: { translationId: translation.id, - text, - }, + text + } }); if (response.errors) { @@ -108,8 +108,8 @@ export default class IndexController extends Controller { mutation: translationUpdateQuery, variables: { translationId: translation.id, - text, - }, + text + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/versions/controller.ts b/webapp/app/controllers/logged-in/project/versions.ts similarity index 100% rename from webapp/app/pods/logged-in/project/versions/controller.ts rename to webapp/app/controllers/logged-in/project/versions.ts diff --git a/webapp/app/pods/logged-in/project/versions/edit/controller.ts b/webapp/app/controllers/logged-in/project/versions/edit.ts similarity index 88% rename from webapp/app/pods/logged-in/project/versions/edit/controller.ts rename to webapp/app/controllers/logged-in/project/versions/edit.ts index 26c2b0868..60577b6ab 100644 --- a/webapp/app/pods/logged-in/project/versions/edit/controller.ts +++ b/webapp/app/controllers/logged-in/project/versions/edit.ts @@ -60,11 +60,16 @@ export default class EditController extends Controller { } @action - async update({name, tag}: {name: string; tag: string}) { + async update(args: { + name: string; + tag: string; + copyOnUpdateTranslation: boolean; + }) { this.error = false; - name = name || ''; - tag = tag || ''; + const name = args.name || ''; + const tag = args.tag || ''; + const copyOnUpdateTranslation = args.copyOnUpdateTranslation || false; const id = this.version.id; const response = await this.apolloMutate.mutate({ @@ -73,7 +78,8 @@ export default class EditController extends Controller { id, name, tag, - }, + copyOnUpdateTranslation + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/project/versions/export/controller.ts b/webapp/app/controllers/logged-in/project/versions/export.ts similarity index 98% rename from webapp/app/pods/logged-in/project/versions/export/controller.ts rename to webapp/app/controllers/logged-in/project/versions/export.ts index 1446de079..0f054ebfc 100644 --- a/webapp/app/pods/logged-in/project/versions/export/controller.ts +++ b/webapp/app/controllers/logged-in/project/versions/export.ts @@ -24,7 +24,7 @@ export default class ExportController extends Controller { 'revisionFilter', 'documentFilter', 'documentFormatFilter', - 'orderByFilter', + 'orderByFilter' ]; @tracked @@ -118,7 +118,7 @@ export default class ExportController extends Controller { exportFile() { const blob = new Blob([this.fileRender], { - type: 'charset=utf-8', + type: 'charset=utf-8' }); const filename = `${this.document.path}.${this.fileExtension}`; diff --git a/webapp/app/pods/logged-in/project/versions/new/controller.ts b/webapp/app/controllers/logged-in/project/versions/new.ts similarity index 86% rename from webapp/app/pods/logged-in/project/versions/new/controller.ts rename to webapp/app/controllers/logged-in/project/versions/new.ts index 9255228ab..a1b11d8f2 100644 --- a/webapp/app/pods/logged-in/project/versions/new/controller.ts +++ b/webapp/app/controllers/logged-in/project/versions/new.ts @@ -46,11 +46,16 @@ export default class NewController extends Controller { } @action - async create({name, tag}: {name: string; tag: string}) { + async create(args: { + name: string; + tag: string; + copyOnUpdateTranslation: boolean; + }) { this.error = false; - name = name || ''; - tag = tag || ''; + const name = args.name || ''; + const tag = args.tag || ''; + const copyOnUpdateTranslation = args.copyOnUpdateTranslation || false; const projectId = this.project.id; const response = await this.apolloMutate.mutate({ @@ -59,7 +64,8 @@ export default class NewController extends Controller { projectId, name, tag, - }, + copyOnUpdateTranslation + } }); if (response.errors) { diff --git a/webapp/app/pods/logged-in/projects/controller.ts b/webapp/app/controllers/logged-in/projects.ts similarity index 100% rename from webapp/app/pods/logged-in/projects/controller.ts rename to webapp/app/controllers/logged-in/projects.ts diff --git a/webapp/app/pods/logged-in/projects/new/controller.ts b/webapp/app/controllers/logged-in/projects/new.ts similarity index 96% rename from webapp/app/pods/logged-in/projects/new/controller.ts rename to webapp/app/controllers/logged-in/projects/new.ts index 18e778645..08202031a 100644 --- a/webapp/app/pods/logged-in/projects/new/controller.ts +++ b/webapp/app/controllers/logged-in/projects/new.ts @@ -5,7 +5,7 @@ import RouterService from '@ember/routing/router-service'; import projectCreateQuery, { CreateProjectVariables, - CreateProjectResponse, + CreateProjectResponse } from 'accent-webapp/queries/create-project'; import ApolloMutate from 'accent-webapp/services/apollo-mutate'; import Session from 'accent-webapp/services/session'; @@ -40,8 +40,8 @@ export default class ProjectsNewController extends Controller { mutation: projectCreateQuery, variables: { ...projectAttributes, - name, - }, + name + } }); if (response.errors) { diff --git a/webapp/app/pods/login/controller.ts b/webapp/app/controllers/login.ts similarity index 100% rename from webapp/app/pods/login/controller.ts rename to webapp/app/controllers/login.ts diff --git a/webapp/app/pods/not-found/controller.ts b/webapp/app/controllers/not-found.ts similarity index 100% rename from webapp/app/pods/not-found/controller.ts rename to webapp/app/controllers/not-found.ts diff --git a/webapp/app/formats.js b/webapp/app/formats.js deleted file mode 100644 index 5144c1d6d..000000000 --- a/webapp/app/formats.js +++ /dev/null @@ -1,24 +0,0 @@ -export default { - time: { - hhmmss: { - hour: 'numeric', - minute: 'numeric', - second: 'numeric', - }, - }, - date: { - hhmmss: { - hour: 'numeric', - minute: 'numeric', - second: 'numeric', - }, - }, - number: { - USD: { - style: 'currency', - currency: 'USD', - minimumFractionDigits: 2, - maximumFractionDigits: 2, - }, - }, -}; diff --git a/webapp/app/helpers/string-diff.ts b/webapp/app/helpers/string-diff.ts index 6fdb87eca..23a3036ed 100644 --- a/webapp/app/helpers/string-diff.ts +++ b/webapp/app/helpers/string-diff.ts @@ -42,7 +42,7 @@ const escape = { '"': '"', "'": ''', '`': '`', - '=': '=', + '=': '=' }; const escapeChar = (chr: keyof typeof escape) => { diff --git a/webapp/app/helpers/time-ago-in-words.ts b/webapp/app/helpers/time-ago-in-words.ts index 29f30aa83..c43526f48 100644 --- a/webapp/app/helpers/time-ago-in-words.ts +++ b/webapp/app/helpers/time-ago-in-words.ts @@ -1,16 +1,16 @@ import {isBlank} from '@ember/utils'; import {helper} from '@ember/component/helper'; -import formatDistanceToNow from 'date-fns/formatDistanceToNow'; +import {formatDistanceToNow} from 'date-fns'; import {frCA, enUS} from 'date-fns/locale'; const LOCALES = { 'fr-ca': frCA, - 'en-us': enUS, + 'en-us': enUS } as any; const OPTIONS = { addSuffix: true, - includeSeconds: false, + includeSeconds: false }; const timeAgoInWords = ([date]: [string]) => { @@ -19,7 +19,7 @@ const timeAgoInWords = ([date]: [string]) => { const options = { locale, - ...OPTIONS, + ...OPTIONS }; return formatDistanceToNow(new Date(date), options); diff --git a/webapp/app/instance-initializers/raven-setup.js b/webapp/app/instance-initializers/raven-setup.js index 83885b8a8..fd3b9825c 100644 --- a/webapp/app/instance-initializers/raven-setup.js +++ b/webapp/app/instance-initializers/raven-setup.js @@ -15,5 +15,5 @@ export const initialize = (application) => { export default { name: 'raven-setup', - initialize, + initialize }; diff --git a/webapp/app/locales/en-us.json b/webapp/app/locales/en-us.json index d87a6594f..cbe08d881 100644 --- a/webapp/app/locales/en-us.json +++ b/webapp/app/locales/en-us.json @@ -62,6 +62,7 @@ "slack": "Login with Slack →", "discord": "Login with Discord →", "microsoft": "Login with Microsoft →", + "oidc": "Login with OpenID Connect →", "dummy": "Enter an email and login →" }, "dummy_login_form": { @@ -141,16 +142,26 @@ "reload_button": "Reload" }, "conflicts_filters": { + "advanced_filters_button": "Advanced filters", "total_entries_count": "{count, plural, =0 {No strings to review} =1 {1 string to review} other {# strings to review}}", "reference_default_option_text": "No reference language", "version_default_option_text": "Latest version", "document_default_option_text": "All documents", - "input_placeholder_text": "Search for a string" + "input_placeholder_text": "Search for a string", + "advanced_filters_title": "Advanced filters", + "advanced_filters": { + "empty": "Text is empty (\"\" or null value)", + "not_empty": "Text is not empty", + "added_last_sync": "Strings added last sync", + "commented_on": "Strings with comments", + "conflicted": "To review", + "translated": "To translate" + } }, - "conflicts_items": { - "translations_version_notice": "You are viewing strings to review for the version", + "conflicts_list": { + "translations_version_notice": "You are viewing the strings to be reviewed in the version", "correct_all_button": "Mark all strings as reviewed for this language", - "no_translations": "No strings to review for: {query}", + "no_translations": "No strings found for: {query}", "all_reviewed_title": "All reviewed", "all_reviewed_subtitle": "All strings have been marked as reviewed" }, @@ -206,7 +217,9 @@ "rtl_badge": "RTL", "rtl": "Right to left text", "correct_all_button": "Mark all strings as reviewed", - "uncorrect_all_button": "Put all strings back in review" + "uncorrect_all_button": "Put all strings back in review", + "stats_translated": "translated", + "stats_to_review": "to review" } }, "date_tag": { @@ -404,6 +417,7 @@ "title": "AI Assistant", "help": "Instructions will be available on all string inputs to help you improve your content. The clearer the instructions you give to the assistant, to better the output will be.", "new_button": "Add a new instruction", + "use_platform_label": "Use the platform config if available", "config_key_help_present": "Config key has been set, but the value will never be exposed via the API.", "config_key_help_not_present": "Once saved, the config key will never be shown.", "delete_confirm": "Are you sure you want to delete this instruction?" @@ -428,7 +442,8 @@ "jipt": { "title": "Just In Place Translations", "integration_help": "This steps will walk you throught the setup you need to have in your project to translate your strings in your browser, inside your project.", - "integration_link": "See official guides on how to integrate live editing in your website ->", + "demo_help": "See a static HTML page that renders the first string of your project in different context. The Accent live integration can be seen at the bottom of the page. Note that the live example use your real project’s data.", + "demo_link": "Open example →", "add_language_title": "Add the pseudo language to your project", "add_language_image_1": "Here is a language like you already have in your project:", "add_language_image_2": "Here is the pseudo language named \"accent\" that act as a normal translation in your project:", @@ -472,6 +487,7 @@ "developer_text": "Can make file operations: Sync, add translations and preview those operations in the UI.", "owner_text": "With the same roles as the admin, the owners are people who the project belongs to.", "reviewer_text": "Can do every tasks except those listed in the above roles. Review, update strings, comments, etc.", + "translator_text": "Translate strings and make updates on documents. The translator is not included in the review process.", "title": "Collaborators" }, "collaborators_item": { @@ -494,9 +510,19 @@ "last_executed_at": "Last executed at:", "data": { "azure_storage_container_sas": "SAS URL", + "aws_s3_bucket": "Bucket name", + "aws_s3_path_prefix": "Path prefix", + "aws_s3_region": "Region", + "aws_s3_access_key_id": "Access Key ID", + "aws_s3_secret_access_key": "Secret Access Key", + "aws_s3_minimum_policy": "Minimum policy", "url": "URL" }, "empty_description": { + "aws_s3": { + "title": "AWS S3", + "text": "Upload your files in Amazon Web Services S3 to serve them directly in your application" + }, "azure_storage_container": { "title": "Azure Storage Container", "text": "Upload your files in Azure Storage Container to serve them directly in your application" @@ -510,21 +536,33 @@ "text": "Post your project’s notifications in Slack" } }, - "token_how": "How?", - "github_webhook_url_how": "How?", - "github_webhook_accent_cli_1": "You need to have a valid accent-cli setup for the hook to work.", - "github_webhook_accent_cli_2": "The accent.json file at the root of your project will be used.", - "github_webhook_url": "Make sure to add the webhook in your project’s settings", "webhook_url_how": "How to obtain a webhook URL?", "events": { "title": "Which events would you like to trigger this webhook?", "options": { "sync": "Sync with any changes", "new_conflicts": "New strings to review", - "complete_review": "Project is 100% reviewed" + "complete_review": "Project is 100% reviewed", + "integration_execute_azure_storage_container": "Files uploaded to Azure Storage Container" } }, "execute": { + "aws_s3": { + "title": "Upload files to AWS S3", + "cancel_button": "Cancel", + "error": "Error while pushing to your bucket. Check your credentials and try again.", + "push_button": "Upload", + "bucket": "Bucket", + "path_prefix": "Path prefix", + "submit_confirm": "Uploading files to AWS S3 can override existing files if you upload for the latest or an already uploaded version tag.", + "target_version": { + "label": "Target Version", + "options": { + "latest": "Latest Version", + "specific": "Specific Version" + } + } + }, "azure_storage_container": { "title": "Upload files to Azure Storage Container", "cancel_button": "Cancel", @@ -661,6 +699,8 @@ "error": "Invalid version", "cancel_button": "Cancel", "name_label": "Name:", + "copy_on_update_translation_help": "Everytime a translation in this version is updated, the same text will be applied to the latest translation. This will make sure that a fix in frozen version will also be present in the latest version without any required actions by the developer or the reviewer.", + "copy_on_update_translation_label": "Copy on update translation", "tag_label": "Tag:", "save_button": "Create" }, @@ -668,6 +708,8 @@ "title": "Update version", "error": "Invalid version", "cancel_button": "Cancel", + "copy_on_update_translation_help": "Everytime a translation in this version is updated, the same text will be applied to the latest translation. This will make sure that a fix in frozen version will also be present in the latest version without any required actions by the developer or the reviewer.", + "copy_on_update_translation_label": "Copy on update translation", "name_label": "Name:", "tag_label": "Tag:", "save_button": "Update" @@ -721,6 +763,7 @@ "translations_link_title": "All strings", "lint_link_title": "Checks", "conflicts_link_title": "Review", + "translate_link_title": "Translate", "dashboard_link_title": "Dashboard", "settings_link_title": "Settings", "sync_link_title": "Files", @@ -999,7 +1042,8 @@ "integration_services": { "DISCORD": "Discord", "SLACK": "Slack", - "AZURE_STORAGE_CONTAINER": "Azure Storage Container" + "AZURE_STORAGE_CONTAINER": "Azure Storage Container", + "AWS_S3": "AWS S3" }, "search_input_placeholder_text": "Search for a string…" }, @@ -1090,7 +1134,9 @@ "unsupported_source_and_target": "The string could not be translated: Unsupported source {source} and target {target}" }, "correct_error": "The string could not be marked as reviewed", - "correct_success": "The string as been marked as reviewed with success" + "correct_success": "The string as been marked as reviewed with success", + "update_error": "The string could not be updated", + "update_success": "The string has been updated with success" } }, "edit": { @@ -1105,6 +1151,8 @@ "collaborator_update_success": "The collaborator has been updated with success", "integration_add_error": "The integration could not be added", "integration_add_success": "The integration has been added with success", + "integration_execute_error": "The integration could not be executed", + "integration_execute_success": "The integration has been executed with success", "integration_update_error": "The integration could not be updated", "integration_update_success": "The integration has been updated with success", "integration_remove_error": "The integration could not be removed", diff --git a/webapp/app/locales/fr-ca.json b/webapp/app/locales/fr-ca.json index 382543322..1d413b944 100644 --- a/webapp/app/locales/fr-ca.json +++ b/webapp/app/locales/fr-ca.json @@ -78,6 +78,7 @@ "slack": "Se connecter avec Slack →", "discord": "Se connecter avec Discord →", "microsoft": "Se connecter avec Microsoft →", + "oidc": "Connectez-vous avec OIDC →", "dummy": "Entrez un courriel et connectez-vous →" }, "dummy_login_form": { @@ -157,13 +158,23 @@ "reload_button": "Recharger" }, "conflicts_filters": { + "advanced_filters_button": "Filtres avancés", "total_entries_count": "{count, plural, =0 {Aucune chaîne à vérifier} =1 {1 chaîne à vérifier} other {# chaînes à vérifier}}", "reference_default_option_text": "Pas de langue de référence", "version_default_option_text": "Dernière version", "document_default_option_text": "Tous les documents", - "input_placeholder_text": "Rechercher une chaîne" + "input_placeholder_text": "Rechercher une chaîne", + "advanced_filters_title": "Filtres avancés", + "advanced_filters": { + "empty": "Le texte est vide (\"\" ou valeur nulle)", + "not_empty": "Le texte n’est pas vide", + "added_last_sync": "Ajoutées à la dernière synchronisation", + "commented_on": "Chaînes avec commentaires", + "conflicted": "À réviser", + "translated": "À traduire" + } }, - "conflicts_items": { + "conflicts_list": { "translations_version_notice": "Vous consultez les textes à réviser de la version", "correct_all_button": "Marquer toutes les chaînes comme révisées pour cette langue", "no_translations": "Aucune chaîne à examiner pour : {query}", @@ -222,7 +233,9 @@ "rtl_badge": "RTL", "rtl": "Texte de droite à gauche", "correct_all_button": "Marquer toutes les chaînes comme vérifiées", - "uncorrect_all_button": "Remettre toutes les chaînes à réviser" + "uncorrect_all_button": "Remettre toutes les chaînes à réviser", + "stats_translated": "traduits", + "stats_to_review": "à réviser" } }, "date_tag": { @@ -392,6 +405,7 @@ "prompts": { "title": "Assistant IA", "help": "Des instructions seront disponibles sur chaque entrée de chaîne pour vous aider à améliorer votre contenu. Plus les instructions que vous donnez à l’assistant sont claires, meilleur sera le résultat.", + "use_platform_label": "Utilisez la configuration de la plate-forme si disponible", "new_button": "Ajouter une nouvelle instruction", "config_key_help_present": "La clé de configuration a été définie, mais la valeur ne sera jamais exposée via l’API.", "config_key_help_not_present": "Une fois enregistrée, la clé de configuration ne sera jamais affichée.", @@ -444,7 +458,8 @@ "jipt": { "title": "Traductions JIPT", "integration_help": "Cette étape vous guidera à travers la configuration dont vous avez besoin dans votre projet pour traduire vos chaînes dans votre navigateur, à l’intérieur de votre projet.", - "integration_link": "Voir les guides officiels sur la façon d’intégrer l’édition en direct dans votre site Web ->", + "demo_help": "Voir une page HTML statique qui montre la première chaîne de votre projet dans un contexte différent. L’intégration d’Accent est visible en bas de la page. Notez que l’exemple utilise les vraies données de votre projet.", + "demo_link": "Ouvrir l’exemple →", "add_language_title": "Ajoutez le pseudo langage à votre projet", "add_language_image_1": "Voici un langage comme vous en avez déjà dans votre projet :", "add_language_image_2": "Voici le pseudo langage nommé \"accent\" qui agit comme une traduction normale dans votre projet :", @@ -488,6 +503,7 @@ "developer_text": "Peut effectuer des opérations sur les fichiers : synchroniser, ajouter des traductions et prévisualiser ces opérations dans l’interface utilisateur.", "owner_text": "Avec les mêmes rôles que l’administrateur, les propriétaires sont les personnes à qui appartient le projet.", "reviewer_text": "Peut effectuer toutes les tâches sauf celles énumérées dans les rôles ci-dessus. Réviser, mettre à jour les chaînes, les commentaires, etc.", + "translator_text": "Traduire des chaînes et effectuez des mises à jour sur les fichiers. Le traducteur n’est pas inclus dans le processus de révision.", "title": "Collaborateurs" }, "collaborators_item": { @@ -510,9 +526,19 @@ "delete": "Supprimer", "data": { "azure_storage_container_sas": "SAS URL", + "aws_s3_bucket": "Nom du bucket", + "aws_s3_path_prefix": "Préfix", + "aws_s3_region": "Région", + "aws_s3_access_key_id": "Access Key ID", + "aws_s3_secret_access_key": "Secret Access Key", + "aws_s3_minimum_policy": "Permissions minimums", "url": "URL" }, "empty_description": { + "aws_s3": { + "title": "AWS S3", + "text": "Téléchargez vos fichiers dans Amazon Web Service S3 pour les servir directement dans votre application" + }, "azure_storage_container": { "title": "Azure Storage Container", "text": "Téléchargez vos fichiers dans Azure Storage Container pour les servir directement dans votre application" @@ -526,21 +552,33 @@ "text": "Publiez les notifications de votre projet dans Slack" } }, - "token_how": "Comment?", - "github_webhook_url_how": "Comment?", - "github_webhook_accent_cli_1": "Vous devez avoir une configuration accent-cli valide pour le crochet au travail.", - "github_webhook_accent_cli_2": "Le accent.json fichier à la racine de votre projet sera utilisé.", - "github_webhook_url": "Assurez-vous d’ajouter le webhook dans les paramètres de votre projet", "webhook_url_how": "Comment obtenir une URL de webhook ?", "events": { "title": "Quels événements souhaitez-vous déclencher ce webhook ?", "options": { "sync": "Synchroniser avec n’importe quelles modifications", "new_conflicts": "Nouvelles chaîne à réviser", - "complete_review": "Le projet est revu à 100%" + "complete_review": "Le projet est revu à 100%", + "integration_execute_azure_storage_container": "Fichiers téléversés sur Azure Storage Container" } }, "execute": { + "aws_s3": { + "title": "Publier sur AWS S3", + "cancel_button": "Annuler", + "error": "Une erreur s’est produite lors de la publication sur Azure. Veuillez vérifier vos informations d’identification et réessayer.", + "push_button": "Publier", + "bucket": "Nom du bucket", + "path_prefix": "Préfix", + "submit_confirm": "La publication de fichiers vers AWS S3 peut remplacer les fichiers existants si vous publiez la dernière version ou une version déjà publiée.", + "target_version": { + "label": "Version Cible", + "options": { + "latest": "Dernière version", + "specific": "Version spécifique" + } + } + }, "azure_storage_container": { "title": "Publier sur Azure Storage Container", "cancel_button": "Annuler", @@ -675,6 +713,8 @@ "title": "Nouvelle version", "text": "La création d’une version crée un instantané de toutes les chaînes actives (révisées ou non) pour être visible avec la certitude qu’il restera intact. Cela peut être utile lors de la maintenance de plusieurs versions de la même application.", "error": "Version invalide", + "copy_on_update_translation_help": "Chaque fois qu'une traduction dans cette version est mise à jour, le même texte sera appliqué à la traduction source (dernière version). Cela garantira qu'un correctif dans la version sera également présent dans la dernière version sans aucune action requise de la part du développeur ou le traducteur.", + "copy_on_update_translation_label": "Copie sur mise à jour de la traduction", "cancel_button": "Annuler", "name_label": "Nom:", "tag_label": "Étiquette:", @@ -683,6 +723,8 @@ "version_update_form": { "title": "Version de mise à jour", "error": "Version invalide", + "copy_on_update_translation_help": "Chaque fois qu'une traduction dans cette version est mise à jour, le même texte sera appliqué à la traduction source (dernière version). Cela garantira qu'un correctif dans la version sera également présent dans la dernière version sans aucune action requise de la part du développeur ou le traducteur.", + "copy_on_update_translation_label": "Copie sur mise à jour de la traduction", "cancel_button": "Annuler", "name_label": "Nom:", "tag_label": "Étiquette:", @@ -720,6 +762,7 @@ "manage_languages_link_title": "Langues", "translations_link_title": "Chaînes", "lint_link_title": "Vérifier", + "translate_link_title": "Traduire", "conflicts_link_title": "Réviser", "dashboard_link_title": "Tableau de bord", "settings_link_title": "Paramètres", @@ -853,7 +896,7 @@ "source_translation": "Voir la dernière version de la chaîne", "correct_button": "Mettre à jour et marquer comme révisé", "previous_text": "Texte précédent :", - "uncorrect_button": "Remettre à révisée", + "uncorrect_button": "Remettre à réviser", "uneditable": "Le texte n’est pas modifiable, car il a été marqué comme révisé", "update_text": "Mettre à jour le texte", "form": { @@ -999,7 +1042,8 @@ "integration_services": { "DISCORD": "Discord", "SLACK": "Slack", - "AZURE_STORAGE_CONTAINER": "Azure Storage Container" + "AZURE_STORAGE_CONTAINER": "Azure Storage Container", + "AWS_S3": "AWS S3" }, "search_input_placeholder_text": "Rechercher une chaîne…" }, @@ -1085,6 +1129,8 @@ "revision_correct_error": "Une erreur s’est produite lors du marquage de toutes les chaînes dans cette langue comme révisées", "correct_error": "Impossible de marquer la chaîne comme vérifiée", "correct_success": "La chaîne a été marquée comme révisée avec succès", + "update_error": "La chaîne n’a pas pu être mise à jour", + "update_success": "La chaîne a été mise à jour avec succès", "translate_provider_error": "La chaîne n’a pas pu être traduite : {provider}", "translate_error": { "unsupported_target": "La chaîne n’a pas pu être traduite : Langue non prise en charge {target}", @@ -1121,6 +1167,8 @@ "integration_update_success": "L’intégration a été mise à jour avec succès", "integration_remove_error": "L’intégration n’a pas pu être supprimée", "integration_remove_success": "L’intégration a été supprimée avec succès", + "integration_execute_error": "L’intégration n’a pas pu être exécutée", + "integration_execute_success": "L’intégration a été exécutée avec succès", "machine_translations_config_error": "La configuration n’a pas pu être mise à jour", "machine_translations_config_success": "La configuration a été mise à jour avec succès", "update_error": "Le projet n’a pas pu être mis à jour", diff --git a/webapp/app/pods/components/acc-emoji-picker/component.ts b/webapp/app/pods/components/acc-emoji-picker/component.ts deleted file mode 100644 index 10bd49adf..000000000 --- a/webapp/app/pods/components/acc-emoji-picker/component.ts +++ /dev/null @@ -1,37 +0,0 @@ -import Component from '@glimmer/component'; -import {action} from '@ember/object'; -import {EmojiButton} from '@joeattardi/emoji-button'; - -interface Args { - onPicked: () => string; -} - -export default class EmojiPicker extends Component { - picker: EmojiButton; - element: HTMLDivElement; - - clickCallback = this.onClick.bind(this); - - @action - initializePicker(element: HTMLDivElement) { - this.element = element; - - this.picker = new EmojiButton({ - showPreview: false, - }); - - this.picker.on('emoji', this.args.onPicked); - } - - @action - destroyPicker() { - this.picker.off('emoji', this.args.onPicked); - } - - @action - onClick() { - this.picker.isPickerVisible() - ? this.picker.hidePicker() - : this.picker.showPicker(this.element); - } -} diff --git a/webapp/app/pods/components/acc-emoji-picker/template.hbs b/webapp/app/pods/components/acc-emoji-picker/template.hbs deleted file mode 100644 index 6a41f2522..000000000 --- a/webapp/app/pods/components/acc-emoji-picker/template.hbs +++ /dev/null @@ -1,5 +0,0 @@ - \ No newline at end of file diff --git a/webapp/app/pods/components/conflicts-filters/styles.scss b/webapp/app/pods/components/conflicts-filters/styles.scss deleted file mode 100644 index 6f4282b60..000000000 --- a/webapp/app/pods/components/conflicts-filters/styles.scss +++ /dev/null @@ -1,66 +0,0 @@ -.filters-wrapper { - display: flex; - justify-content: space-between; - align-items: flex-start; -} - -.filters-content { - flex-grow: 1; - margin-right: 15px; -} - -.queryForm { - position: relative; -} - -.totalEntries { - flex-shrink: 0; - margin-top: 10px; - color: var(--color-grey); - font-size: 12px; -} - -.search-icon { - position: absolute; - top: 50%; - margin-top: -10px; - left: 7px; - width: 20px; - height: 20px; - stroke: #b7b7b7; -} - -.input { - @extend %textInput; - width: 100%; - padding: 7px 7px 7px 30px; - font-family: var(--font-primary); - font-size: 14px; - color: var(--color-black); - - &:focus { - box-shadow: inset 0 1px 2px rgba(#000, 0.1), 0 1px 2px var(--shadow-color); - } - - &::placeholder { - color: var(--color-grey); - } -} - -.totalEntries { - margin-top: 10px; - color: var(--color-grey); - font-size: 12px; -} - -@media (max-width: 440px) { - .filters-wrapper { - flex-direction: column; - } - - .queryForm, - .filters-content { - width: 100%; - margin-right: 0; - } -} diff --git a/webapp/app/pods/components/conflicts-filters/template.hbs b/webapp/app/pods/components/conflicts-filters/template.hbs deleted file mode 100644 index ef6dcbf4c..000000000 --- a/webapp/app/pods/components/conflicts-filters/template.hbs +++ /dev/null @@ -1,58 +0,0 @@ -
-
-
-
-
- {{inline-svg '/assets/search.svg' local-class='search-icon'}} - - -
- -
-
- {{#if this.showDocumentsSelect}} -
-
- -
-
- {{/if}} - - {{#if this.showVersionsSelect}} -
-
- -
-
- {{/if}} -
-
-
- - {{#if @meta.totalEntries}} - - {{t 'components.conflicts_filters.total_entries_count' count=@meta.totalEntries}} - - {{/if}} -
-
-
\ No newline at end of file diff --git a/webapp/app/pods/components/conflicts-list/component.ts b/webapp/app/pods/components/conflicts-list/component.ts deleted file mode 100644 index dff789463..000000000 --- a/webapp/app/pods/components/conflicts-list/component.ts +++ /dev/null @@ -1,27 +0,0 @@ -import Component from '@glimmer/component'; - -interface Args { - permissions: Record; - project: any; - conflicts: any; - version: any; - versions: any[]; - query: any; - onCorrect: (conflict: any, textInput: string) => Promise; - onCopyTranslation: ( - text: string, - sourceLanguageSlug: string, - targetLanguageSlug: string - ) => void; -} - -export default class ConflictsItems extends Component { - get currentVersion() { - if (!this.args.versions) return; - if (!this.args.version) return; - - return this.args.versions.find( - (version) => version.id === this.args.version - ); - } -} diff --git a/webapp/app/pods/components/conflicts-list/item/component.ts b/webapp/app/pods/components/conflicts-list/item/component.ts deleted file mode 100644 index bdcc9e20e..000000000 --- a/webapp/app/pods/components/conflicts-list/item/component.ts +++ /dev/null @@ -1,182 +0,0 @@ -import {action} from '@ember/object'; -import {empty} from '@ember/object/computed'; -import Component from '@glimmer/component'; -import parsedKeyProperty from 'accent-webapp/computed-macros/parsed-key'; -import {dropTask} from 'ember-concurrency-decorators'; -import {tracked} from '@glimmer/tracking'; -import {MutationResponse} from 'accent-webapp/services/apollo-mutate'; - -interface Conflict { - id: string; - key: string; - conflictedText: string; - correctedText: string; - revision: { - name: string | null; - slug: string | null; - rtl: boolean | null; - isMaster: boolean; - language: { - name: string; - slug: string; - rtl: boolean; - }; - }; - relatedTranslations: Array<{ - id: string; - correctedText: string; - revision: { - isMaster: boolean; - }; - }>; -} - -interface Args { - permissions: Record; - index: number; - project: any; - prompts: any[]; - conflict: Conflict; - onCorrect: (conflict: any, textInput: string) => Promise; - onCopyTranslation: ( - text: string, - sourceLanguageSlug: string | null, - targetLanguageSlug: string - ) => Promise<{text: string | null}>; -} - -export default class ConflictItem extends Component { - @empty('args.conflict.conflictedText') - emptyPreviousText: boolean; - - @tracked - textInput = this.args.conflict.correctedText; - - @tracked - loading = false; - - @tracked - error = false; - - @tracked - resolved = false; - - @tracked - inputDisabled = false; - - conflictKey = parsedKeyProperty(this.args.conflict.key); - textOriginal = this.args.conflict.correctedText; - - get relatedTranslations() { - const masterConflict = this.args.conflict.relatedTranslations.find( - (translation) => translation.revision.isMaster - ); - if (!masterConflict) return []; - - return this.args.conflict.relatedTranslations.filter((translation) => { - return ( - translation.id === masterConflict.id || - translation.correctedText !== masterConflict.correctedText - ); - }); - } - - get showTextDiff() { - if (!this.args.conflict.conflictedText) return false; - - return this.textInput !== this.args.conflict.conflictedText; - } - - get showOriginalButton() { - return this.textInput !== this.textOriginal; - } - - get revisionName() { - return ( - this.args.conflict.revision.name || - this.args.conflict.revision.language.name - ); - } - - get revisionSlug() { - return ( - this.args.conflict.revision.slug || - this.args.conflict.revision.language.slug - ); - } - - get revisionTextDirRtl() { - return this.args.conflict.revision.rtl !== null - ? this.args.conflict.revision.rtl - : this.args.conflict.revision.language.rtl; - } - - @action - changeTranslationText(text: string) { - this.textInput = text; - } - - @action - setOriginalText() { - this.textInput = this.textOriginal; - } - - @action - onUpdatingText() { - this.inputDisabled = true; - } - - @action - onUpdateText(value: string) { - this.textInput = value; - this.inputDisabled = false; - } - - @dropTask - *copyTranslationTask(text: string, sourceLanguageSlug: string): any { - this.inputDisabled = true; - - const copyTranslation = yield this.args.onCopyTranslation( - text, - sourceLanguageSlug, - this.revisionSlug - ); - - this.inputDisabled = false; - - if (copyTranslation.text) { - this.textInput = copyTranslation.text; - } - } - - @action - async correct() { - this.onLoading(); - - const response = await this.args.onCorrect( - this.args.conflict, - this.textInput - ); - - if (response.errors) { - this.onError(); - } else { - this.onCorrectSuccess(); - } - } - - private onLoading() { - this.error = false; - this.loading = true; - } - - private onError() { - this.error = true; - this.loading = false; - } - - private onCorrectSuccess() { - this.resolved = true; - this.loading = false; - } -} diff --git a/webapp/app/pods/components/conflicts-list/item/related-translation/component.ts b/webapp/app/pods/components/conflicts-list/item/related-translation/component.ts deleted file mode 100644 index 778ef6d65..000000000 --- a/webapp/app/pods/components/conflicts-list/item/related-translation/component.ts +++ /dev/null @@ -1,42 +0,0 @@ -import Component from '@glimmer/component'; -import {action} from '@ember/object'; - -interface Args { - project: any; - translation: any; - onCopyTranslation: (text: string, languageSlug: string) => void; -} - -const MAX_TEXT_LENGTH = 600; - -export default class ConflictItemRelatedTranslation extends Component { - get text() { - const text = this.args.translation.correctedText; - - if (text.length < MAX_TEXT_LENGTH) return text; - - return `${text.substring(0, MAX_TEXT_LENGTH - 1)}…`; - } - - get revisionName() { - return ( - this.args.translation.revision.slug || - this.args.translation.revision.language.slug - ); - } - - @action - translate() { - this.args.onCopyTranslation( - this.args.translation.correctedText, - this.revisionSlug - ); - } - - private get revisionSlug() { - return ( - this.args.translation.revision.slug || - this.args.translation.revision.language.slug - ); - } -} diff --git a/webapp/app/pods/components/conflicts-list/item/related-translation/styles.scss b/webapp/app/pods/components/conflicts-list/item/related-translation/styles.scss deleted file mode 100644 index a1eb07f87..000000000 --- a/webapp/app/pods/components/conflicts-list/item/related-translation/styles.scss +++ /dev/null @@ -1,107 +0,0 @@ -.item { - display: flex; - align-items: flex-start; - flex-direction: row; - padding: 5px 0 3px; - color: var(--color-grey); - font-size: 13px; -} - -.item-link { - display: inline-flex; - place-items: center; - flex-shrink: 0; - margin-right: 2px; - color: var(--color-black); - opacity: 0.8; - text-decoration: none; - padding: 1px 5px; - background: var(--background-light-highlight); - border-radius: var(--border-radius); - font-size: 11px; - transition: 0.2s ease-in-out; - transition-property: opacity; - - &:focus, - &:hover { - opacity: 1; - } -} - -.item-text { - width: 100%; - display: flex; - align-items: flex-start; -} - -.item-text-content { - display: block; - white-space: pre-wrap; - margin: 0 0 0 5px; - color: var(--color-black); - opacity: 0.8; - font-size: 12px; -} - -.item-actions { - margin-left: 6px; - flex-shrink: 0; - flex-grow: 0; -} - -.item-actions-link { - display: inline-flex; - align-items: center; - justify-content: center; - border-radius: 50%; - height: 17px; - width: 17px; - padding: 3px; - background: transparent; - color: var(--color-black); - background: var(--color-primary-opacity-20); - line-height: 1; -} - -.item-actions-link--loading { - pointer-events: none; - background: transparent; - - &:focus, - &:hover { - background: transparent; - } -} - -.item-actions-flag--link, -.item-actions-link { - cursor: pointer; - - &:focus, - &:hover { - background: var(--background-light-highlight); - } -} - -.item-actions-flag { - position: relative; - top: -1px; - padding: 0 5px; - margin-left: 0; - margin-right: 5px; - width: auto; - border-radius: 5px; - border: 1px solid var(--background-light-highlight); - font-size: 10px; - text-transform: uppercase; - text-decoration: none; - - &:last-of-type { - margin-right: 0; - } -} - -.item-actions-icon { - width: 10px; - height: 10px; -} diff --git a/webapp/app/pods/components/conflicts-list/item/related-translation/template.hbs b/webapp/app/pods/components/conflicts-list/item/related-translation/template.hbs deleted file mode 100644 index 794040a87..000000000 --- a/webapp/app/pods/components/conflicts-list/item/related-translation/template.hbs +++ /dev/null @@ -1,17 +0,0 @@ -
- - {{this.revisionName}} - - -
- {{this.text}} - -
- {{#if (get @permissions 'machine_translations_translate')}} - - {{/if}} -
-
-
\ No newline at end of file diff --git a/webapp/app/pods/components/conflicts-list/item/styles.scss b/webapp/app/pods/components/conflicts-list/item/styles.scss deleted file mode 100644 index f873aaa5c..000000000 --- a/webapp/app/pods/components/conflicts-list/item/styles.scss +++ /dev/null @@ -1,341 +0,0 @@ -.root { - &:nth-child(even) { - background: var(--background-light); - } -} - -.conflict-item { - padding: 10px; - margin-bottom: 15px; - border-radius: var(--border-radius); - - &:hover { - .form-helpers { - pointer-events: all; - opacity: 1; - } - } -} - -.revert-button { - :global(.label) { - padding-left: 3px; - padding-right: 3px; - } -} - -.item-details__column { - position: relative; -} - -.item-details { - display: flex; - flex-direction: column; - - &[data-dir='rtl'] { - .item-details__column { - align-items: flex-end; - } - .item-details__column:first-of-type { - margin-right: 0; - margin-left: 15px; - } - - .item-key { - margin-right: 0; - margin-left: 15px; - flex-direction: row-reverse; - } - - .item-key-prefix::before { - content: ''; - } - - .item-key-prefix::after { - content: '/'; - } - } -} - -.item-details__column { - display: flex; - flex-direction: column; - align-items: flex-start; -} - -.item-details__column:first-of-type { - margin-right: 15px; -} - -.item-detail-conflict { - display: flex; - width: 100%; -} - -.item-detail-conflict-actions { - margin-left: 6px; - flex-shrink: 0; - flex-grow: 1; -} - -.item-detail-conflict-actions-flag, -.item-detail-conflict-actions-link { - display: inline-flex; - align-items: center; - justify-content: center; - margin-left: 3px; - border-radius: 50%; - width: 24px; - height: 24px; - background: transparent; - color: var(--color-black); - line-height: 1; -} - -.item-detail-conflict-actions-link--loading { - pointer-events: none; - background: transparent; - - &:focus, - &:hover { - background: transparent; - } -} - -.item-detail-conflict-actions-flag--link, -.item-detail-conflict-actions-link { - cursor: pointer; - - &:focus, - &:hover { - background: var(--background-light-highlight); - } -} - -.item-detail-conflict-actions-flag { - position: relative; - top: -1px; - padding: 0 5px; - margin-left: 0; - margin-right: 5px; - width: auto; - border-radius: 5px; - border: 1px solid var(--background-light-highlight); - font-size: 10px; - text-transform: uppercase; - text-decoration: none; - - &:last-of-type { - margin-right: 0; - } -} - -.item-detail-conflict-actions-flag--link { - &:focus, - &:hover { - color: var(--content-background); - border-color: var(--color-primary); - background: var(--color-primary); - } -} - -.item-detail-conflict-actions-icon { - width: 12px; - height: 12px; -} - -.conflict-item.resolved { - padding: 5px 10px; - margin: 5px 0; - background: hsl( - var(--color-green-hue), - var(--color-green-saturation), - var(--color-highlight-lighteness) - ); - - .item-key-prefix { - opacity: 0.7; - color: var(--color-green); - } - - .item-key { - margin-bottom: 0; - font-size: 12px; - color: var(--color-green); - } -} - -.conflict-item.errored { - .textInput { - border-color: var(--color-error); - } -} - -.error { - font-size: 12px; - font-weight: bold; - color: var(--color-error); -} - -.item-key-prefix { - display: inline-flex; - font-size: 11px; - color: #959595; - gap: 6px; - font-weight: 300; - - &::before { - content: '/'; - } -} - -.item-key { - display: flex; - align-items: center; - gap: 5px; - margin-bottom: 12px; - font-family: var(--font-monospace); - word-break: break-all; - transition: 0.2s ease-in-out; - transition-property: color; - margin-right: 15px; - line-height: 1.5; - font-size: 11px; - color: var(--color-primary); -} - -.key { - text-decoration: none; -} - -.textResolved { - display: flex; - align-items: flex-start; -} - -.textResolved-content { - flex-grow: 1; -} - -.textResolved-text { - margin-bottom: 5px; - font-size: 13px; -} - -.textDiff { - padding: 8px 0 3px; - color: var(--color-grey); -} - -.textDiff { - white-space: pre-wrap; - word-break: break-word; - font-size: 12px; -} - -.button-submit { - display: flex; - justify-content: flex-end; - position: absolute; - gap: 10px; - top: 10px; - right: 10px; - z-index: 3; - - &[data-dir='rtl'] { - right: auto; - left: 7px; - flex-direction: row-reverse; - } -} - -.item-revision { - flex-shrink: 0; - margin-right: 4px; - color: var(--color-black); - opacity: 0.6; - text-decoration: none; - font-weight: bold; - font-size: 12px; - text-align: right; - - &:focus, - &:hover { - opacity: 1; - color: var(--color-primary); - } -} - -.item-revision { - opacity: 1; - padding: 4px 0 3px; - margin: 0 9px 0 0; - font-size: 13px; -} - -.textInput { - flex-grow: 1; - flex-shrink: 0; - width: 100%; -} - -.item-text { - display: block; - width: 100%; - color: var(--color-black); - line-height: 1.4; - padding: 3px 10px 10px 0; - font-size: 13px; - line-height: 1.6; - cursor: pointer; - word-break: break-word; - - &:focus, - &:hover { - outline: none; - opacity: 0.8; - } -} - -.conflictedText-references { - width: 100%; - padding-left: 4px; -} - -.conflictedText-references-conflicted { - display: flex; - align-items: flex-start; - margin-top: 7px; - font-size: 12px; - color: var(--color-black); -} - -.form-helpers { - pointer-events: none; - opacity: 0; - position: relative; - z-index: 1; - transition: 0.2s ease-in-out; - transition-property: opacity; -} - -.conflictedText-references-conflicted-label { - display: flex; - align-items: center; - justify-content: flex-end; - flex-shrink: 0; - margin-right: 10px; - margin-top: 4px; - font-weight: 500; - font-size: 11px; - text-align: right; -} - -.conflictedText-references-conflicted-value { - white-space: pre-wrap; -} - -.conflictedText-references-conflicted-icon { - width: 12px; - height: 12px; -} diff --git a/webapp/app/pods/components/conflicts-list/item/template.hbs b/webapp/app/pods/components/conflicts-list/item/template.hbs deleted file mode 100644 index 5620604a2..000000000 --- a/webapp/app/pods/components/conflicts-list/item/template.hbs +++ /dev/null @@ -1,123 +0,0 @@ -
-
-
  • - {{#if this.resolved}} -
    - - - {{this.conflictKey.value}} - - {{#if this.conflictKey.prefix}} - {{this.conflictKey.prefix}} - {{else}} - {{@conflict.document.path}} - {{/if}} - - - - -
    - {{#if this.error}} -
    - {{t 'components.conflict_item.uncorrect_error_text'}} -
    - {{/if}} -
    -
    - {{else}} -
    -
    - - - {{this.conflictKey.value}} - - {{#if this.conflictKey.prefix}} - {{this.conflictKey.prefix}} - {{else}} - {{@conflict.document.path}} - {{/if}} - - - - - {{#if this.error}} -
    - {{t 'components.conflict_item.correct_error_text'}} -
    - {{/if}} -
    -
    -
    - - {{#component form.submit}} -
    - {{#if this.showOriginalButton}} - - {{inline-svg '/assets/revert.svg' class='button-icon'}} - - {{/if}} - -
    - -
    - - {{#if (get @permissions 'correct_translation')}} - - {{inline-svg '/assets/check.svg' class='button-icon'}} - - {{/if}} -
    - {{/component}} -
    -
    - -
    - {{#if this.showTextDiff}} -
    - - {{inline-svg '/assets/diff.svg' local-class='conflictedText-references-conflicted-icon'}} - - -
    {{string-diff this.textInput @conflict.conflictedText}}
    -
    - {{/if}} - - {{#if @conflict.relatedTranslations}} - {{#each this.relatedTranslations key='id' as |relatedTranslation|}} - - {{/each}} - {{/if}} -
    -
    -
    - {{/if}} -
  • -
    -
    \ No newline at end of file diff --git a/webapp/app/pods/components/conflicts-list/template.hbs b/webapp/app/pods/components/conflicts-list/template.hbs deleted file mode 100644 index 6e25ce219..000000000 --- a/webapp/app/pods/components/conflicts-list/template.hbs +++ /dev/null @@ -1,35 +0,0 @@ -
      - {{#if this.currentVersion}} -
      - - {{t 'components.conflicts_items.translations_version_notice'}} - {{this.currentVersion.tag}} -
      - {{/if}} - {{#each @conflicts key='id' as |conflict index|}} - - {{else if @query}} - - {{else}} -
      - - -
      - {{t 'components.conflicts_items.all_reviewed_title'}} -
      - -
      - {{t 'components.conflicts_items.all_reviewed_subtitle'}} -
      -
      - {{/each}} -
    \ No newline at end of file diff --git a/webapp/app/pods/components/conflicts-page/component.ts b/webapp/app/pods/components/conflicts-page/component.ts deleted file mode 100644 index 5bd84c7cd..000000000 --- a/webapp/app/pods/components/conflicts-page/component.ts +++ /dev/null @@ -1,29 +0,0 @@ -import Component from '@glimmer/component'; - -interface Args { - project: any; - revision: any; - translations: any; - isLoading: boolean; - showLoading: boolean; - document: any; - version: any; - permissions: Record; - documents: any; - versions: any; - showSkeleton: boolean; - query: any; - reference: any; - referenceRevision: any; - referenceRevisions: any; - onCorrect: () => void; - onCopyTranslation: () => void; - onCorrectAll: () => void; - onSelectPage: () => void; - onChangeDocument: () => void; - onChangeVersion: () => void; - onChangeReference: () => void; - onChangeQuery: () => void; -} - -export default class ConflictsPage extends Component {} diff --git a/webapp/app/pods/components/conflicts-page/styles.scss b/webapp/app/pods/components/conflicts-page/styles.scss deleted file mode 100644 index b74dc0868..000000000 --- a/webapp/app/pods/components/conflicts-page/styles.scss +++ /dev/null @@ -1,3 +0,0 @@ -.conflicts-page { - background: var(--content-background); -} diff --git a/webapp/app/pods/components/conflicts-page/template.hbs b/webapp/app/pods/components/conflicts-page/template.hbs deleted file mode 100644 index a9f0ff179..000000000 --- a/webapp/app/pods/components/conflicts-page/template.hbs +++ /dev/null @@ -1,38 +0,0 @@ -
    - - - {{#if @isLoading}} - - {{/if}} - - {{#if @showSkeleton}} - - {{else if @showLoading}} - - {{else}} - - - {{/if}} -
    \ No newline at end of file diff --git a/webapp/app/pods/components/date-tag/template.hbs b/webapp/app/pods/components/date-tag/template.hbs deleted file mode 100644 index ed12ea38f..000000000 --- a/webapp/app/pods/components/date-tag/template.hbs +++ /dev/null @@ -1,5 +0,0 @@ - - - \ No newline at end of file diff --git a/webapp/app/pods/components/inline-machine-translate/styles.scss b/webapp/app/pods/components/inline-machine-translate/styles.scss deleted file mode 100644 index d30370201..000000000 --- a/webapp/app/pods/components/inline-machine-translate/styles.scss +++ /dev/null @@ -1,74 +0,0 @@ -.prompt-button { - position: relative; - display: flex; - align-items: center; - - > .button { - transition: opacity 0.2s ease-in-out; - box-shadow: none; - opacity: 0.7; - border-radius: var(--border-radius); - } - - > button:focus, - > button:hover { - transform: translate3d(0, 0, 0); - } - - &:focus, - &:hover { - outline: none; - - > .button { - opacity: 1; - } - - .prompt-button-quick-access[data-rtl] { - left: 36px; - right: auto; - } - - .prompt-button-quick-access { - opacity: 1; - right: 36px; - padding: 0 10px; - pointer-events: all; - } - } -} - -.prompt-button-quick-access[data-rtl] { - left: 0; - right: auto; - justify-content: flex-start; -} - -.prompt-button-quick-access { - background: var(--input-background); - opacity: 0; - pointer-events: none; - right: 0; - display: flex; - gap: 0; - justify-content: flex-end; - align-items: center; - position: absolute; - top: 0; - transition: all 0.2s ease-in-out; - - > button { - font-size: 11px; - flex-shrink: 0; - } - - > button:focus, - > button:hover { - background: var(--background-light); - transform: translate3d(0, 0, 0); - } - - > button :global(.label) { - padding-left: 8px; - padding-right: 8px; - } -} diff --git a/webapp/app/pods/components/inline-machine-translate/template.hbs b/webapp/app/pods/components/inline-machine-translate/template.hbs deleted file mode 100644 index 9df40296c..000000000 --- a/webapp/app/pods/components/inline-machine-translate/template.hbs +++ /dev/null @@ -1,19 +0,0 @@ -{{#if @languages}} -
    -
    - {{#each @languages as |language|}} - - {{language.slug}} - - {{/each}} -
    - - -
    -{{/if}} \ No newline at end of file diff --git a/webapp/app/pods/components/lint-options/template.hbs b/webapp/app/pods/components/lint-options/template.hbs deleted file mode 100644 index 789feba1a..000000000 --- a/webapp/app/pods/components/lint-options/template.hbs +++ /dev/null @@ -1,38 +0,0 @@ -
    -
    -
    -
    -
    - {{inline-svg '/assets/search.svg' local-class='search-icon'}} - - -
    - -
    - {{#if this.showDocuments}} -
    -
    - -
    -
    - {{/if}} - - {{#if this.showVersions}} -
    -
    - -
    -
    - {{/if}} -
    -
    -
    -
    -
    \ No newline at end of file diff --git a/webapp/app/pods/components/lint-translations-page/component.ts b/webapp/app/pods/components/lint-translations-page/component.ts deleted file mode 100644 index bfde43f3f..000000000 --- a/webapp/app/pods/components/lint-translations-page/component.ts +++ /dev/null @@ -1,40 +0,0 @@ -import Component from '@glimmer/component'; - -interface Args { - project: any; - lintTranslations: any[]; - permissions: Record; -} - -interface Stat { - title: string; - count: number; -} - -export default class LintTranslationsPage extends Component { - get lintTranslationsStatsCount() { - return this.lintTranslationsStats.reduce( - (total, stat) => stat.count + total, - 0 - ); - } - - get lintTranslationsStats() { - const stats = this.args.lintTranslations.reduce((acc, lintTranslation) => { - lintTranslation.messages.forEach((message: {check: string}) => { - if (acc[message.check]) { - acc[message.check]++; - } else { - acc[message.check] = 1; - } - }); - - return acc; - }, {}); - - return Object.entries(stats).map(([title, count]) => ({ - title, - count, - })) as Stat[]; - } -} diff --git a/webapp/app/pods/components/lint-translations-page/item/template.hbs b/webapp/app/pods/components/lint-translations-page/item/template.hbs deleted file mode 100644 index c926fe712..000000000 --- a/webapp/app/pods/components/lint-translations-page/item/template.hbs +++ /dev/null @@ -1,34 +0,0 @@ -{{#if @lintTranslation.messages}} -
    -
      - {{#each this.messages as |message|}} - - {{#if message.message}} - {{message.message}} - {{else}} - {{t (concat 'components.translation_edit.lint_message.checks.' message.check)}} - {{/if}} - - {{/each}} -
    - - {{#if @project}} -
    - - - {{this.translationKey.value}} - - {{#if this.translationKey.prefix}} - {{this.translationKey.prefix}} - {{else}} - {{@lintTranslation.translation.document.path}} - {{/if}} - - - -
    - {{/if}} - -
    {{{this.annotatedText}}}
    -
    -{{/if}} \ No newline at end of file diff --git a/webapp/app/pods/components/project-settings/integrations/form/azure-storage-container/styles.scss b/webapp/app/pods/components/project-settings/integrations/form/azure-storage-container/styles.scss deleted file mode 100644 index 7ad947c92..000000000 --- a/webapp/app/pods/components/project-settings/integrations/form/azure-storage-container/styles.scss +++ /dev/null @@ -1,60 +0,0 @@ -.instructions { - border-top: 1px solid var(--background-light-highlight); - padding-top: 10px; - margin: 20px 0 10px; -} - -.instructions-text { - margin-top: 7px; - font-style: italic; - color: #555; - - a { - font-family: var(--font-monospace); - color: var(--color-primary); - text-decoration: none; - - &:focus, - &:hover { - text-decoration: underline; - opacity: 0.8; - } - } -} - -.data-control { - margin-bottom: 15px; -} - -.data-title { - display: block; - margin-bottom: 5px; - font-size: 13px; - font-weight: bold; -} - -.data-title-help { - margin-left: 4px; - font-size: 11px; - font-weight: normal; - text-decoration: none; - color: var(--color-primary); - - &:focus, - &:hover { - text-decoration: underline; - } -} - -.textInput { - @extend %textInput; - flex-grow: 1; - flex-shrink: 0; - padding: 8px 10px; - margin-right: 10px; - background: #fafafa; - max-width: 500px; - width: 100%; - font-family: var(--font-monospace); - font-size: 11px; -} diff --git a/webapp/app/pods/components/project-settings/prompts/config/template.hbs b/webapp/app/pods/components/project-settings/prompts/config/template.hbs deleted file mode 100644 index 33d00eebc..000000000 --- a/webapp/app/pods/components/project-settings/prompts/config/template.hbs +++ /dev/null @@ -1,39 +0,0 @@ -
    -
    - {{inline-svg this.logoProvider local-class='logo'}} - - - {{#if @project.promptConfig}} - {{inline-svg 'assets/check.svg' local-class='check'}} - {{/if}} -
    - -

    - {{#if @project.promptConfig}} - {{t 'components.project_settings.prompts.config_key_help_present'}} - {{else}} - {{t 'components.project_settings.prompts.config_key_help_not_present'}} - {{/if}} -

    - - -
    - -
    - {{#if @project.promptConfig}} - - {{t 'components.project_settings.integrations.delete'}} - - {{/if}} - - - {{t 'components.project_settings.integrations.save'}} - -
    \ No newline at end of file diff --git a/webapp/app/pods/components/project-settings/prompts/template.hbs b/webapp/app/pods/components/project-settings/prompts/template.hbs deleted file mode 100644 index ff2133ff3..000000000 --- a/webapp/app/pods/components/project-settings/prompts/template.hbs +++ /dev/null @@ -1,28 +0,0 @@ -
    -
    -

    - {{t 'components.project_settings.prompts.title'}} -

    -

    - - {{t 'components.project_settings.prompts.help'}} -

    - -
      - {{#each @prompts as |prompt|}} -
    • - -
    • - {{/each}} -
    - - - {{inline-svg '/assets/add.svg' class='button-icon'}} - {{t 'components.project_settings.prompts.new_button'}} - -
    - -
    - -
    -
    \ No newline at end of file diff --git a/webapp/app/pods/components/projects-list/item/component.ts b/webapp/app/pods/components/projects-list/item/component.ts deleted file mode 100644 index 46563da74..000000000 --- a/webapp/app/pods/components/projects-list/item/component.ts +++ /dev/null @@ -1,50 +0,0 @@ -import Component from '@glimmer/component'; -import {lt, gte} from '@ember/object/computed'; -import percentage from 'accent-webapp/component-helpers/percentage'; - -const LOW_PERCENTAGE = 50; -const HIGH_PERCENTAGE = 90; - -interface Args { - project: any; -} - -export default class ProjectsListItem extends Component { - @lt('correctedKeysPercentage', LOW_PERCENTAGE) - lowPercentage: boolean; // Lower than low percentage - - @gte('correctedKeysPercentage', LOW_PERCENTAGE) - mediumPercentage: boolean; // higher or equal than low percentage - - @gte('correctedKeysPercentage', HIGH_PERCENTAGE) - highPercentage: boolean; // higher or equal than high percentage - - get colors() { - return ` - .projectId-${this.args.project.id} { - --color-primary: ${this.args.project.mainColor}; - } - `; - } - - get totalStrings() { - return this.args.project.translationsCount; - } - - get totalConflicts() { - return this.args.project.conflictsCount; - } - - get totalReviewed() { - return ( - this.args.project.translationsCount - this.args.project.conflictsCount - ); - } - - get correctedKeysPercentage() { - return percentage( - this.totalStrings - this.totalConflicts, - this.totalStrings - ); - } -} diff --git a/webapp/app/pods/components/projects-list/template.hbs b/webapp/app/pods/components/projects-list/template.hbs deleted file mode 100644 index 73a23677e..000000000 --- a/webapp/app/pods/components/projects-list/template.hbs +++ /dev/null @@ -1,23 +0,0 @@ -
      - {{#each @projects key='id' as |project index|}} -
    • - - - -
    • - {{else if @query}} - - {{else}} - - {{inline-svg 'assets/empty.svg' class='icon'}} - {{t 'components.projects_list.no_projects'}} - {{#if (get @permissions 'create_project')}} -
      - - {{t 'components.projects_list.maybe_create_one'}} - -
      - {{/if}} -
      - {{/each}} -
    \ No newline at end of file diff --git a/webapp/app/pods/components/translations-list/template.hbs b/webapp/app/pods/components/translations-list/template.hbs deleted file mode 100644 index 0bb8931d2..000000000 --- a/webapp/app/pods/components/translations-list/template.hbs +++ /dev/null @@ -1,35 +0,0 @@ -
      - {{#if this.currentVersion}} -
      - - {{t 'components.translations_list.translations_version_notice'}} - {{this.currentVersion.tag}} -
      - {{/if}} - - {{#each @translations key='id' as |translation|}} - - {{else if @query}} - - {{else if @withAdvancedFilters}} - - {{else}} - - {{inline-svg 'assets/empty.svg' local-class='icon'}} - {{t 'components.translations_list.no_translations'}} -
      - {{t 'components.translations_list.maybe_sync_before'}} - - {{t 'components.translations_list.maybe_sync_link'}} - -
      -
      - {{/each}} -
    \ No newline at end of file diff --git a/webapp/app/pods/logged-in/jipt/template.hbs b/webapp/app/pods/logged-in/jipt/template.hbs deleted file mode 100644 index 319be03b7..000000000 --- a/webapp/app/pods/logged-in/jipt/template.hbs +++ /dev/null @@ -1,13 +0,0 @@ - - -
    - - - - - {{outlet}} -
    \ No newline at end of file diff --git a/webapp/app/pods/logged-in/project/revision/conflicts/controller.ts b/webapp/app/pods/logged-in/project/revision/conflicts/controller.ts deleted file mode 100644 index 5d1e9d8c5..000000000 --- a/webapp/app/pods/logged-in/project/revision/conflicts/controller.ts +++ /dev/null @@ -1,159 +0,0 @@ -import {camelize} from '@ember/string'; -import {action} from '@ember/object'; -import {inject as service} from '@ember/service'; -import {readOnly, equal, empty, and} from '@ember/object/computed'; -import Controller from '@ember/controller'; -import translationCorrectQuery from 'accent-webapp/queries/correct-translation'; -import IntlService from 'ember-intl/services/intl'; -import FlashMessages from 'ember-cli-flash/services/flash-messages'; -import Apollo from 'accent-webapp/services/apollo'; -import ApolloMutate from 'accent-webapp/services/apollo-mutate'; -import GlobalState from 'accent-webapp/services/global-state'; -import {tracked} from '@glimmer/tracking'; -import projectTranslateTextQuery from 'accent-webapp/queries/translate-text-project'; - -const FLASH_MESSAGE_CORRECT_SUCCESS = - 'pods.project.conflicts.flash_messages.correct_success'; -const FLASH_MESSAGE_CORRECT_ERROR = - 'pods.project.conflicts.flash_messages.correct_error'; -const FLASH_MESSAGE_TRANSLATE_ERROR_PREFIX = - 'pods.project.conflicts.flash_messages.translate_error'; -const FLASH_MESSAGE_TRANSLATE_PROVIDER_ERROR = - 'pods.project.conflicts.flash_messages.translate_provider_error'; - -export default class ConflictsController extends Controller { - @tracked - model: any; - - @service('intl') - intl: IntlService; - - @service('flash-messages') - flashMessages: FlashMessages; - - @service('apollo') - apollo: Apollo; - - @service('apollo-mutate') - apolloMutate: ApolloMutate; - - @service('global-state') - globalState: GlobalState; - - queryParams = ['page', 'query', 'document', 'version']; - - @tracked - query = ''; - - @tracked - document: string | null = null; - - @tracked - version: string | null = null; - - @tracked - page = 1; - - @readOnly('globalState.permissions') - permissions: any; - - @equal('model.translations.entries', undefined) - emptyEntries: boolean; - - @readOnly('model.revisionModel.project.revisions') - revisions: any; - - @empty('document') - emptyDocument: boolean; - - @equal('query', '') - emptyQuery: boolean; - - @and('emptyEntries', 'model.loading') - showLoading: boolean; - - @and('emptyEntries', 'model.loading', 'emptyQuery', 'emptyDocument') - showSkeleton: boolean; - - @action - async copyTranslation( - text: string, - sourceLanguageSlug: string, - targetLanguageSlug: string - ) { - const {data} = await this.apollo.client.query({ - fetchPolicy: 'network-only', - query: projectTranslateTextQuery, - variables: { - text, - sourceLanguageSlug, - targetLanguageSlug, - projectId: this.model.project.id, - }, - }); - - if (data.viewer?.project?.translatedText?.text) { - return data.viewer.project.translatedText; - } else if (data.viewer?.project?.translatedText?.error) { - const error = data.viewer?.project?.translatedText?.error; - const source = sourceLanguageSlug; - const target = targetLanguageSlug; - - this.flashMessages.error( - this.intl.t(`${FLASH_MESSAGE_TRANSLATE_ERROR_PREFIX}.${error}`, { - source, - target, - }) - ); - } else { - const provider = camelize(data.viewer?.project?.translatedText?.provider); - this.flashMessages.error( - this.intl.t(FLASH_MESSAGE_TRANSLATE_PROVIDER_ERROR, {provider}) - ); - } - } - - @action - async correctConflict(conflict: any, text: string) { - const response = await this.apolloMutate.mutate({ - mutation: translationCorrectQuery, - variables: { - translationId: conflict.id, - text, - }, - }); - - if (response.errors) { - this.flashMessages.error(this.intl.t(FLASH_MESSAGE_CORRECT_ERROR)); - } else { - this.flashMessages.success(this.intl.t(FLASH_MESSAGE_CORRECT_SUCCESS)); - } - - return response; - } - - @action - changeQuery(query: string) { - this.page = 1; - this.query = query; - } - - @action - changeDocument(select: HTMLSelectElement) { - this.page = 1; - this.document = select.value ? select.value : null; - } - - @action - changeVersion(select: HTMLSelectElement) { - this.page = 1; - this.version = select.value ? select.value : null; - } - - @action - selectPage(page: number) { - window.scrollTo(0, 0); - - this.page = page; - } -} diff --git a/webapp/app/pods/logged-in/project/revision/conflicts/template.hbs b/webapp/app/pods/logged-in/project/revision/conflicts/template.hbs deleted file mode 100644 index c7f2f0913..000000000 --- a/webapp/app/pods/logged-in/project/revision/conflicts/template.hbs +++ /dev/null @@ -1,21 +0,0 @@ - \ No newline at end of file diff --git a/webapp/app/queries/activity-activities.ts b/webapp/app/queries/activity-activities.ts index 8c3ae9c21..52d88a0b9 100644 --- a/webapp/app/queries/activity-activities.ts +++ b/webapp/app/queries/activity-activities.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ActivityActivities($projectId: ID!, $activityId: ID!, $page: Int) { diff --git a/webapp/app/queries/authentication-providers.ts b/webapp/app/queries/authentication-providers.ts index e7b8467f2..72b8219a3 100644 --- a/webapp/app/queries/authentication-providers.ts +++ b/webapp/app/queries/authentication-providers.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query AuthenticationProviders { diff --git a/webapp/app/queries/conflicts.ts b/webapp/app/queries/conflicts.ts index 28086e7e9..04fada702 100644 --- a/webapp/app/queries/conflicts.ts +++ b/webapp/app/queries/conflicts.ts @@ -1,13 +1,18 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Conflicts( $projectId: ID! - $revisionId: ID! $query: String $page: Int $document: ID + $relatedRevisions: [ID!] $version: ID + $isTextEmpty: Boolean + $isTextNotEmpty: Boolean + $isAddedLastSync: Boolean + $isCommentedOn: Boolean + $isTranslated: Boolean ) { viewer { project(id: $projectId) { @@ -34,30 +39,60 @@ export default gql` name } - revision(id: $revisionId) { + revisions { id + isMaster + slug + name + + language { + id + slug + name + } + } + + groupedTranslations( + query: $query + page: $page + pageSize: 20 + document: $document + version: $version + relatedRevisions: $relatedRevisions + isConflicted: true + isTextEmpty: $isTextEmpty + isTextNotEmpty: $isTextNotEmpty + isAddedLastSync: $isAddedLastSync + isTranslated: $isTranslated + isCommentedOn: $isCommentedOn + ) { + meta { + totalEntries + totalPages + currentPage + nextPage + previousPage + } - translations( - query: $query - page: $page - pageSize: 20 - document: $document - version: $version - isConflicted: true - ) { - meta { - totalEntries - totalPages - currentPage - nextPage - previousPage + revisions { + id + } + + entries { + key + document { + id + path } - entries { + + translations { id key conflictedText correctedText valueType + isConflicted + isTranslated lintMessages { text @@ -83,31 +118,6 @@ export default gql` rtl } } - - relatedTranslations { - id - correctedText - isConflicted - revision { - id - isMaster - name - slug - rtl - - language { - id - name - slug - rtl - } - } - } - - document { - id - path - } } } } diff --git a/webapp/app/queries/correct-all-revision.ts b/webapp/app/queries/correct-all-revision.ts index 251b5e4d7..21f84ff9e 100644 --- a/webapp/app/queries/correct-all-revision.ts +++ b/webapp/app/queries/correct-all-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CorrectAll($revisionId: ID!) { diff --git a/webapp/app/queries/correct-translation.ts b/webapp/app/queries/correct-translation.ts index 392ba4296..7d94bc026 100644 --- a/webapp/app/queries/correct-translation.ts +++ b/webapp/app/queries/correct-translation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TranslationCorrect($translationId: ID!, $text: String!) { diff --git a/webapp/app/queries/create-api-token.ts b/webapp/app/queries/create-api-token.ts index 3205b2733..a5cd40d65 100644 --- a/webapp/app/queries/create-api-token.ts +++ b/webapp/app/queries/create-api-token.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface CreateApiTokenVariables { name: string; diff --git a/webapp/app/queries/create-collaborator.ts b/webapp/app/queries/create-collaborator.ts index dd758b729..d6edf2347 100644 --- a/webapp/app/queries/create-collaborator.ts +++ b/webapp/app/queries/create-collaborator.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CollaboratorCreate($role: Role!, $email: String!, $projectId: ID!) { diff --git a/webapp/app/queries/create-comment.ts b/webapp/app/queries/create-comment.ts index 9a580a1b6..27063f119 100644 --- a/webapp/app/queries/create-comment.ts +++ b/webapp/app/queries/create-comment.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CommentCreate($translationId: ID!, $text: String!) { diff --git a/webapp/app/queries/create-integration.ts b/webapp/app/queries/create-integration.ts index 4e0053259..7e024ca1d 100644 --- a/webapp/app/queries/create-integration.ts +++ b/webapp/app/queries/create-integration.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation IntegrationCreate( diff --git a/webapp/app/queries/create-project-prompt.ts b/webapp/app/queries/create-project-prompt.ts index c5ff6189d..3db7f4185 100644 --- a/webapp/app/queries/create-project-prompt.ts +++ b/webapp/app/queries/create-project-prompt.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface CreatePromptResponse { prompt: { diff --git a/webapp/app/queries/create-project.ts b/webapp/app/queries/create-project.ts index 19ab2091f..aec02850f 100644 --- a/webapp/app/queries/create-project.ts +++ b/webapp/app/queries/create-project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface CreateProjectVariables { name: string; diff --git a/webapp/app/queries/create-revision.ts b/webapp/app/queries/create-revision.ts index 51073a03c..cdeddaf18 100644 --- a/webapp/app/queries/create-revision.ts +++ b/webapp/app/queries/create-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation RevisionCreate( diff --git a/webapp/app/queries/create-translation-comments-subscription.ts b/webapp/app/queries/create-translation-comments-subscription.ts index 124d0e7ce..3e97f6b9c 100644 --- a/webapp/app/queries/create-translation-comments-subscription.ts +++ b/webapp/app/queries/create-translation-comments-subscription.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TranslationCommentsSubscriptionCreate( diff --git a/webapp/app/queries/create-version.ts b/webapp/app/queries/create-version.ts index 9f6dc2e18..274150dcb 100644 --- a/webapp/app/queries/create-version.ts +++ b/webapp/app/queries/create-version.ts @@ -1,8 +1,18 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` - mutation VersionCreate($name: String!, $tag: String!, $projectId: ID!) { - createVersion(name: $name, tag: $tag, projectId: $projectId) { + mutation VersionCreate( + $name: String! + $tag: String! + $projectId: ID! + $copyOnUpdateTranslation: Boolean! + ) { + createVersion( + name: $name + tag: $tag + projectId: $projectId + copyOnUpdateTranslation: $copyOnUpdateTranslation + ) { version { id } diff --git a/webapp/app/queries/delete-collaborator.ts b/webapp/app/queries/delete-collaborator.ts index 0dfeaf015..f02404b58 100644 --- a/webapp/app/queries/delete-collaborator.ts +++ b/webapp/app/queries/delete-collaborator.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CollaboratorDelete($collaboratorId: ID!) { diff --git a/webapp/app/queries/delete-comment.ts b/webapp/app/queries/delete-comment.ts index 88edf4cc1..c807f9a3c 100644 --- a/webapp/app/queries/delete-comment.ts +++ b/webapp/app/queries/delete-comment.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CommentDelete($commentId: ID!) { diff --git a/webapp/app/queries/delete-document.ts b/webapp/app/queries/delete-document.ts index 767fd09b9..690648ef2 100644 --- a/webapp/app/queries/delete-document.ts +++ b/webapp/app/queries/delete-document.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation DocumentDelete($documentId: ID!) { diff --git a/webapp/app/queries/delete-integration.ts b/webapp/app/queries/delete-integration.ts index 0059bfca8..5a5c9dffa 100644 --- a/webapp/app/queries/delete-integration.ts +++ b/webapp/app/queries/delete-integration.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation IntegrationDelete($integrationId: ID!) { diff --git a/webapp/app/queries/delete-project-machine-translations-config.ts b/webapp/app/queries/delete-project-machine-translations-config.ts index c5252642c..3cccbb669 100644 --- a/webapp/app/queries/delete-project-machine-translations-config.ts +++ b/webapp/app/queries/delete-project-machine-translations-config.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface DeleteProjectMachineTranslationsConfigVariables { projectId: string; diff --git a/webapp/app/queries/delete-project-prompt-config.ts b/webapp/app/queries/delete-project-prompt-config.ts index 7fa590485..616872c12 100644 --- a/webapp/app/queries/delete-project-prompt-config.ts +++ b/webapp/app/queries/delete-project-prompt-config.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface DeleteProjectPromptConfigVariables { projectId: string; diff --git a/webapp/app/queries/delete-project-prompt.ts b/webapp/app/queries/delete-project-prompt.ts index 5d811db47..9702d62cc 100644 --- a/webapp/app/queries/delete-project-prompt.ts +++ b/webapp/app/queries/delete-project-prompt.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface DeleteProjectPromptVariables { promptId: string; diff --git a/webapp/app/queries/delete-project.ts b/webapp/app/queries/delete-project.ts index 21cd493fc..cb08840af 100644 --- a/webapp/app/queries/delete-project.ts +++ b/webapp/app/queries/delete-project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation ProjectDelete($projectId: ID!) { diff --git a/webapp/app/queries/delete-revision.ts b/webapp/app/queries/delete-revision.ts index da70549cd..e1dd4fec6 100644 --- a/webapp/app/queries/delete-revision.ts +++ b/webapp/app/queries/delete-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation RevisionDelete($revisionId: ID!) { diff --git a/webapp/app/queries/delete-translation-comments-subscription.ts b/webapp/app/queries/delete-translation-comments-subscription.ts index 266d62484..1293d33d2 100644 --- a/webapp/app/queries/delete-translation-comments-subscription.ts +++ b/webapp/app/queries/delete-translation-comments-subscription.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TranslationCommentsSubscriptionDelete( diff --git a/webapp/app/queries/execute-integration.ts b/webapp/app/queries/execute-integration.ts index 504754810..032d06d9f 100644 --- a/webapp/app/queries/execute-integration.ts +++ b/webapp/app/queries/execute-integration.ts @@ -1,13 +1,15 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation IntegrationExecute( $integrationId: ID! $azureStorageContainer: ProjectIntegrationExecuteAzureStorageContainerInput + $awsS3: ProjectIntegrationExecuteAwsS3Input ) { executeProjectIntegration( id: $integrationId azureStorageContainer: $azureStorageContainer + awsS3: $awsS3 ) { projectIntegration: result { id diff --git a/webapp/app/queries/improve-text-prompt.ts b/webapp/app/queries/improve-text-prompt.ts index 27e2e88e7..f895c5715 100644 --- a/webapp/app/queries/improve-text-prompt.ts +++ b/webapp/app/queries/improve-text-prompt.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TextPromptImprove($promptId: ID!, $text: String!) { diff --git a/webapp/app/queries/jipt-example.ts b/webapp/app/queries/jipt-example.ts new file mode 100644 index 000000000..d30bdec48 --- /dev/null +++ b/webapp/app/queries/jipt-example.ts @@ -0,0 +1,27 @@ +import {gql} from '@apollo/client/core'; + +export default gql` + query JIPTExample($projectId: ID!) { + viewer { + project(id: $projectId) { + id + name + mainColor + + revision { + id + translations(pageSize: 1) { + entries { + id + key + document { + id + path + } + } + } + } + } + } + } +`; diff --git a/webapp/app/queries/jipt-project.ts b/webapp/app/queries/jipt-project.ts index 025f25eb5..78e24c974 100644 --- a/webapp/app/queries/jipt-project.ts +++ b/webapp/app/queries/jipt-project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Project($projectId: ID!, $revisionId: ID) { diff --git a/webapp/app/queries/jipt-translations.ts b/webapp/app/queries/jipt-translations.ts index a6d4ff613..77b731d67 100644 --- a/webapp/app/queries/jipt-translations.ts +++ b/webapp/app/queries/jipt-translations.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Translations( diff --git a/webapp/app/queries/languages-search.ts b/webapp/app/queries/languages-search.ts index 2681fde06..072bcd948 100644 --- a/webapp/app/queries/languages-search.ts +++ b/webapp/app/queries/languages-search.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query LanguagesSearch($query: String!) { diff --git a/webapp/app/queries/lint-translation.ts b/webapp/app/queries/lint-translation.ts index 3f4075f4a..28ae607d2 100644 --- a/webapp/app/queries/lint-translation.ts +++ b/webapp/app/queries/lint-translation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query LintTranslation($projectId: ID!, $text: String, $translationId: ID!) { diff --git a/webapp/app/queries/lint-translations.ts b/webapp/app/queries/lint-translations.ts index 7483373c4..1973df500 100644 --- a/webapp/app/queries/lint-translations.ts +++ b/webapp/app/queries/lint-translations.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Lint( diff --git a/webapp/app/queries/project-activities.ts b/webapp/app/queries/project-activities.ts index c4af2dd52..b18ab6d1a 100644 --- a/webapp/app/queries/project-activities.ts +++ b/webapp/app/queries/project-activities.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectActivities( diff --git a/webapp/app/queries/project-activity.ts b/webapp/app/queries/project-activity.ts index 6b9470810..dbd4101cf 100644 --- a/webapp/app/queries/project-activity.ts +++ b/webapp/app/queries/project-activity.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectActivity($projectId: ID!, $activityId: ID!) { diff --git a/webapp/app/queries/project-api-token.ts b/webapp/app/queries/project-api-token.ts index a3c4af84b..2c3a71155 100644 --- a/webapp/app/queries/project-api-token.ts +++ b/webapp/app/queries/project-api-token.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectApiToken($projectId: ID!) { diff --git a/webapp/app/queries/project-collaborators.ts b/webapp/app/queries/project-collaborators.ts index bbc98f6c3..e2aa05218 100644 --- a/webapp/app/queries/project-collaborators.ts +++ b/webapp/app/queries/project-collaborators.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectCollaborators($projectId: ID!) { diff --git a/webapp/app/queries/project-comments.ts b/webapp/app/queries/project-comments.ts index b01248c4b..43759d7ac 100644 --- a/webapp/app/queries/project-comments.ts +++ b/webapp/app/queries/project-comments.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectComments($projectId: ID!, $page: Int) { diff --git a/webapp/app/queries/project-dashboard.ts b/webapp/app/queries/project-dashboard.ts index cc8beeef3..089dd7431 100644 --- a/webapp/app/queries/project-dashboard.ts +++ b/webapp/app/queries/project-dashboard.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface ProjectDashboardQueryVariables { projectId: string; @@ -21,6 +21,7 @@ export interface ProjectDashboardQueryResponse { id: string; conflictsCount: number; reviewedCount: number; + translatedCount: number; translationsCount: number; isMaster: boolean; name: string; @@ -131,6 +132,7 @@ export default gql` id conflictsCount reviewedCount + translatedCount translationsCount isMaster name diff --git a/webapp/app/queries/project-documents.ts b/webapp/app/queries/project-documents.ts index 46808ff89..320a5ad15 100644 --- a/webapp/app/queries/project-documents.ts +++ b/webapp/app/queries/project-documents.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectDocuments( diff --git a/webapp/app/queries/project-edit.ts b/webapp/app/queries/project-edit.ts index 2ac6269ce..ec9b410df 100644 --- a/webapp/app/queries/project-edit.ts +++ b/webapp/app/queries/project-edit.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectEdit($projectId: ID!) { @@ -9,6 +9,9 @@ export default gql` mainColor logo isFileOperationsLocked + integrations { + id + } } } } diff --git a/webapp/app/queries/project-machine-translations-config.ts b/webapp/app/queries/project-machine-translations-config.ts index 24a8dc315..1ef34b983 100644 --- a/webapp/app/queries/project-machine-translations-config.ts +++ b/webapp/app/queries/project-machine-translations-config.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectMachineTranslationsConfig($projectId: ID!) { @@ -7,8 +7,8 @@ export default gql` id machineTranslationsConfig { provider - usePlatform enabledActions + usePlatform useConfigKey } } diff --git a/webapp/app/queries/project-new-language.ts b/webapp/app/queries/project-new-language.ts index b8a856ee5..b53f01a70 100644 --- a/webapp/app/queries/project-new-language.ts +++ b/webapp/app/queries/project-new-language.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectNewLanguage($projectId: ID!) { diff --git a/webapp/app/queries/project-prompt-config.ts b/webapp/app/queries/project-prompt-config.ts index 95e6c578e..0a499b48c 100644 --- a/webapp/app/queries/project-prompt-config.ts +++ b/webapp/app/queries/project-prompt-config.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface ProjectPromptConfigResponse { viewer: { @@ -31,6 +31,8 @@ export default gql` promptConfig { provider + usePlatform + useConfigKey } } } diff --git a/webapp/app/queries/project-prompts.ts b/webapp/app/queries/project-prompts.ts index 12eb6543c..c85834554 100644 --- a/webapp/app/queries/project-prompts.ts +++ b/webapp/app/queries/project-prompts.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectPrompts($projectId: ID!) { diff --git a/webapp/app/queries/project-service-integrations.ts b/webapp/app/queries/project-service-integrations.ts index 95a8aaf38..0d71fe80b 100644 --- a/webapp/app/queries/project-service-integrations.ts +++ b/webapp/app/queries/project-service-integrations.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectServiceIntegrations($projectId: ID!) { @@ -27,6 +27,17 @@ export default gql` } } + ... on ProjectIntegrationAwsS3 { + lastExecutedAt + data { + id + bucket + pathPrefix + region + accessKeyId + } + } + ... on ProjectIntegrationAzureStorageContainer { lastExecutedAt data { diff --git a/webapp/app/queries/project-versions.ts b/webapp/app/queries/project-versions.ts index 1c633b19e..ccdfcd5e6 100644 --- a/webapp/app/queries/project-versions.ts +++ b/webapp/app/queries/project-versions.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query ProjectVersions($projectId: ID!, $page: Int) { @@ -28,6 +28,7 @@ export default gql` name tag insertedAt + copyOnUpdateTranslation user { id diff --git a/webapp/app/queries/project.ts b/webapp/app/queries/project.ts index 80356355e..ca58a9b6e 100644 --- a/webapp/app/queries/project.ts +++ b/webapp/app/queries/project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface ProjectQueryVariables { projectId: string; diff --git a/webapp/app/queries/projects.ts b/webapp/app/queries/projects.ts index 8935eb269..40530b964 100644 --- a/webapp/app/queries/projects.ts +++ b/webapp/app/queries/projects.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Projects($query: String, $page: Int, $nodeIds: [ID!]) { @@ -35,8 +35,6 @@ export default gql` lastSyncedAt mainColor logo - translationsCount - conflictsCount } } } diff --git a/webapp/app/queries/promote-master-revision.ts b/webapp/app/queries/promote-master-revision.ts index 8820a2d28..72e8dabbc 100644 --- a/webapp/app/queries/promote-master-revision.ts +++ b/webapp/app/queries/promote-master-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation RevisionMasterPromote($revisionId: ID!) { diff --git a/webapp/app/queries/related-translations.ts b/webapp/app/queries/related-translations.ts index f871f9878..b616375af 100644 --- a/webapp/app/queries/related-translations.ts +++ b/webapp/app/queries/related-translations.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query RelatedTranslations($projectId: ID!, $translationId: ID!) { diff --git a/webapp/app/queries/revoke-api-token.ts b/webapp/app/queries/revoke-api-token.ts index 078bb4a22..f4cd60bf3 100644 --- a/webapp/app/queries/revoke-api-token.ts +++ b/webapp/app/queries/revoke-api-token.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface RevokeApiTokenVariables { id: string; diff --git a/webapp/app/queries/rollback-operation.ts b/webapp/app/queries/rollback-operation.ts index e2b63cdae..e8f7d7ff1 100644 --- a/webapp/app/queries/rollback-operation.ts +++ b/webapp/app/queries/rollback-operation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation OperationRollback($operationId: ID!) { diff --git a/webapp/app/queries/save-project-machine-translations-config.ts b/webapp/app/queries/save-project-machine-translations-config.ts index de1f54039..f794ed777 100644 --- a/webapp/app/queries/save-project-machine-translations-config.ts +++ b/webapp/app/queries/save-project-machine-translations-config.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface SaveProjectMachineTranslationsConfigVariables { projectId: string; diff --git a/webapp/app/queries/save-project-prompt-config.ts b/webapp/app/queries/save-project-prompt-config.ts index 1d0c08363..f88cd6638 100644 --- a/webapp/app/queries/save-project-prompt-config.ts +++ b/webapp/app/queries/save-project-prompt-config.ts @@ -1,9 +1,10 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export interface SaveProjectPromptConfigVariables { projectId: string; provider: string; configKey: string; + usePlatform: boolean; } export interface SaveProjectPromptConfigResponse { @@ -18,11 +19,13 @@ export default gql` mutation ProjectPromptConfigSave( $provider: String! $configKey: String + $usePlatform: Boolean! $projectId: ID! ) { saveProjectPromptConfig( provider: $provider configKey: $configKey + usePlatform: $usePlatform projectId: $projectId ) { project { diff --git a/webapp/app/queries/translate-text-project.ts b/webapp/app/queries/translate-text-project.ts index 077617837..99e646693 100644 --- a/webapp/app/queries/translate-text-project.ts +++ b/webapp/app/queries/translate-text-project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query TranslateTextProject( diff --git a/webapp/app/queries/translation-activities.ts b/webapp/app/queries/translation-activities.ts index 7da2e1d85..3eb28bbc5 100644 --- a/webapp/app/queries/translation-activities.ts +++ b/webapp/app/queries/translation-activities.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query TranslationActivities( diff --git a/webapp/app/queries/translation-comments.ts b/webapp/app/queries/translation-comments.ts index 921f0e2d8..6867ff35a 100644 --- a/webapp/app/queries/translation-comments.ts +++ b/webapp/app/queries/translation-comments.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query TranslationComments($projectId: ID!, $translationId: ID!, $page: Int) { diff --git a/webapp/app/queries/translation-editions.ts b/webapp/app/queries/translation-editions.ts index b3d68debf..534d5c962 100644 --- a/webapp/app/queries/translation-editions.ts +++ b/webapp/app/queries/translation-editions.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Translations($projectId: ID!, $translationId: ID!) { @@ -41,18 +41,6 @@ export default gql` tag } - revision { - id - slug - rtl - - language { - id - slug - rtl - } - } - document { id path diff --git a/webapp/app/queries/translation.ts b/webapp/app/queries/translation.ts index 38700a7de..59180cd7d 100644 --- a/webapp/app/queries/translation.ts +++ b/webapp/app/queries/translation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Translation($projectId: ID!, $translationId: ID!) { diff --git a/webapp/app/queries/translations.ts b/webapp/app/queries/translations.ts index 88432710d..77ab048c3 100644 --- a/webapp/app/queries/translations.ts +++ b/webapp/app/queries/translations.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` query Translations( @@ -44,6 +44,7 @@ export default gql` id translations( query: $query + pageSize: 20 page: $page document: $document version: $version @@ -93,6 +94,7 @@ export default gql` rtl } } + document { id path diff --git a/webapp/app/queries/uncorrect-all-revision.ts b/webapp/app/queries/uncorrect-all-revision.ts index f97ffea7a..2a7369b28 100644 --- a/webapp/app/queries/uncorrect-all-revision.ts +++ b/webapp/app/queries/uncorrect-all-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation UncorrectAll($revisionId: ID!) { diff --git a/webapp/app/queries/uncorrect-translation.ts b/webapp/app/queries/uncorrect-translation.ts index b89d47213..bf8dc6956 100644 --- a/webapp/app/queries/uncorrect-translation.ts +++ b/webapp/app/queries/uncorrect-translation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TranslationUncorrect($translationId: ID!, $text: String!) { diff --git a/webapp/app/queries/update-collaborator.ts b/webapp/app/queries/update-collaborator.ts index f69aab83b..ad5c0bfac 100644 --- a/webapp/app/queries/update-collaborator.ts +++ b/webapp/app/queries/update-collaborator.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CollaboratorUpdate($collaboratorId: ID!, $role: Role!) { diff --git a/webapp/app/queries/update-comment.ts b/webapp/app/queries/update-comment.ts index ad844ccc6..a294d7df8 100644 --- a/webapp/app/queries/update-comment.ts +++ b/webapp/app/queries/update-comment.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation CommentUpdate($commentId: ID!, $text: String!) { diff --git a/webapp/app/queries/update-document.ts b/webapp/app/queries/update-document.ts index 368a5f7ed..ce577424d 100644 --- a/webapp/app/queries/update-document.ts +++ b/webapp/app/queries/update-document.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation DocumentUpdate($documentId: ID!, $path: String!) { diff --git a/webapp/app/queries/update-integration.ts b/webapp/app/queries/update-integration.ts index 17b52b318..e523e1d64 100644 --- a/webapp/app/queries/update-integration.ts +++ b/webapp/app/queries/update-integration.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation IntegrationUpdate( diff --git a/webapp/app/queries/update-project-prompt.ts b/webapp/app/queries/update-project-prompt.ts index 063d8f1c9..edc29ec56 100644 --- a/webapp/app/queries/update-project-prompt.ts +++ b/webapp/app/queries/update-project-prompt.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation PromptUpdate( diff --git a/webapp/app/queries/update-project.ts b/webapp/app/queries/update-project.ts index 1a2f79d16..1453318b1 100644 --- a/webapp/app/queries/update-project.ts +++ b/webapp/app/queries/update-project.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation ProjectUpdate( diff --git a/webapp/app/queries/update-revision.ts b/webapp/app/queries/update-revision.ts index 46f9cac70..b5b648f13 100644 --- a/webapp/app/queries/update-revision.ts +++ b/webapp/app/queries/update-revision.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation RevisionUpdate($revisionId: ID!, $name: String, $slug: String) { diff --git a/webapp/app/queries/update-translation.ts b/webapp/app/queries/update-translation.ts index aaaafeb9a..9a2c6eee0 100644 --- a/webapp/app/queries/update-translation.ts +++ b/webapp/app/queries/update-translation.ts @@ -1,4 +1,4 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` mutation TranslationUpdate($translationId: ID!, $text: String!) { diff --git a/webapp/app/queries/update-version.ts b/webapp/app/queries/update-version.ts index 408250a68..53f0c880f 100644 --- a/webapp/app/queries/update-version.ts +++ b/webapp/app/queries/update-version.ts @@ -1,8 +1,18 @@ -import gql from 'graphql-tag'; +import {gql} from '@apollo/client/core'; export default gql` - mutation VersionUpdate($id: ID!, $name: String!, $tag: String!) { - updateVersion(id: $id, name: $name, tag: $tag) { + mutation VersionUpdate( + $id: ID! + $name: String! + $tag: String! + $copyOnUpdateTranslation: Boolean! + ) { + updateVersion( + id: $id + name: $name + tag: $tag + copyOnUpdateTranslation: $copyOnUpdateTranslation + ) { version { id name diff --git a/webapp/app/router.js b/webapp/app/router.js index 3b10a73f4..2f57d08ba 100644 --- a/webapp/app/router.js +++ b/webapp/app/router.js @@ -11,6 +11,8 @@ export default Router.map(function () { this.route('login', {path: ''}); this.route('logged-in', {path: 'app'}, function () { + this.route('jipt-example', {path: 'projects/:projectId/jipt-example'}); + this.route('jipt', {path: 'projects/:projectId/jipt'}, function () { this.route( 'translation', @@ -54,7 +56,7 @@ export default Router.map(function () { this.route('export', {path: ':fileId/export'}); this.route('jipt', {path: ':fileId/jipt'}); this.route('machine-translations', { - path: ':fileId/machine-translations', + path: ':fileId/machine-translations' }); }); this.route('versions', function () { @@ -64,9 +66,10 @@ export default Router.map(function () { }); this.route('comments', {path: 'conversation'}); + this.route('conflicts'); + this.route('revision', {path: 'revisions/:revisionId'}, function () { this.route('translations'); - this.route('conflicts'); this.route('lint-translations'); }); diff --git a/webapp/app/pods/application/route.ts b/webapp/app/routes/application.ts similarity index 100% rename from webapp/app/pods/application/route.ts rename to webapp/app/routes/application.ts diff --git a/webapp/app/pods/logged-in/route.ts b/webapp/app/routes/logged-in.ts similarity index 73% rename from webapp/app/pods/logged-in/route.ts rename to webapp/app/routes/logged-in.ts index 78bae430f..c0765c507 100644 --- a/webapp/app/pods/logged-in/route.ts +++ b/webapp/app/routes/logged-in.ts @@ -2,18 +2,22 @@ import {inject as service} from '@ember/service'; import Route from '@ember/routing/route'; import Raven from 'raven-js'; import Session from 'accent-webapp/services/session'; +import RouterService from '@ember/routing/router-service'; export default class LoggedInRoute extends Route { @service('session') session: Session; + @service('router') + router: RouterService; + afterModel() { Raven.setUserContext(this.session.credentials.user); } redirect() { if (!this.session.isAuthenticated) { - this.transitionTo('login'); + this.router.transitionTo('login'); } } } diff --git a/webapp/app/routes/logged-in/jipt-example.ts b/webapp/app/routes/logged-in/jipt-example.ts new file mode 100644 index 000000000..5925f43f8 --- /dev/null +++ b/webapp/app/routes/logged-in/jipt-example.ts @@ -0,0 +1,42 @@ +import Route from '@ember/routing/route'; +import {inject as service} from '@ember/service'; + +import jiptExampleQuery from 'accent-webapp/queries/jipt-example'; +import ApolloSubscription, { + Subscription +} from 'accent-webapp/services/apollo-subscription'; + +export default class JIPTExampleRoute extends Route { + @service('apollo-subscription') + apolloSubscription: ApolloSubscription; + + subscription: Subscription; + + model(params: any) { + (window as any).accentJiptInit = { + h: window.location.origin, + i: params.projectId + }; + + this.subscription = this.apolloSubscription.graphql( + () => this.modelFor(this.routeName), + jiptExampleQuery, + { + props: (data) => ({ + project: data.viewer.project + }), + options: { + variables: { + projectId: params.projectId + } + } + } + ); + + return this.subscription.currentResult(); + } + + deactivate() { + this.apolloSubscription.clearSubscription(this.subscription); + } +} diff --git a/webapp/app/pods/logged-in/jipt/route.ts b/webapp/app/routes/logged-in/jipt.ts similarity index 93% rename from webapp/app/pods/logged-in/jipt/route.ts rename to webapp/app/routes/logged-in/jipt.ts index 059a98379..3320939fe 100644 --- a/webapp/app/pods/logged-in/jipt/route.ts +++ b/webapp/app/routes/logged-in/jipt.ts @@ -3,7 +3,7 @@ import {inject as service} from '@ember/service'; import projectQuery from 'accent-webapp/queries/jipt-project'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import JIPT from 'accent-webapp/services/jipt'; import GlobalState from 'accent-webapp/services/global-state'; @@ -20,8 +20,8 @@ export default class JIPTRoute extends Route { queryParams = { revisionId: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; @@ -35,9 +35,9 @@ export default class JIPTRoute extends Route { options: { variables: { projectId: params.projectId, - revisionId: params.revisionId, - }, - }, + revisionId: params.revisionId + } + } } ); @@ -50,6 +50,7 @@ export default class JIPTRoute extends Route { private props(data: any) { if (!data.viewer || !data.viewer.project) return {permissions: []}; + this.globalState.mainColor = data.viewer.project.mainColor; this.jipt.listTranslations( data.viewer.project.revision.translations.entries, @@ -65,7 +66,6 @@ export default class JIPTRoute extends Route { ); this.globalState.permissions = permissions; - this.globalState.mainColor = data.viewer.project.mainColor; return {project: data.viewer.project, permissions, roles: data.roles}; } diff --git a/webapp/app/pods/logged-in/jipt/index/route.ts b/webapp/app/routes/logged-in/jipt/index.ts similarity index 71% rename from webapp/app/pods/logged-in/jipt/index/route.ts rename to webapp/app/routes/logged-in/jipt/index.ts index b31b460f9..e0c2d8e9d 100644 --- a/webapp/app/pods/logged-in/jipt/index/route.ts +++ b/webapp/app/routes/logged-in/jipt/index.ts @@ -3,11 +3,11 @@ import Route from '@ember/routing/route'; import translationsQuery from 'accent-webapp/queries/jipt-translations'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; -import IndexController from 'accent-webapp/pods/logged-in/jipt/index/controller'; +import Transition from '@ember/routing/transition'; +import IndexController from 'accent-webapp/controllers/logged-in/jipt/index'; export default class IndexRoute extends Route { @service('apollo-subscription') @@ -18,17 +18,17 @@ export default class IndexRoute extends Route { queryParams = { query: { - refreshModel: true, + refreshModel: true }, page: { - refreshModel: true, + refreshModel: true }, document: { - refreshModel: true, + refreshModel: true }, version: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; @@ -38,10 +38,18 @@ export default class IndexRoute extends Route { query, page, document, - version, + version }: {query: any; page: number; document: any; version: any}, transition: Transition ) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + + const projectId = this.routeParams.fetch( + transition, + 'logged-in.jipt' + ).projectId; + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationsQuery, @@ -51,23 +59,19 @@ export default class IndexRoute extends Route { documents: data.viewer.project.documents.entries, versions: data.viewer.project.versions.entries, translations: data.viewer.project.revision.translations, - selectedTranslationIds: this.routeParams.fetch( - transition, - 'logged-in.jipt' - ).transitionIds, + selectedTranslationIds: transition?.to?.queryParams.translationIds }), options: { fetchPolicy: 'cache-and-network', variables: { - projectId: this.routeParams.fetch(transition, 'logged-in.jipt') - .projectId, - revisionId: transition.to.queryParams.revisionId, + projectId, + revisionId: transition?.to?.queryParams.revisionId, query, page, document, - version, - }, - }, + version + } + } } ); diff --git a/webapp/app/pods/logged-in/jipt/translation/route.ts b/webapp/app/routes/logged-in/jipt/translation.ts similarity index 79% rename from webapp/app/pods/logged-in/jipt/translation/route.ts rename to webapp/app/routes/logged-in/jipt/translation.ts index 2f1417a68..451516da6 100644 --- a/webapp/app/pods/logged-in/jipt/translation/route.ts +++ b/webapp/app/routes/logged-in/jipt/translation.ts @@ -3,10 +3,10 @@ import Route from '@ember/routing/route'; import translationQuery from 'accent-webapp/queries/translation'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class TranslationRoute extends Route { @service('apollo-subscription') @@ -18,6 +18,9 @@ export default class TranslationRoute extends Route { subscription: Subscription; model({translationId}: {translationId: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationQuery, @@ -25,15 +28,17 @@ export default class TranslationRoute extends Route { props: (data) => ({ project: data.viewer.project, translation: data.viewer.project.translation, + revisions: data.viewer.project.revisions, + prompts: data.viewer.project.prompts }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.jipt') .projectId, - translationId, - }, - }, + translationId + } + } } ); diff --git a/webapp/app/pods/logged-in/jipt/translation/index/route.ts b/webapp/app/routes/logged-in/jipt/translation/index.ts similarity index 100% rename from webapp/app/pods/logged-in/jipt/translation/index/route.ts rename to webapp/app/routes/logged-in/jipt/translation/index.ts diff --git a/webapp/app/pods/logged-in/project/route.ts b/webapp/app/routes/logged-in/project.ts similarity index 89% rename from webapp/app/pods/logged-in/project/route.ts rename to webapp/app/routes/logged-in/project.ts index e30dfedc5..32e8d43fb 100644 --- a/webapp/app/pods/logged-in/project/route.ts +++ b/webapp/app/routes/logged-in/project.ts @@ -6,17 +6,13 @@ import projectQuery, { ProjectQueryResponse, ProjectQueryResponseProject, ProjectQueryResponseRole, - ProjectQueryResponseDocumentFormat, + ProjectQueryResponseDocumentFormat } from 'accent-webapp/queries/project'; import GlobalState from 'accent-webapp/services/global-state'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -interface RouteParams { - projectId: string; -} - interface Model { project?: ProjectQueryResponseProject; permissions: object; @@ -33,7 +29,7 @@ export default class ProjectRoute extends Route { subscription: Subscription; - model(params: RouteParams): Model { + model(params: any): Model { const props = (data: any) => this.transformData(data); this.subscription = this.apolloSubscription.graphql( @@ -43,9 +39,9 @@ export default class ProjectRoute extends Route { props, options: { variables: { - projectId: params.projectId, - }, - }, + projectId: params.projectId + } + } } ); @@ -88,7 +84,7 @@ export default class ProjectRoute extends Route { project: data.viewer.project, permissions, roles: data.roles, - documentFormats: data.documentFormats, + documentFormats: data.documentFormats }; } } diff --git a/webapp/app/pods/logged-in/project/activities/route.ts b/webapp/app/routes/logged-in/project/activities.ts similarity index 80% rename from webapp/app/pods/logged-in/project/activities/route.ts rename to webapp/app/routes/logged-in/project/activities.ts index 72f915536..c6d3f586c 100644 --- a/webapp/app/pods/logged-in/project/activities/route.ts +++ b/webapp/app/routes/logged-in/project/activities.ts @@ -4,11 +4,11 @@ import Route from '@ember/routing/route'; import projectActivitiesQuery from 'accent-webapp/queries/project-activities'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import ActivitiesController from 'accent-webapp/pods/logged-in/project/activities/controller'; +import ActivitiesController from 'accent-webapp/controllers/logged-in/project/activities'; export default class ActivitiesRoute extends Route { @service('apollo-subscription') @@ -19,20 +19,20 @@ export default class ActivitiesRoute extends Route { queryParams = { batchFilter: { - refreshModel: true, + refreshModel: true }, actionFilter: { - refreshModel: true, + refreshModel: true }, userFilter: { - refreshModel: true, + refreshModel: true }, versionFilter: { - refreshModel: true, + refreshModel: true }, page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; @@ -43,7 +43,7 @@ export default class ActivitiesRoute extends Route { actionFilter, userFilter, versionFilter, - page, + page }: { batchFilter: any; actionFilter: any; @@ -53,6 +53,9 @@ export default class ActivitiesRoute extends Route { }, transition: Transition ) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectActivitiesQuery, @@ -61,7 +64,7 @@ export default class ActivitiesRoute extends Route { project: data.viewer.project, activities: data.viewer.project.activities, collaborators: data.viewer.project.collaborators, - versions: data.viewer.project.versions.entries, + versions: data.viewer.project.versions.entries }), options: { fetchPolicy: 'cache-and-network', @@ -72,9 +75,9 @@ export default class ActivitiesRoute extends Route { action: actionFilter === '' ? null : actionFilter, userId: userFilter === '' ? null : userFilter, versionId: versionFilter === '' ? null : versionFilter, - page, - }, - }, + page + } + } } ); diff --git a/webapp/app/pods/logged-in/project/activity/route.ts b/webapp/app/routes/logged-in/project/activity.ts similarity index 82% rename from webapp/app/pods/logged-in/project/activity/route.ts rename to webapp/app/routes/logged-in/project/activity.ts index cba38334d..eeb9a3187 100644 --- a/webapp/app/pods/logged-in/project/activity/route.ts +++ b/webapp/app/routes/logged-in/project/activity.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectActivityQuery from 'accent-webapp/queries/project-activity'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class Activity extends Route { @service('apollo-subscription') @@ -19,22 +19,25 @@ export default class Activity extends Route { subscription: Subscription; model({activityId}: {activityId: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectActivityQuery, { props: (data) => ({ project: data.viewer.project, - activity: data.viewer.project.activity, + activity: data.viewer.project.activity }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, - activityId, - }, - }, + activityId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/collaborators/route.ts b/webapp/app/routes/logged-in/project/collaborators.ts similarity index 82% rename from webapp/app/pods/logged-in/project/collaborators/route.ts rename to webapp/app/routes/logged-in/project/collaborators.ts index f7d052476..677a21298 100644 --- a/webapp/app/pods/logged-in/project/collaborators/route.ts +++ b/webapp/app/routes/logged-in/project/collaborators.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectCollaboratorsQuery from 'accent-webapp/queries/project-collaborators'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class CollaboratorsRoute extends Route { @service('apollo-subscription') @@ -19,20 +19,23 @@ export default class CollaboratorsRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectCollaboratorsQuery, { props: (data) => ({ - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/comments/route.ts b/webapp/app/routes/logged-in/project/comments.ts similarity index 78% rename from webapp/app/pods/logged-in/project/comments/route.ts rename to webapp/app/routes/logged-in/project/comments.ts index 5d69a97df..2f493353a 100644 --- a/webapp/app/pods/logged-in/project/comments/route.ts +++ b/webapp/app/routes/logged-in/project/comments.ts @@ -4,10 +4,10 @@ import Route from '@ember/routing/route'; import projectCommentsQuery from 'accent-webapp/queries/project-comments'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; -import CommentsController from 'accent-webapp/pods/logged-in/project/comments/controller'; +import Transition from '@ember/routing/transition'; +import CommentsController from 'accent-webapp/controllers/logged-in/project/comments'; export default class CommentsRoute extends Route { @service('apollo-subscription') @@ -18,13 +18,16 @@ export default class CommentsRoute extends Route { queryParams = { page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; model({page}: {page: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + const pageNumber = page ? parseInt(page, 10) : null; this.subscription = this.apolloSubscription.graphql( @@ -33,16 +36,16 @@ export default class CommentsRoute extends Route { { props: (data) => ({ comments: data.viewer.project.comments, - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, - page: pageNumber, - }, - }, + page: pageNumber + } + } } ); diff --git a/webapp/app/pods/logged-in/project/revision/conflicts/route.ts b/webapp/app/routes/logged-in/project/conflicts.ts similarity index 54% rename from webapp/app/pods/logged-in/project/revision/conflicts/route.ts rename to webapp/app/routes/logged-in/project/conflicts.ts index e4b2a3507..847156085 100644 --- a/webapp/app/pods/logged-in/project/revision/conflicts/route.ts +++ b/webapp/app/routes/logged-in/project/conflicts.ts @@ -4,11 +4,11 @@ import Route from '@ember/routing/route'; import translationsQuery from 'accent-webapp/queries/conflicts'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; -import ConflictsController from 'accent-webapp/pods/logged-in/project/revision/conflicts/controller'; +import Transition from '@ember/routing/transition'; +import ConflictsController from 'accent-webapp/controllers/logged-in/project/conflicts'; import RouterService from '@ember/routing/router-service'; export default class ConflictsRoute extends Route { @@ -23,57 +23,71 @@ export default class ConflictsRoute extends Route { queryParams = { query: { - refreshModel: true, + refreshModel: true }, page: { - refreshModel: true, + refreshModel: true }, document: { - refreshModel: true, + refreshModel: true }, version: { - refreshModel: true, + refreshModel: true }, + relatedRevisions: { + refreshModel: true + }, + isTextEmpty: { + refreshModel: true + }, + isTextNotEmpty: { + refreshModel: true + }, + isAddedLastSync: { + refreshModel: true + }, + isCommentedOn: { + refreshModel: true + }, + isConflicted: { + refreshModel: true + }, + isTranslated: { + refreshModel: true + } }; subscription: Subscription; - model( - { - query, - page, - document, - version, - }: {query: any; page: number; document: any; version: any}, - transition: Transition - ) { + model(params: any, transition: Transition) { + params.isTextEmpty = params.isTextEmpty === 'true' ? true : null; + params.isTextNotEmpty = params.isTextNotEmpty === 'true' ? true : null; + params.isAddedLastSync = params.isAddedLastSync === 'true' ? true : null; + params.isCommentedOn = params.isCommentedOn === 'true' ? true : null; + params.isTranslated = params.isTranslated === 'true' ? false : null; + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationsQuery, { props: (data) => ({ - revisionModel: this.modelFor('logged-in.project.revision'), + projectModel: this.modelFor('logged-in.project'), + prompts: data.viewer.project.prompts, documents: data.viewer.project.documents.entries, versions: data.viewer.project.versions.entries, - prompts: data.viewer.project.prompts, + revisions: data.viewer.project.revisions, + relatedRevisions: data.viewer.project.groupedTranslations.revisions, project: data.viewer.project, - translations: data.viewer.project.revision.translations, + groupedTranslations: data.viewer.project.groupedTranslations }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, - revisionId: this.routeParams.fetch( - transition, - 'logged-in.project.revision' - ).revisionId, - query, - page, - document, - version, - }, - }, + ...params + } + } } ); @@ -98,30 +112,4 @@ export default class ConflictsRoute extends Route { onRefresh() { this.refresh(); } - - @action - onRevisionChange({revisionId}: {revisionId: string}) { - const {project} = this.modelFor('logged-in.project') as {project: any}; - - this.apolloSubscription.clearSubscription(this.subscription); - - this.router.transitionTo( - 'logged-in.project.revision.conflicts', - project.id, - revisionId, - { - queryParams: this.fetchQueryParams( - this.controller as ConflictsController - ), - } - ); - } - - private fetchQueryParams(controller: ConflictsController) { - const query = controller.query; - - return { - query, - }; - } } diff --git a/webapp/app/pods/logged-in/project/edit/api-token/route.ts b/webapp/app/routes/logged-in/project/edit/api-token.ts similarity index 81% rename from webapp/app/pods/logged-in/project/edit/api-token/route.ts rename to webapp/app/routes/logged-in/project/edit/api-token.ts index f126021e3..9ad088cc7 100644 --- a/webapp/app/pods/logged-in/project/edit/api-token/route.ts +++ b/webapp/app/routes/logged-in/project/edit/api-token.ts @@ -4,9 +4,9 @@ import Route from '@ember/routing/route'; import projectApiTokenQuery from 'accent-webapp/queries/project-api-token'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class APITokenRoute extends Route { @service('apollo-subscription') @@ -18,6 +18,9 @@ export default class APITokenRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectApiTokenQuery, @@ -25,15 +28,15 @@ export default class APITokenRoute extends Route { props: (data) => ({ project: data.viewer.project, accessToken: data.viewer.accessToken, - apiToken: data.viewer.project.apiTokens, + apiToken: data.viewer.project.apiTokens }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/edit/badges/route.ts b/webapp/app/routes/logged-in/project/edit/badges.ts similarity index 82% rename from webapp/app/pods/logged-in/project/edit/badges/route.ts rename to webapp/app/routes/logged-in/project/edit/badges.ts index 4e8a38c87..83f32d222 100644 --- a/webapp/app/pods/logged-in/project/edit/badges/route.ts +++ b/webapp/app/routes/logged-in/project/edit/badges.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectEditQuery from 'accent-webapp/queries/project-edit'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class BadgesRoute extends Route { @service('apollo-subscription') @@ -19,20 +19,23 @@ export default class BadgesRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectEditQuery, { props: (data) => ({ - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/edit/index/route.ts b/webapp/app/routes/logged-in/project/edit/index.ts similarity index 82% rename from webapp/app/pods/logged-in/project/edit/index/route.ts rename to webapp/app/routes/logged-in/project/edit/index.ts index 357670959..cc0792a58 100644 --- a/webapp/app/pods/logged-in/project/edit/index/route.ts +++ b/webapp/app/routes/logged-in/project/edit/index.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectEditQuery from 'accent-webapp/queries/project-edit'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class ProjectEditIndexRoute extends Route { @service('apollo-subscription') @@ -19,20 +19,23 @@ export default class ProjectEditIndexRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectEditQuery, { props: (data) => ({ - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/edit/jipt/route.ts b/webapp/app/routes/logged-in/project/edit/jipt.ts similarity index 100% rename from webapp/app/pods/logged-in/project/edit/jipt/route.ts rename to webapp/app/routes/logged-in/project/edit/jipt.ts diff --git a/webapp/app/pods/logged-in/project/edit/machine-translations/route.ts b/webapp/app/routes/logged-in/project/edit/machine-translations.ts similarity index 82% rename from webapp/app/pods/logged-in/project/edit/machine-translations/route.ts rename to webapp/app/routes/logged-in/project/edit/machine-translations.ts index 7be5e829b..c8800d61f 100644 --- a/webapp/app/pods/logged-in/project/edit/machine-translations/route.ts +++ b/webapp/app/routes/logged-in/project/edit/machine-translations.ts @@ -4,9 +4,9 @@ import Route from '@ember/routing/route'; import projectMachineTranslationsConfigQuery from 'accent-webapp/queries/project-machine-translations-config'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class MachineTranslationsRoute extends Route { @service('apollo-subscription') @@ -18,20 +18,23 @@ export default class MachineTranslationsRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectMachineTranslationsConfigQuery, { props: (data) => ({ - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/edit/prompts/route.ts b/webapp/app/routes/logged-in/project/edit/prompts.ts similarity index 81% rename from webapp/app/pods/logged-in/project/edit/prompts/route.ts rename to webapp/app/routes/logged-in/project/edit/prompts.ts index 6dad2f21d..7d0d5de4e 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/route.ts +++ b/webapp/app/routes/logged-in/project/edit/prompts.ts @@ -4,9 +4,9 @@ import Route from '@ember/routing/route'; import projectPromptConfigQuery from 'accent-webapp/queries/project-prompt-config'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class PomptsRoute extends Route { @service('apollo-subscription') @@ -18,21 +18,24 @@ export default class PomptsRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectPromptConfigQuery, { props: (data) => ({ project: data.viewer.project, - prompts: data.viewer.project.prompts, + prompts: data.viewer.project.prompts }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/edit/prompts/edit/route.ts b/webapp/app/routes/logged-in/project/edit/prompts/edit.ts similarity index 94% rename from webapp/app/pods/logged-in/project/edit/prompts/edit/route.ts rename to webapp/app/routes/logged-in/project/edit/prompts/edit.ts index bb3ea8023..06a3cde51 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/edit/route.ts +++ b/webapp/app/routes/logged-in/project/edit/prompts/edit.ts @@ -5,7 +5,7 @@ export default class PromptEditRoute extends Route { return { projectModel: this.modelFor('logged-in.project'), promptsModel: this.modelFor('logged-in.project.edit.prompts'), - promptId, + promptId }; } } diff --git a/webapp/app/pods/logged-in/project/edit/prompts/new/route.ts b/webapp/app/routes/logged-in/project/edit/prompts/new.ts similarity index 70% rename from webapp/app/pods/logged-in/project/edit/prompts/new/route.ts rename to webapp/app/routes/logged-in/project/edit/prompts/new.ts index cce03c27d..31cf4fa0f 100644 --- a/webapp/app/pods/logged-in/project/edit/prompts/new/route.ts +++ b/webapp/app/routes/logged-in/project/edit/prompts/new.ts @@ -3,7 +3,7 @@ import Route from '@ember/routing/route'; export default class PromptNewRoute extends Route { model() { return { - projectModel: this.modelFor('logged-in.project'), + projectModel: this.modelFor('logged-in.project') }; } } diff --git a/webapp/app/pods/logged-in/project/edit/service-integrations/route.ts b/webapp/app/routes/logged-in/project/edit/service-integrations.ts similarity index 82% rename from webapp/app/pods/logged-in/project/edit/service-integrations/route.ts rename to webapp/app/routes/logged-in/project/edit/service-integrations.ts index 2d464cc3f..172a517fc 100644 --- a/webapp/app/pods/logged-in/project/edit/service-integrations/route.ts +++ b/webapp/app/routes/logged-in/project/edit/service-integrations.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectServiceIntegrationsQuery from 'accent-webapp/queries/project-service-integrations'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class ServiceIntegrationsController extends Route { @service('apollo-subscription') @@ -19,20 +19,23 @@ export default class ServiceIntegrationsController extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectServiceIntegrationsQuery, { props: (data) => ({ - project: data.viewer.project, + project: data.viewer.project }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/files/route.ts b/webapp/app/routes/logged-in/project/files.ts similarity index 81% rename from webapp/app/pods/logged-in/project/files/route.ts rename to webapp/app/routes/logged-in/project/files.ts index 80ae2705e..3d12d9317 100644 --- a/webapp/app/pods/logged-in/project/files/route.ts +++ b/webapp/app/routes/logged-in/project/files.ts @@ -5,9 +5,9 @@ import Route from '@ember/routing/route'; import projectDocumentsQuery from 'accent-webapp/queries/project-documents'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class FilesRoute extends Route { @service('apollo-subscription') @@ -18,11 +18,11 @@ export default class FilesRoute extends Route { queryParams = { page: { - refreshModel: true, + refreshModel: true }, excludeEmptyTranslations: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; @@ -30,10 +30,13 @@ export default class FilesRoute extends Route { model( { page, - excludeEmptyTranslations, + excludeEmptyTranslations }: {page: string; excludeEmptyTranslations: boolean}, transition: Transition ) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + const pageNumber = page ? parseInt(page, 10) : null; this.subscription = this.apolloSubscription.graphql( @@ -43,7 +46,7 @@ export default class FilesRoute extends Route { props: (data) => ({ project: data.viewer.project, documents: data.viewer.project.documents, - versions: data.viewer.project.versions, + versions: data.viewer.project.versions }), options: { fetchPolicy: 'cache-and-network', @@ -51,9 +54,9 @@ export default class FilesRoute extends Route { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, excludeEmptyTranslations, - page: pageNumber, - }, - }, + page: pageNumber + } + } } ); diff --git a/webapp/app/pods/logged-in/project/files/add-translations/route.ts b/webapp/app/routes/logged-in/project/files/add-translations.ts similarity index 77% rename from webapp/app/pods/logged-in/project/files/add-translations/route.ts rename to webapp/app/routes/logged-in/project/files/add-translations.ts index 19a000612..161f0bb8e 100644 --- a/webapp/app/pods/logged-in/project/files/add-translations/route.ts +++ b/webapp/app/routes/logged-in/project/files/add-translations.ts @@ -1,12 +1,12 @@ import Route from '@ember/routing/route'; -import AddTranslationsController from 'accent-webapp/pods/logged-in/project/files/add-translations/controller'; +import AddTranslationsController from 'accent-webapp/controllers/logged-in/project/files/add-translations'; export default class AddTranslationsRoute extends Route { model({fileId}: {fileId: string}) { return { projectModel: this.modelFor('logged-in.project'), fileModel: this.modelFor('logged-in.project.files'), - fileId, + fileId }; } diff --git a/webapp/app/pods/logged-in/project/files/export-all/route.ts b/webapp/app/routes/logged-in/project/files/export-all.ts similarity index 66% rename from webapp/app/pods/logged-in/project/files/export-all/route.ts rename to webapp/app/routes/logged-in/project/files/export-all.ts index 9a74aa97e..8d3a81858 100644 --- a/webapp/app/pods/logged-in/project/files/export-all/route.ts +++ b/webapp/app/routes/logged-in/project/files/export-all.ts @@ -1,8 +1,8 @@ import {inject as service} from '@ember/service'; import Route from '@ember/routing/route'; import Exporter from 'accent-webapp/services/exporter'; -import ExportController from 'accent-webapp/pods/logged-in/project/files/export/controller'; -import Transition from '@ember/routing/-private/transition'; +import ExportController from 'accent-webapp/controllers/logged-in/project/files/export'; +import Transition from '@ember/routing/transition'; export default class ExportRoute extends Route { @service('exporter') @@ -10,33 +10,33 @@ export default class ExportRoute extends Route { queryParams = { revisionFilter: { - refreshModel: true, + refreshModel: true }, documentFormatFilter: { - refreshModel: true, + refreshModel: true }, versionFilter: { - refreshModel: true, + refreshModel: true }, orderByFilter: { - refreshModel: true, + refreshModel: true }, isTextEmpty: { - refreshModel: true, + refreshModel: true }, isAddedLastSync: { - refreshModel: true, + refreshModel: true }, isConflicted: { - refreshModel: true, - }, + refreshModel: true + } }; model(_params: unknown, transition: Transition) { return { from: transition.from, projectModel: this.modelFor('logged-in.project'), - fileModel: this.modelFor('logged-in.project.files'), + fileModel: this.modelFor('logged-in.project.files') }; } diff --git a/webapp/app/pods/logged-in/project/files/export/route.ts b/webapp/app/routes/logged-in/project/files/export.ts similarity index 70% rename from webapp/app/pods/logged-in/project/files/export/route.ts rename to webapp/app/routes/logged-in/project/files/export.ts index 5c1a8ef80..f942d9a4b 100644 --- a/webapp/app/pods/logged-in/project/files/export/route.ts +++ b/webapp/app/routes/logged-in/project/files/export.ts @@ -1,8 +1,8 @@ import {inject as service} from '@ember/service'; import Route from '@ember/routing/route'; import Exporter from 'accent-webapp/services/exporter'; -import ExportController from 'accent-webapp/pods/logged-in/project/files/export/controller'; -import Transition from '@ember/routing/-private/transition'; +import ExportController from 'accent-webapp/controllers/logged-in/project/files/export'; +import Transition from '@ember/routing/transition'; export default class ExportRoute extends Route { @service('exporter') @@ -10,26 +10,26 @@ export default class ExportRoute extends Route { queryParams = { revisionFilter: { - refreshModel: true, + refreshModel: true }, documentFormatFilter: { - refreshModel: true, + refreshModel: true }, versionFilter: { - refreshModel: true, + refreshModel: true }, orderByFilter: { - refreshModel: true, + refreshModel: true }, isTextEmpty: { - refreshModel: true, + refreshModel: true }, isAddedLastSync: { - refreshModel: true, + refreshModel: true }, isConflicted: { - refreshModel: true, - }, + refreshModel: true + } }; model({fileId}: {fileId: string}, transition: Transition) { @@ -37,7 +37,7 @@ export default class ExportRoute extends Route { from: transition.from, projectModel: this.modelFor('logged-in.project'), fileModel: this.modelFor('logged-in.project.files'), - fileId, + fileId }; } diff --git a/webapp/app/pods/logged-in/project/files/jipt/route.ts b/webapp/app/routes/logged-in/project/files/jipt.ts similarity index 82% rename from webapp/app/pods/logged-in/project/files/jipt/route.ts rename to webapp/app/routes/logged-in/project/files/jipt.ts index b36bdb0ac..70b9b405e 100644 --- a/webapp/app/pods/logged-in/project/files/jipt/route.ts +++ b/webapp/app/routes/logged-in/project/files/jipt.ts @@ -1,7 +1,7 @@ import {inject as service} from '@ember/service'; import Route from '@ember/routing/route'; import Exporter from 'accent-webapp/services/exporter'; -import JIPTController from 'accent-webapp/pods/logged-in/project/files/jipt/controller'; +import JIPTController from 'accent-webapp/controllers/logged-in/project/files/jipt'; export default class JIPTRoute extends Route { @service('exporter') @@ -9,15 +9,15 @@ export default class JIPTRoute extends Route { queryParams = { documentFormatFilter: { - refreshModel: true, - }, + refreshModel: true + } }; model({fileId}: {fileId: string}) { return { projectModel: this.modelFor('logged-in.project'), fileModel: this.modelFor('logged-in.project.files'), - fileId, + fileId }; } diff --git a/webapp/app/pods/logged-in/project/files/machine-translations/route.ts b/webapp/app/routes/logged-in/project/files/machine-translations.ts similarity index 77% rename from webapp/app/pods/logged-in/project/files/machine-translations/route.ts rename to webapp/app/routes/logged-in/project/files/machine-translations.ts index 53705aee5..3bff2e4e9 100644 --- a/webapp/app/pods/logged-in/project/files/machine-translations/route.ts +++ b/webapp/app/routes/logged-in/project/files/machine-translations.ts @@ -1,12 +1,12 @@ import Route from '@ember/routing/route'; -import MachineTranslationsController from 'accent-webapp/pods/logged-in/project/files/machine-translations/controller'; +import MachineTranslationsController from 'accent-webapp/controllers/logged-in/project/files/machine-translations'; export default class MachineTranslationsRoute extends Route { model({fileId}: {fileId: string}) { return { projectModel: this.modelFor('logged-in.project'), fileModel: this.modelFor('logged-in.project.files'), - fileId, + fileId }; } diff --git a/webapp/app/pods/logged-in/project/files/new-machine-translations/route.ts b/webapp/app/routes/logged-in/project/files/new-machine-translations.ts similarity index 69% rename from webapp/app/pods/logged-in/project/files/new-machine-translations/route.ts rename to webapp/app/routes/logged-in/project/files/new-machine-translations.ts index f1adb602e..2b162fcc5 100644 --- a/webapp/app/pods/logged-in/project/files/new-machine-translations/route.ts +++ b/webapp/app/routes/logged-in/project/files/new-machine-translations.ts @@ -1,5 +1,5 @@ import Route from '@ember/routing/route'; -import NewMachineTranslationsController from 'accent-webapp/pods/logged-in/project/files/new-machine-translations/controller'; +import NewMachineTranslationsController from 'accent-webapp/controllers/logged-in/project/files/new-machine-translations'; export default class NewMachineTranslationsRoute extends Route { resetController( diff --git a/webapp/app/pods/logged-in/project/files/new-sync/route.ts b/webapp/app/routes/logged-in/project/files/new-sync.ts similarity index 76% rename from webapp/app/pods/logged-in/project/files/new-sync/route.ts rename to webapp/app/routes/logged-in/project/files/new-sync.ts index 0d360b5ff..ae48d730d 100644 --- a/webapp/app/pods/logged-in/project/files/new-sync/route.ts +++ b/webapp/app/routes/logged-in/project/files/new-sync.ts @@ -1,5 +1,5 @@ import Route from '@ember/routing/route'; -import NewSyncController from 'accent-webapp/pods/logged-in/project/files/new-sync/controller'; +import NewSyncController from 'accent-webapp/controllers/logged-in/project/files/new-sync'; export default class NewSyncRoute extends Route { model() { diff --git a/webapp/app/pods/logged-in/project/files/sync/route.ts b/webapp/app/routes/logged-in/project/files/sync.ts similarity index 80% rename from webapp/app/pods/logged-in/project/files/sync/route.ts rename to webapp/app/routes/logged-in/project/files/sync.ts index 9d2774dbb..4b47f8f7b 100644 --- a/webapp/app/pods/logged-in/project/files/sync/route.ts +++ b/webapp/app/routes/logged-in/project/files/sync.ts @@ -1,12 +1,12 @@ import Route from '@ember/routing/route'; -import SyncController from 'accent-webapp/pods/logged-in/project/files/sync/controller'; +import SyncController from 'accent-webapp/controllers/logged-in/project/files/sync'; export default class SyncRoute extends Route { model({fileId}: {fileId: string}) { return { projectModel: this.modelFor('logged-in.project'), fileModel: this.modelFor('logged-in.project.files'), - fileId, + fileId }; } diff --git a/webapp/app/pods/logged-in/project/index/route.ts b/webapp/app/routes/logged-in/project/index.ts similarity index 84% rename from webapp/app/pods/logged-in/project/index/route.ts rename to webapp/app/routes/logged-in/project/index.ts index fe94ec948..0e3b95c23 100644 --- a/webapp/app/pods/logged-in/project/index/route.ts +++ b/webapp/app/routes/logged-in/project/index.ts @@ -4,9 +4,9 @@ import Route from '@ember/routing/route'; import projectDashboardQuery from 'accent-webapp/queries/project-dashboard'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class ProjectIndexRoute extends Route { @service('apollo-subscription') @@ -18,6 +18,9 @@ export default class ProjectIndexRoute extends Route { subscription: Subscription; model(_params: object, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + const props = (data: any) => ({project: data.viewer.project}); this.subscription = this.apolloSubscription.graphql( @@ -29,9 +32,9 @@ export default class ProjectIndexRoute extends Route { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/manage-languages/route.ts b/webapp/app/routes/logged-in/project/manage-languages.ts similarity index 79% rename from webapp/app/pods/logged-in/project/manage-languages/route.ts rename to webapp/app/routes/logged-in/project/manage-languages.ts index 4212dba94..e01c7b752 100644 --- a/webapp/app/pods/logged-in/project/manage-languages/route.ts +++ b/webapp/app/routes/logged-in/project/manage-languages.ts @@ -5,10 +5,10 @@ import Route from '@ember/routing/route'; import projectNewLanguageQuery from 'accent-webapp/queries/project-new-language'; import RouteParams from 'accent-webapp/services/route-params'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import Transition from '@ember/routing/-private/transition'; -import ManageLanguagesController from 'accent-webapp/pods/logged-in/project/manage-languages/controller'; +import Transition from '@ember/routing/transition'; +import ManageLanguagesController from 'accent-webapp/controllers/logged-in/project/manage-languages'; export default class ManageLanguagesRoute extends Route { @service('apollo-subscription') @@ -20,21 +20,24 @@ export default class ManageLanguagesRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), projectNewLanguageQuery, { props: (data) => ({ project: data.viewer.project, - languages: data.languages.entries, + languages: data.languages.entries }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/manage-languages/edit/route.ts b/webapp/app/routes/logged-in/project/manage-languages/edit.ts similarity index 93% rename from webapp/app/pods/logged-in/project/manage-languages/edit/route.ts rename to webapp/app/routes/logged-in/project/manage-languages/edit.ts index 71f27fa18..0b313027c 100644 --- a/webapp/app/pods/logged-in/project/manage-languages/edit/route.ts +++ b/webapp/app/routes/logged-in/project/manage-languages/edit.ts @@ -4,7 +4,7 @@ export default class ManageLanguagesEditController extends Route { model({revisionId}: {revisionId: string}) { return { revisionsModel: this.modelFor('logged-in.project.manage-languages'), - revisionId, + revisionId }; } } diff --git a/webapp/app/pods/logged-in/project/revision/route.ts b/webapp/app/routes/logged-in/project/revision.ts similarity index 100% rename from webapp/app/pods/logged-in/project/revision/route.ts rename to webapp/app/routes/logged-in/project/revision.ts diff --git a/webapp/app/pods/logged-in/project/revision/lint-translations/route.ts b/webapp/app/routes/logged-in/project/revision/lint-translations.ts similarity index 84% rename from webapp/app/pods/logged-in/project/revision/lint-translations/route.ts rename to webapp/app/routes/logged-in/project/revision/lint-translations.ts index 8f44705f6..d70291027 100644 --- a/webapp/app/pods/logged-in/project/revision/lint-translations/route.ts +++ b/webapp/app/routes/logged-in/project/revision/lint-translations.ts @@ -5,11 +5,11 @@ import GlobalState from 'accent-webapp/services/global-state'; import lintQuery from 'accent-webapp/queries/lint-translations'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; -import LintTranslationsController from 'accent-webapp/pods/logged-in/project/revision/lint-translations/controller'; +import Transition from '@ember/routing/transition'; +import LintTranslationsController from 'accent-webapp/controllers/logged-in/project/revision/lint-translations'; import RouterService from '@ember/routing/router-service'; export default class LintRoute extends Route { @@ -29,20 +29,23 @@ export default class LintRoute extends Route { queryParams = { documentFilter: { - refreshModel: true, + refreshModel: true }, versionFilter: { - refreshModel: true, + refreshModel: true }, ruleFilter: { - refreshModel: true, + refreshModel: true }, query: { - refreshModel: true, - }, + refreshModel: true + } }; model(params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), lintQuery, @@ -51,7 +54,7 @@ export default class LintRoute extends Route { project: data.viewer.project, documents: data.viewer.project.documents, versions: data.viewer.project.versions, - lintTranslations: data.viewer.project.lintTranslations, + lintTranslations: data.viewer.project.lintTranslations }), options: { fetchPolicy: 'cache-and-network', @@ -65,9 +68,9 @@ export default class LintRoute extends Route { ruleIds: params.ruleFilter, query: params.query, projectId: this.routeParams.fetch(transition, 'logged-in.project') - .projectId, - }, - }, + .projectId + } + } } ); @@ -87,7 +90,7 @@ export default class LintRoute extends Route { { queryParams: this.fetchQueryParams( this.controller as LintTranslationsController - ), + ) } ); } @@ -96,7 +99,7 @@ export default class LintRoute extends Route { const query = controller.query; return { - query, + query }; } } diff --git a/webapp/app/pods/logged-in/project/revision/translations/route.ts b/webapp/app/routes/logged-in/project/revision/translations.ts similarity index 84% rename from webapp/app/pods/logged-in/project/revision/translations/route.ts rename to webapp/app/routes/logged-in/project/revision/translations.ts index efc385578..64219f916 100644 --- a/webapp/app/pods/logged-in/project/revision/translations/route.ts +++ b/webapp/app/routes/logged-in/project/revision/translations.ts @@ -4,12 +4,12 @@ import Route from '@ember/routing/route'; import translationsQuery from 'accent-webapp/queries/translations'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import TranslationsController from 'accent-webapp/pods/logged-in/project/revision/translations/controller'; +import TranslationsController from 'accent-webapp/controllers/logged-in/project/revision/translations'; import RouterService from '@ember/routing/router-service'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class TranslationsRoute extends Route { @service('apollo-subscription') @@ -23,40 +23,43 @@ export default class TranslationsRoute extends Route { queryParams = { query: { - refreshModel: true, + refreshModel: true }, page: { - refreshModel: true, + refreshModel: true }, document: { - refreshModel: true, + refreshModel: true }, version: { - refreshModel: true, + refreshModel: true }, isTextEmpty: { - refreshModel: true, + refreshModel: true }, isTextNotEmpty: { - refreshModel: true, + refreshModel: true }, isAddedLastSync: { - refreshModel: true, + refreshModel: true }, isCommentedOn: { - refreshModel: true, + refreshModel: true }, isConflicted: { - refreshModel: true, + refreshModel: true }, isTranslated: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; model(params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + params.isTextEmpty = params.isTextEmpty === 'true' ? true : null; params.isTextNotEmpty = params.isTextNotEmpty === 'true' ? true : null; params.isAddedLastSync = params.isAddedLastSync === 'true' ? true : null; @@ -74,7 +77,7 @@ export default class TranslationsRoute extends Route { prompts: data.viewer.project.prompts, documents: data.viewer.project.documents.entries, versions: data.viewer.project.versions.entries, - translations: data.viewer.project.revision.translations, + translations: data.viewer.project.revision.translations }), options: { fetchPolicy: 'cache-and-network', @@ -85,9 +88,9 @@ export default class TranslationsRoute extends Route { transition, 'logged-in.project.revision' ).revisionId, - ...params, - }, - }, + ...params + } + } } ); @@ -121,7 +124,7 @@ export default class TranslationsRoute extends Route { { queryParams: this.fetchQueryParams( this.controller as TranslationsController - ), + ) } ); } @@ -130,7 +133,7 @@ export default class TranslationsRoute extends Route { const query = controller.query; return { - query, + query }; } } diff --git a/webapp/app/pods/logged-in/project/translation/route.ts b/webapp/app/routes/logged-in/project/translation.ts similarity index 83% rename from webapp/app/pods/logged-in/project/translation/route.ts rename to webapp/app/routes/logged-in/project/translation.ts index 4dde10485..636d9c0ed 100644 --- a/webapp/app/pods/logged-in/project/translation/route.ts +++ b/webapp/app/routes/logged-in/project/translation.ts @@ -3,10 +3,10 @@ import Route from '@ember/routing/route'; import translationQuery from 'accent-webapp/queries/translation'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class TranslationsRoute extends Route { @service('apollo-subscription') @@ -18,6 +18,9 @@ export default class TranslationsRoute extends Route { subscription: Subscription; model({translationId}: {translationId: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationQuery, @@ -28,16 +31,16 @@ export default class TranslationsRoute extends Route { translation: data.viewer.project.translation, prompts: data.viewer.project.prompts, relatedTranslations: - data.viewer.project.translation.relatedTranslations, + data.viewer.project.translation.relatedTranslations }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, - translationId, - }, - }, + translationId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/translation/activities/route.ts b/webapp/app/routes/logged-in/project/translation/activities.ts similarity index 83% rename from webapp/app/pods/logged-in/project/translation/activities/route.ts rename to webapp/app/routes/logged-in/project/translation/activities.ts index f8da240a1..479d50b53 100644 --- a/webapp/app/pods/logged-in/project/translation/activities/route.ts +++ b/webapp/app/routes/logged-in/project/translation/activities.ts @@ -4,11 +4,11 @@ import Route from '@ember/routing/route'; import translationActivitiesQuery from 'accent-webapp/queries/translation-activities'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; -import ActivitiesController from 'accent-webapp/pods/logged-in/project/activities/controller'; +import Transition from '@ember/routing/transition'; +import ActivitiesController from 'accent-webapp/controllers/logged-in/project/activities'; export default class ActivitiesRoute extends Route { @service('apollo-subscription') @@ -19,20 +19,23 @@ export default class ActivitiesRoute extends Route { queryParams = { page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; model({page}: {page: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationActivitiesQuery, { props: (data) => ({ project: data.viewer.project, - activities: data.viewer.project.translation.activities, + activities: data.viewer.project.translation.activities }), options: { fetchPolicy: 'cache-and-network', @@ -43,9 +46,9 @@ export default class ActivitiesRoute extends Route { transition, 'logged-in.project.translation' ).translationId, - page, - }, - }, + page + } + } } ); diff --git a/webapp/app/pods/logged-in/project/translation/comments/route.ts b/webapp/app/routes/logged-in/project/translation/comments.ts similarity index 84% rename from webapp/app/pods/logged-in/project/translation/comments/route.ts rename to webapp/app/routes/logged-in/project/translation/comments.ts index 62330fc49..acac08777 100644 --- a/webapp/app/pods/logged-in/project/translation/comments/route.ts +++ b/webapp/app/routes/logged-in/project/translation/comments.ts @@ -4,11 +4,11 @@ import Route from '@ember/routing/route'; import translationCommentsQuery from 'accent-webapp/queries/translation-comments'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; -import CommentsController from 'accent-webapp/pods/logged-in/project/comments/controller'; +import Transition from '@ember/routing/transition'; +import CommentsController from 'accent-webapp/controllers/logged-in/project/comments'; export default class CommentsRoute extends Route { @service('apollo-subscription') @@ -19,13 +19,16 @@ export default class CommentsRoute extends Route { queryParams = { page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; model({page}: {page: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + const pageNumber = page ? parseInt(page, 10) : null; this.subscription = this.apolloSubscription.graphql( @@ -37,7 +40,7 @@ export default class CommentsRoute extends Route { comments: data.viewer.project.translation.comments, collaborators: data.viewer.project.collaborators, commentsSubscriptions: - data.viewer.project.translation.commentsSubscriptions, + data.viewer.project.translation.commentsSubscriptions }), options: { fetchPolicy: 'cache-and-network', @@ -48,9 +51,9 @@ export default class CommentsRoute extends Route { transition, 'logged-in.project.translation' ).translationId, - page: pageNumber, - }, - }, + page: pageNumber + } + } } ); diff --git a/webapp/app/pods/logged-in/project/translation/editions/route.ts b/webapp/app/routes/logged-in/project/translation/editions.ts similarity index 81% rename from webapp/app/pods/logged-in/project/translation/editions/route.ts rename to webapp/app/routes/logged-in/project/translation/editions.ts index 38a4489ae..50d3ec402 100644 --- a/webapp/app/pods/logged-in/project/translation/editions/route.ts +++ b/webapp/app/routes/logged-in/project/translation/editions.ts @@ -3,12 +3,12 @@ import Route from '@ember/routing/route'; import translationEditionsQuery from 'accent-webapp/queries/translation-editions'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import TranslationsController from 'accent-webapp/pods/logged-in/project/revision/translations/controller'; +import TranslationsController from 'accent-webapp/controllers/logged-in/project/revision/translations'; import RouterService from '@ember/routing/router-service'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class TranslationEditionsRoute extends Route { @service('apollo-subscription') @@ -23,15 +23,18 @@ export default class TranslationEditionsRoute extends Route { subscription: Subscription; model(_params: any, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), translationEditionsQuery, { props: (data) => ({ - revisionModel: this.modelFor('logged-in.project.revision'), + translationModel: this.modelFor('logged-in.project.translation'), project: data.viewer.project, prompts: data.viewer.project.prompts, - translations: data.viewer.project.translation.editions, + translations: data.viewer.project.translation.editions }), options: { fetchPolicy: 'cache-and-network', @@ -41,9 +44,9 @@ export default class TranslationEditionsRoute extends Route { translationId: this.routeParams.fetch( transition, 'logged-in.project.translation' - ).translationId, - }, - }, + ).translationId + } + } } ); diff --git a/webapp/app/pods/logged-in/project/translation/index/route.ts b/webapp/app/routes/logged-in/project/translation/index.ts similarity index 100% rename from webapp/app/pods/logged-in/project/translation/index/route.ts rename to webapp/app/routes/logged-in/project/translation/index.ts diff --git a/webapp/app/pods/logged-in/project/versions/route.ts b/webapp/app/routes/logged-in/project/versions.ts similarity index 82% rename from webapp/app/pods/logged-in/project/versions/route.ts rename to webapp/app/routes/logged-in/project/versions.ts index dac0c5b3b..0487e8854 100644 --- a/webapp/app/pods/logged-in/project/versions/route.ts +++ b/webapp/app/routes/logged-in/project/versions.ts @@ -4,10 +4,10 @@ import Route from '@ember/routing/route'; import projectVersionsQuery from 'accent-webapp/queries/project-versions'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import RouteParams from 'accent-webapp/services/route-params'; -import Transition from '@ember/routing/-private/transition'; +import Transition from '@ember/routing/transition'; export default class VersionsRoute extends Route { @service('apollo-subscription') @@ -18,13 +18,16 @@ export default class VersionsRoute extends Route { queryParams = { page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; model({page}: {page: string}, transition: Transition) { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + const pageNumber = page ? parseInt(page, 10) : null; this.subscription = this.apolloSubscription.graphql( @@ -34,16 +37,16 @@ export default class VersionsRoute extends Route { props: (data) => ({ project: data.viewer.project, versions: data.viewer.project.versions, - documents: data.viewer.project.documents, + documents: data.viewer.project.documents }), options: { fetchPolicy: 'cache-and-network', variables: { projectId: this.routeParams.fetch(transition, 'logged-in.project') .projectId, - page: pageNumber, - }, - }, + page: pageNumber + } + } } ); diff --git a/webapp/app/pods/logged-in/project/versions/edit/route.ts b/webapp/app/routes/logged-in/project/versions/edit.ts similarity index 94% rename from webapp/app/pods/logged-in/project/versions/edit/route.ts rename to webapp/app/routes/logged-in/project/versions/edit.ts index f03a94a26..9a88e1f5a 100644 --- a/webapp/app/pods/logged-in/project/versions/edit/route.ts +++ b/webapp/app/routes/logged-in/project/versions/edit.ts @@ -5,7 +5,7 @@ export default class EditRoute extends Route { return { projectModel: this.modelFor('logged-in.project'), versionModel: this.modelFor('logged-in.project.versions'), - versionId, + versionId }; } } diff --git a/webapp/app/pods/logged-in/project/versions/export/route.ts b/webapp/app/routes/logged-in/project/versions/export.ts similarity index 76% rename from webapp/app/pods/logged-in/project/versions/export/route.ts rename to webapp/app/routes/logged-in/project/versions/export.ts index c18941338..829fdb6d8 100644 --- a/webapp/app/pods/logged-in/project/versions/export/route.ts +++ b/webapp/app/routes/logged-in/project/versions/export.ts @@ -1,7 +1,7 @@ import {inject as service} from '@ember/service'; import Route from '@ember/routing/route'; import Exporter from 'accent-webapp/services/exporter'; -import ExportController from 'accent-webapp/pods/logged-in/project/versions/export/controller'; +import ExportController from 'accent-webapp/controllers/logged-in/project/versions/export'; export default class ExportRoute extends Route { @service('exporter') @@ -9,24 +9,24 @@ export default class ExportRoute extends Route { queryParams = { revisionFilter: { - refreshModel: true, + refreshModel: true }, documentFilter: { - refreshModel: true, + refreshModel: true }, documentFormatFilter: { - refreshModel: true, + refreshModel: true }, orderByFilter: { - refreshModel: true, - }, + refreshModel: true + } }; model({versionId}: {versionId: string}) { return { projectModel: this.modelFor('logged-in.project'), versionModel: this.modelFor('logged-in.project.versions'), - versionId, + versionId }; } diff --git a/webapp/app/pods/logged-in/project/versions/new/route.ts b/webapp/app/routes/logged-in/project/versions/new.ts similarity index 76% rename from webapp/app/pods/logged-in/project/versions/new/route.ts rename to webapp/app/routes/logged-in/project/versions/new.ts index 526d53b5b..7acf3f020 100644 --- a/webapp/app/pods/logged-in/project/versions/new/route.ts +++ b/webapp/app/routes/logged-in/project/versions/new.ts @@ -1,5 +1,5 @@ import Route from '@ember/routing/route'; -import NewController from 'accent-webapp/pods/logged-in/project/versions/new/controller'; +import NewController from 'accent-webapp/controllers/logged-in/project/versions/new'; export default class NewRoute extends Route { model() { diff --git a/webapp/app/pods/logged-in/projects/route.ts b/webapp/app/routes/logged-in/projects.ts similarity index 85% rename from webapp/app/pods/logged-in/projects/route.ts rename to webapp/app/routes/logged-in/projects.ts index a8ddf3cb7..65141c0c3 100644 --- a/webapp/app/pods/logged-in/projects/route.ts +++ b/webapp/app/routes/logged-in/projects.ts @@ -4,10 +4,11 @@ import {inject as service} from '@ember/service'; import projectsQuery from 'accent-webapp/queries/projects'; import Session from 'accent-webapp/services/session'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; -import ProjectsController from 'accent-webapp/pods/logged-in/projects/controller'; +import ProjectsController from 'accent-webapp/controllers/logged-in/projects'; import RecentProjects from 'accent-webapp/services/recent-projects'; +import RouterService from '@ember/routing/router-service'; const transformData = (data: any, recentProjectIds: string[]) => { const permissions = data.viewer.permissions.reduce( @@ -30,7 +31,7 @@ const transformData = (data: any, recentProjectIds: string[]) => { projects: data.viewer.projects, languages: data.languages.entries, recentProjects, - permissions, + permissions }; }; @@ -44,13 +45,16 @@ export default class ProjectsRoute extends Route { @service('apollo-subscription') apolloSubscription: ApolloSubscription; + @service('router') + router: RouterService; + queryParams = { query: { - refreshModel: true, + refreshModel: true }, page: { - refreshModel: true, - }, + refreshModel: true + } }; subscription: Subscription; @@ -80,9 +84,9 @@ export default class ProjectsRoute extends Route { variables: { page, query, - nodeIds: recentProjectIds, - }, - }, + nodeIds: recentProjectIds + } + } } ); @@ -91,7 +95,7 @@ export default class ProjectsRoute extends Route { redirect() { if (!this.session.isAuthenticated) { - this.transitionTo('login'); + this.router.transitionTo('login'); } } diff --git a/webapp/app/pods/logged-in/projects/new/route.ts b/webapp/app/routes/logged-in/projects/new.ts similarity index 79% rename from webapp/app/pods/logged-in/projects/new/route.ts rename to webapp/app/routes/logged-in/projects/new.ts index e8274bd6b..a8fe80172 100644 --- a/webapp/app/pods/logged-in/projects/new/route.ts +++ b/webapp/app/routes/logged-in/projects/new.ts @@ -1,5 +1,5 @@ import Route from '@ember/routing/route'; -import Controller from './controller'; +import Controller from 'accent-webapp/controllers/logged-in/projects/new'; export default class ProjectsNewRoute extends Route { model() { diff --git a/webapp/app/pods/login/route.ts b/webapp/app/routes/login.ts similarity index 76% rename from webapp/app/pods/login/route.ts rename to webapp/app/routes/login.ts index c61d2ccd5..ac4a9ad00 100644 --- a/webapp/app/pods/login/route.ts +++ b/webapp/app/routes/login.ts @@ -3,20 +3,27 @@ import Route from '@ember/routing/route'; import authenticationProvidersQuery from 'accent-webapp/queries/authentication-providers'; import ApolloSubscription, { - Subscription, + Subscription } from 'accent-webapp/services/apollo-subscription'; import Session from 'accent-webapp/services/session'; +import RouterService from '@ember/routing/router-service'; export default class LoginRoute extends Route { @service('session') session: Session; + @service('router') + router: RouterService; + @service('apollo-subscription') apolloSubscription: ApolloSubscription; subscription: Subscription; model() { + if (this.subscription) + this.apolloSubscription.clearSubscription(this.subscription); + this.subscription = this.apolloSubscription.graphql( () => this.modelFor(this.routeName), authenticationProvidersQuery, @@ -32,7 +39,7 @@ export default class LoginRoute extends Route { redirect() { if (this.session.isAuthenticated) { - this.transitionTo('logged-in.projects'); + this.router.transitionTo('logged-in.projects'); } } } diff --git a/webapp/app/services/apollo-mutate.ts b/webapp/app/services/apollo-mutate.ts index 966c43d2c..3912cb9de 100644 --- a/webapp/app/services/apollo-mutate.ts +++ b/webapp/app/services/apollo-mutate.ts @@ -15,9 +15,7 @@ export default class ApolloMutate extends Service { const operationName = Object.keys(data)[0]; if (!data[operationName]?.errors?.length) { - data[operationName].errors = null; - - return data[operationName]; + return {...data[operationName], errors: null}; } return data[operationName]; diff --git a/webapp/app/services/apollo-subscription.ts b/webapp/app/services/apollo-subscription.ts index 1caab938e..20b2d64e5 100644 --- a/webapp/app/services/apollo-subscription.ts +++ b/webapp/app/services/apollo-subscription.ts @@ -34,7 +34,7 @@ export class Subscription { currentResult() { const queryObservable = this.queryObservable; - const result = queryObservable.currentResult(); + const result = queryObservable.getCurrentResult(); const mappedResult = this.mapResult(result, this.props); return mappedResult; @@ -51,7 +51,7 @@ export class Subscription { return this.apollo.client.watchQuery({ query, - ...options, + ...options }); } @@ -79,7 +79,7 @@ export class Subscription { refetch: this.queryObservable.refetch, fetchMore: this.queryObservable.fetchMore, startPolling: this.queryObservable.startPolling, - stopPolling: this.queryObservable.stopPolling, + stopPolling: this.queryObservable.stopPolling }; } else { return result; diff --git a/webapp/app/services/apollo.ts b/webapp/app/services/apollo.ts index 3b9cfb8c5..f143c9b67 100644 --- a/webapp/app/services/apollo.ts +++ b/webapp/app/services/apollo.ts @@ -1,42 +1,21 @@ import Service, {inject as service} from '@ember/service'; import RouterService from '@ember/routing/router-service'; -import {ApolloClient} from 'apollo-client'; -import {BatchHttpLink} from 'apollo-link-batch-http'; -import { - IntrospectionFragmentMatcher, - InMemoryCache, - from, - ApolloLink, -} from 'apollo-boost'; + +import {InMemoryCache} from '@apollo/client/cache'; +import {ApolloClient} from '@apollo/client/core'; +import {ApolloLink} from '@apollo/client/link/core'; +import {BatchHttpLink} from '@apollo/client/link/batch-http'; import Session from 'accent-webapp/services/session'; const dataIdFromObject = (result: {id?: string; __typename: string}) => { if (result.id && result.__typename) return `${result.__typename}${result.id}`; - return null; + return false; }; -const fragmentMatcher = new IntrospectionFragmentMatcher({ - introspectionQueryResultData: { - __schema: { - types: [ - { - kind: 'INTERFACE', - name: 'ProjectIntegration', - possibleTypes: [ - {name: 'ProjectIntegrationAzureStorageContainer'}, - {name: 'ProjectIntegrationDiscord'}, - {name: 'ProjectIntegrationSlack'}, - ], - }, - ], - }, - }, -}); - const uri = '/graphql'; -const cache = new InMemoryCache({dataIdFromObject, fragmentMatcher}); +const cache = new InMemoryCache({dataIdFromObject}); const link = new BatchHttpLink({uri, batchInterval: 1, batchMax: 1}); const absintheBatchLink = new ApolloLink((operation, forward) => { @@ -51,8 +30,8 @@ const authLink = (getSession: any) => { operation.setContext(({headers = {}}: any) => ({ headers: { ...headers, - authorization: `Bearer ${token}`, - }, + authorization: `Bearer ${token}` + } })); } @@ -68,8 +47,12 @@ export default class Apollo extends Service { session: Session; client = new ApolloClient({ - link: from([authLink(() => this.session), absintheBatchLink, link]), - cache, + link: ApolloLink.from([ + authLink(() => this.session), + absintheBatchLink, + link + ]), + cache }); } diff --git a/webapp/app/services/authenticated-request.ts b/webapp/app/services/authenticated-request.ts index 90c97d0d6..9a964a8af 100644 --- a/webapp/app/services/authenticated-request.ts +++ b/webapp/app/services/authenticated-request.ts @@ -38,7 +38,7 @@ export default class AuthenticatedRequest extends Service { fetchOptions.method = 'POST'; fetchOptions.headers = { - Authorization: `Bearer ${this.session.credentials.token}`, + Authorization: `Bearer ${this.session.credentials.token}` }; const response = await fetch(url, fetchOptions); @@ -70,7 +70,7 @@ export default class AuthenticatedRequest extends Service { const options: RequestInit = {}; options.headers = { - Authorization: `Bearer ${this.session.credentials.token}`, + Authorization: `Bearer ${this.session.credentials.token}` }; const response = await fetch(url, options); @@ -89,7 +89,7 @@ export default class AuthenticatedRequest extends Service { fetchOptions.method = 'POST'; fetchOptions.headers = { - Authorization: `Bearer ${this.session.credentials.token}`, + Authorization: `Bearer ${this.session.credentials.token}` }; fetchOptions.body = this.setupFormFile(options); diff --git a/webapp/app/services/exporter.ts b/webapp/app/services/exporter.ts index 0b8c27d20..1705cbb4c 100644 --- a/webapp/app/services/exporter.ts +++ b/webapp/app/services/exporter.ts @@ -57,7 +57,7 @@ export default class Exporter extends Service { document_format: (documentFormat || document.format).toLowerCase(), 'filters[is_text_empty]': options.filters?.isTextEmptyFilter, 'filters[is_added_last_sync]': options.filters?.isAddedLastSyncFilter, - 'filters[is_conflicted]': options.filters?.isConflictedFilter, + 'filters[is_conflicted]': options.filters?.isConflictedFilter })}` ); /* eslint-enable camelcase */ @@ -79,7 +79,7 @@ export default class Exporter extends Service { document_format: documentFormat.toLowerCase(), 'filters[is_text_empty]': options.filters?.isTextEmptyFilter, 'filters[is_added_last_sync]': options.filters?.isAddedLastSyncFilter, - 'filters[is_conflicted]': options.filters?.isConflictedFilter, + 'filters[is_conflicted]': options.filters?.isConflictedFilter })}` ); /* eslint-enable camelcase */ @@ -96,7 +96,7 @@ export default class Exporter extends Service { project_id: project.id, version, document_path: document.path, - document_format: documentFormat, + document_format: documentFormat })}` ); /* eslint-enable camelcase */ diff --git a/webapp/app/services/jipt.ts b/webapp/app/services/jipt.ts index 4a62473be..743e5438b 100644 --- a/webapp/app/services/jipt.ts +++ b/webapp/app/services/jipt.ts @@ -13,7 +13,7 @@ export default class JIPT extends Service { text: translation.correctedText, id: translation.id, key, - isConflicted: translation.isConflicted, + isConflicted: translation.isConflicted }; return memo; }, @@ -31,7 +31,7 @@ export default class JIPT extends Service { changeText(translationId: string, text: string) { const payload = { translationId, - text, + text }; window.parent.postMessage({jipt: true, action: 'changeText', payload}, '*'); @@ -40,7 +40,7 @@ export default class JIPT extends Service { updateTranslation(translationId: string, translation: any) { const payload = { translationId, - ...translation, + ...translation }; window.parent.postMessage( diff --git a/webapp/app/services/language-searcher.ts b/webapp/app/services/language-searcher.ts index 9846f1bb1..caa036d23 100644 --- a/webapp/app/services/language-searcher.ts +++ b/webapp/app/services/language-searcher.ts @@ -11,15 +11,15 @@ export default class LanguageSearcher extends Service { async search({term}: {term: string}) { const searchQuery = { query: searchLanguagesQuery, - variables: {query: term}, + variables: {query: term} }; if (term.length < MINIMUM_TERM_LENGTH) return []; const { data: { - languages: {entries}, - }, + languages: {entries} + } } = await this.apollo.client.query(searchQuery); return entries; diff --git a/webapp/app/services/machine-translations.ts b/webapp/app/services/machine-translations.ts index 61b5008fe..ea076d55e 100644 --- a/webapp/app/services/machine-translations.ts +++ b/webapp/app/services/machine-translations.ts @@ -28,7 +28,7 @@ export default class MachineTranslations extends Service { documentId, toLanguage, fromLanguage, - documentFormat, + documentFormat }: TranslateDocumentOptions) { const url = fmt( config.API.MACHINE_TRANSLATIONS_TRANSLATE_DOCUMENT_PROJECT_PATH, @@ -47,7 +47,7 @@ export default class MachineTranslations extends Service { file, toLanguage, fromLanguage, - documentFormat, + documentFormat }: TranslateFileOptions) { const url = fmt( config.API.MACHINE_TRANSLATIONS_TRANSLATE_FILE_PROJECT_PATH, @@ -58,7 +58,7 @@ export default class MachineTranslations extends Service { ); return this.authenticatedRequest.machineTranslationsTranslateFile(url, { - file, + file }); } } diff --git a/webapp/app/services/merger.ts b/webapp/app/services/merger.ts index 3755a1794..5299ce436 100644 --- a/webapp/app/services/merger.ts +++ b/webapp/app/services/merger.ts @@ -24,7 +24,7 @@ export default class Merger extends Service { documentPath, documentFormat, mergeType, - mergeOptions, + mergeOptions }: MergeOptions) { const language = revision.language; const url = fmt( @@ -40,7 +40,7 @@ export default class Merger extends Service { return this.authenticatedRequest.commit(url, { file, documentPath, - documentFormat, + documentFormat }); } } diff --git a/webapp/app/services/peeker.ts b/webapp/app/services/peeker.ts index a97862805..308504e65 100644 --- a/webapp/app/services/peeker.ts +++ b/webapp/app/services/peeker.ts @@ -46,7 +46,7 @@ export default class Peeker extends Service { file, documentPath, documentFormat, - syncType, + syncType }: SyncOptions) { const url = fmt( config.API.SYNC_PEEK_PROJECT_PATH, @@ -58,12 +58,12 @@ export default class Peeker extends Service { documentFormat = documentFormat.toLowerCase(); const { - data: {operations, stats}, + data: {operations, stats} } = await this.authenticatedRequest.peek(url, { file, documentPath, version, - documentFormat, + documentFormat }); return revisions.map((revision) => @@ -79,7 +79,7 @@ export default class Peeker extends Service { mergeType, documentPath, documentFormat, - mergeOptions, + mergeOptions }: MergeOptions) { const url = fmt( config.API.MERGE_PEEK_PROJECT_PATH, @@ -92,12 +92,12 @@ export default class Peeker extends Service { documentFormat = documentFormat.toLowerCase(); const { - data: {operations, stats}, + data: {operations, stats} } = await this.authenticatedRequest.peek(url, { file, version, documentPath, - documentFormat, + documentFormat }); return [this.mapOperations(revision, operations, stats)]; @@ -111,7 +111,7 @@ export default class Peeker extends Service { return { language: revision.language, stats: this.mapOperationStats(stats[revision.id]), - operations: this.mapOperationItems(operations[revision.id]), + operations: this.mapOperationItems(operations[revision.id]) }; } @@ -133,7 +133,7 @@ export default class Peeker extends Service { action: operation.action, key: operation.key, text: operation.text, - previousText: operation['previous-text'], + previousText: operation['previous-text'] }; }); } diff --git a/webapp/app/services/phoenix.ts b/webapp/app/services/phoenix.ts index d4004c326..2abee845c 100644 --- a/webapp/app/services/phoenix.ts +++ b/webapp/app/services/phoenix.ts @@ -45,7 +45,7 @@ export default class Phoenix extends Service { socket({token}: {token: string}) { const socket = new Socket('/socket', { - params: {token}, + params: {token} }); socket.connect(); @@ -81,7 +81,7 @@ export default class Phoenix extends Service { const events = { sync: this.handleSync, create_collaborator: this.handleCreateCollaborator, - create_comment: this.handleCreateComment, + create_comment: this.handleCreateComment }; /* eslint-enable camelcase */ @@ -100,7 +100,7 @@ export default class Phoenix extends Service { /* eslint camelcase:0 */ this.showFlashMessage('sync', { user: user.name, - documentPath: payload.document_path, + documentPath: payload.document_path }); /* eslint camelcase:1 */ } @@ -108,7 +108,7 @@ export default class Phoenix extends Service { private handleCreateCollaborator({payload, user}: WebsocketMessage) { this.showFlashMessage('create_collaborator', { user: user.name, - collaboratorEmail: payload?.collaborator?.email, + collaboratorEmail: payload?.collaborator?.email }); } @@ -116,7 +116,7 @@ export default class Phoenix extends Service { this.showFlashMessage('create_comment', { user: user.name, commentText: payload.text, - translationKey: payload?.translation?.key, + translationKey: payload?.translation?.key }); } diff --git a/webapp/app/services/raven.js b/webapp/app/services/raven.js index c9a7b427c..fc739a809 100644 --- a/webapp/app/services/raven.js +++ b/webapp/app/services/raven.js @@ -108,16 +108,16 @@ export default Service.extend({ if (typeOf(reason) === 'error') { this.captureException(reason, { extra: { - context: label || this.unhandledPromiseErrorMessage, - }, + context: label || this.unhandledPromiseErrorMessage + } }); this.didCaptureException(reason); } else { this.captureMessage(this._extractMessage(reason), { extra: { reason, - context: label, - }, + context: label + } }); } }); @@ -199,5 +199,5 @@ export default Service.extend({ return false; } } - }, + } }); diff --git a/webapp/app/services/session/creator.ts b/webapp/app/services/session/creator.ts index fd6ea2c55..b4a6be8c3 100644 --- a/webapp/app/services/session/creator.ts +++ b/webapp/app/services/session/creator.ts @@ -7,7 +7,7 @@ export default class SessionCreator extends Service { method: 'POST', headers: { 'Content-Type': 'application/json', - Authorization: `Bearer ${token}`, + Authorization: `Bearer ${token}` }, body: JSON.stringify({ query: ` @@ -21,8 +21,8 @@ export default class SessionCreator extends Service { } } } - `, - }), + ` + }) }; const response = await fetch('/graphql', options); diff --git a/webapp/app/services/syncer.ts b/webapp/app/services/syncer.ts index 22e192014..7677ce635 100644 --- a/webapp/app/services/syncer.ts +++ b/webapp/app/services/syncer.ts @@ -24,7 +24,7 @@ export default class Syncer extends Service { file, documentPath, documentFormat, - syncType, + syncType }: SyncOptions) { const url = fmt( config.API.SYNC_PROJECT_PATH, @@ -39,7 +39,7 @@ export default class Syncer extends Service { documentPath, documentFormat, version, - syncType, + syncType }); } } diff --git a/webapp/app/styles/app.scss b/webapp/app/styles/app.scss index 5fa8d38a5..574c62ded 100644 --- a/webapp/app/styles/app.scss +++ b/webapp/app/styles/app.scss @@ -5,7 +5,6 @@ @import 'highlight-theme'; @import 'html-components/button'; -@import 'html-components/emoji-picker'; @import 'html-components/filters'; @import 'html-components/power-select'; @import 'html-components/pell'; @@ -14,7 +13,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 100; - src: url('fonts/Inter-Thin.woff2') format('woff2'), + src: + url('fonts/Inter-Thin.woff2') format('woff2'), url('fonts/Inter-Thin.woff') format('woff'); } @@ -22,7 +22,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 100; - src: url('fonts/Inter-ThinItalic.woff2') format('woff2'), + src: + url('fonts/Inter-ThinItalic.woff2') format('woff2'), url('fonts/Inter-ThinItalic.woff') format('woff'); } @@ -30,7 +31,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 200; - src: url('fonts/Inter-ExtraLight.woff2') format('woff2'), + src: + url('fonts/Inter-ExtraLight.woff2') format('woff2'), url('fonts/Inter-ExtraLight.woff') format('woff'); } @@ -38,7 +40,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 200; - src: url('fonts/Inter-ExtraLightItalic.woff2') format('woff2'), + src: + url('fonts/Inter-ExtraLightItalic.woff2') format('woff2'), url('fonts/Inter-ExtraLightItalic.woff') format('woff'); } @@ -46,7 +49,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 300; - src: url('fonts/Inter-Light.woff2') format('woff2'), + src: + url('fonts/Inter-Light.woff2') format('woff2'), url('fonts/Inter-Light.woff') format('woff'); } @@ -54,7 +58,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 300; - src: url('fonts/Inter-LightItalic.woff2') format('woff2'), + src: + url('fonts/Inter-LightItalic.woff2') format('woff2'), url('fonts/Inter-LightItalic.woff') format('woff'); } @@ -62,7 +67,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 400; - src: url('fonts/Inter-Regular.woff2') format('woff2'), + src: + url('fonts/Inter-Regular.woff2') format('woff2'), url('fonts/Inter-Regular.woff') format('woff'); } @@ -70,7 +76,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 400; - src: url('fonts/Inter-Italic.woff2') format('woff2'), + src: + url('fonts/Inter-Italic.woff2') format('woff2'), url('fonts/Inter-Italic.woff') format('woff'); } @@ -78,7 +85,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 500; - src: url('fonts/Inter-Medium.woff2') format('woff2'), + src: + url('fonts/Inter-Medium.woff2') format('woff2'), url('fonts/Inter-Medium.woff') format('woff'); } @@ -86,7 +94,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 500; - src: url('fonts/Inter-MediumItalic.woff2') format('woff2'), + src: + url('fonts/Inter-MediumItalic.woff2') format('woff2'), url('fonts/Inter-MediumItalic.woff') format('woff'); } @@ -94,14 +103,16 @@ font-family: 'Inter'; font-style: normal; font-weight: 600; - src: url('fonts/Inter-SemiBold.woff2') format('woff2'), + src: + url('fonts/Inter-SemiBold.woff2') format('woff2'), url('fonts/Inter-SemiBold.woff') format('woff'); } @font-face { font-family: 'Inter'; font-style: italic; font-weight: 600; - src: url('fonts/Inter-SemiBoldItalic.woff2') format('woff2'), + src: + url('fonts/Inter-SemiBoldItalic.woff2') format('woff2'), url('fonts/Inter-SemiBoldItalic.woff') format('woff'); } @@ -109,7 +120,8 @@ font-family: 'Inter'; font-style: normal; font-weight: 700; - src: url('fonts/Inter-Bold.woff2') format('woff2'), + src: + url('fonts/Inter-Bold.woff2') format('woff2'), url('fonts/Inter-Bold.woff') format('woff'); } @@ -117,7 +129,8 @@ font-family: 'Inter'; font-style: italic; font-weight: 700; - src: url('fonts/Inter-BoldItalic.woff2') format('woff2'), + src: + url('fonts/Inter-BoldItalic.woff2') format('woff2'), url('fonts/Inter-BoldItalic.woff') format('woff'); } @@ -125,7 +138,8 @@ font-family: 'Fira Code'; font-weight: 300; font-style: normal; - src: url('fonts/FiraCode-Light.woff2') format('woff2'), + src: + url('fonts/FiraCode-Light.woff2') format('woff2'), url('fonts/FiraCode-Light.woff') format('woff'); } @@ -133,7 +147,8 @@ font-family: 'Fira Code'; font-weight: 400; font-style: normal; - src: url('fonts/FiraCode-Regular.woff2') format('woff2'), + src: + url('fonts/FiraCode-Regular.woff2') format('woff2'), url('fonts/FiraCode-Regular.woff') format('woff'); } @@ -141,7 +156,8 @@ font-family: 'Fira Code'; font-weight: 500; font-style: normal; - src: url('fonts/FiraCode-Medium.woff2') format('woff2'), + src: + url('fonts/FiraCode-Medium.woff2') format('woff2'), url('fonts/FiraCode-Medium.woff') format('woff'); } @@ -149,7 +165,8 @@ font-family: 'Fira Code'; font-weight: 700; font-style: normal; - src: url('fonts/FiraCode-Bold.woff2') format('woff2'), + src: + url('fonts/FiraCode-Bold.woff2') format('woff2'), url('fonts/FiraCode-Bold.woff') format('woff'); } @@ -345,6 +362,21 @@ input[type='radio'] { border-radius: 18px; } +emoji-picker { + --num-columns: 9; + --emoji-size: 1.3rem; + --background: var(--background-light); + --border-color: var(--background-light-highlight); + --input-border-color: transparent; + --input-padding: 4px 0; + --outline-size: 0; + --input-font-color: var(--text-color-normal); + --input-placeholder-color: var(--color-grey); + --indicator-color: hsl(var(--accent-hue), 55%, 55%); + --button-hover-background: var(--background-light-highlight); + --button-active-background: var(--background-light-highlight); +} + :global(.app) { display: flex; min-height: 100vh; diff --git a/webapp/app/pods/components/acc-avatar-img/styles.scss b/webapp/app/styles/components/acc-avatar-img.scss similarity index 100% rename from webapp/app/pods/components/acc-avatar-img/styles.scss rename to webapp/app/styles/components/acc-avatar-img.scss diff --git a/webapp/app/pods/components/acc-badge/styles.scss b/webapp/app/styles/components/acc-badge.scss similarity index 100% rename from webapp/app/pods/components/acc-badge/styles.scss rename to webapp/app/styles/components/acc-badge.scss diff --git a/webapp/app/styles/components/acc-emoji-picker.scss b/webapp/app/styles/components/acc-emoji-picker.scss new file mode 100644 index 000000000..2ae85d1d0 --- /dev/null +++ b/webapp/app/styles/components/acc-emoji-picker.scss @@ -0,0 +1,71 @@ +.wrapper { + display: flex; + align-items: flex-start; + justify-content: center; + position: fixed; + z-index: 4000; + height: 100vh; + left: 0; + right: 0; + top: 0; + overflow-y: scroll; + padding: 130px 30px 100px; +} + +.overlay { + position: fixed; + top: 0; + left: 0; + bottom: 0; + right: 0; + min-height: calc(100vh - 130px); + background-color: #000; + opacity: 0.6; + + animation-name: animate-overlay; + animation-timing-function: ease-in-out; + animation-duration: 0.2s; + animation-delay: 0; + animation-fill-mode: forwards; +} + +.container { + display: block; + position: relative; + z-index: 3; + border-radius: var(--border-radius); + overflow: hidden; + box-shadow: 0 2px 20px 0 var(--shadow-color); + + animation-name: animate-content; + animation-timing-function: ease-in-out; + animation-duration: 0.3s; + animation-delay: 0; + animation-fill-mode: forwards; +} + +@media (max-width: 440px) { + .wrapper { + padding: 30px 10px; + } +} + +@keyframes animate-content { + 0% { + opacity: 0; + transform: translateY(10px); + } + 100% { + opacity: 1; + transform: translateY(0); + } +} + +@keyframes animate-overlay { + 0% { + opacity: 0; + } + 100% { + opacity: 0.6; + } +} diff --git a/webapp/app/pods/components/acc-flash-message/styles.scss b/webapp/app/styles/components/acc-flash-message.scss similarity index 100% rename from webapp/app/pods/components/acc-flash-message/styles.scss rename to webapp/app/styles/components/acc-flash-message.scss diff --git a/webapp/app/pods/components/acc-modal/styles.scss b/webapp/app/styles/components/acc-modal.scss similarity index 100% rename from webapp/app/pods/components/acc-modal/styles.scss rename to webapp/app/styles/components/acc-modal.scss diff --git a/webapp/app/pods/components/acc-select/styles.scss b/webapp/app/styles/components/acc-select.scss similarity index 100% rename from webapp/app/pods/components/acc-select/styles.scss rename to webapp/app/styles/components/acc-select.scss diff --git a/webapp/app/pods/components/acc-wrapper/styles.scss b/webapp/app/styles/components/acc-wrapper.scss similarity index 100% rename from webapp/app/pods/components/acc-wrapper/styles.scss rename to webapp/app/styles/components/acc-wrapper.scss diff --git a/webapp/app/pods/components/acc-wrapper/content/styles.scss b/webapp/app/styles/components/acc-wrapper/content.scss similarity index 100% rename from webapp/app/pods/components/acc-wrapper/content/styles.scss rename to webapp/app/styles/components/acc-wrapper/content.scss diff --git a/webapp/app/pods/components/acc-wrapper/projects/styles.scss b/webapp/app/styles/components/acc-wrapper/projects.scss similarity index 100% rename from webapp/app/pods/components/acc-wrapper/projects/styles.scss rename to webapp/app/styles/components/acc-wrapper/projects.scss diff --git a/webapp/app/pods/components/acc-wrapper/sidebar/styles.scss b/webapp/app/styles/components/acc-wrapper/sidebar.scss similarity index 100% rename from webapp/app/pods/components/acc-wrapper/sidebar/styles.scss rename to webapp/app/styles/components/acc-wrapper/sidebar.scss diff --git a/webapp/app/pods/components/activity-item/styles.scss b/webapp/app/styles/components/activity-item.scss similarity index 100% rename from webapp/app/pods/components/activity-item/styles.scss rename to webapp/app/styles/components/activity-item.scss diff --git a/webapp/app/pods/components/application-footer/styles.scss b/webapp/app/styles/components/application-footer.scss similarity index 96% rename from webapp/app/pods/components/application-footer/styles.scss rename to webapp/app/styles/components/application-footer.scss index 242fa2929..a0cf6a16f 100644 --- a/webapp/app/pods/components/application-footer/styles.scss +++ b/webapp/app/styles/components/application-footer.scss @@ -73,7 +73,9 @@ background: none; opacity: 0.2; color: var(--text-color-normal); - transition: color 0.2s ease-in-out, opacity 0.2s ease-in-out, + transition: + color 0.2s ease-in-out, + opacity 0.2s ease-in-out, transform 0.2s ease-in; &:focus, diff --git a/webapp/app/pods/components/async-button/styles.scss b/webapp/app/styles/components/async-button.scss similarity index 90% rename from webapp/app/pods/components/async-button/styles.scss rename to webapp/app/styles/components/async-button.scss index 8941d3236..2b4e2cec5 100644 --- a/webapp/app/pods/components/async-button/styles.scss +++ b/webapp/app/styles/components/async-button.scss @@ -26,6 +26,13 @@ } } + &:global(.button--iconOnly) { + .label { + padding-left: 5px; + padding-right: 5px; + } + } + &:global(.button--small) { &.button--loading { .loading { diff --git a/webapp/app/pods/components/azure-push-form/styles.scss b/webapp/app/styles/components/azure-push-form.scss similarity index 100% rename from webapp/app/pods/components/azure-push-form/styles.scss rename to webapp/app/styles/components/azure-push-form.scss diff --git a/webapp/app/pods/components/commit-file/styles.scss b/webapp/app/styles/components/commit-file.scss similarity index 100% rename from webapp/app/pods/components/commit-file/styles.scss rename to webapp/app/styles/components/commit-file.scss diff --git a/webapp/app/styles/components/conflicts-filters.scss b/webapp/app/styles/components/conflicts-filters.scss new file mode 100644 index 000000000..a04e7f1f5 --- /dev/null +++ b/webapp/app/styles/components/conflicts-filters.scss @@ -0,0 +1,76 @@ +.conflicts-filters { + :global(.filters) { + margin-bottom: 20px; + + :global(.ember-power-select-multiple-trigger) { + width: 100%; + border-width: 2px; + margin-bottom: 10px; + &:focus { + border-width: 2px; + } + } + } +} + +.search-icon { + position: absolute; + top: 50%; + margin-top: -10px; + left: 7px; + width: 20px; + height: 20px; + stroke: #b7b7b7; +} + +.input { + @extend %textInput; + width: 100%; + padding: 7px 7px 7px 30px; + font-family: var(--font-primary); + font-size: 14px; + color: var(--color-black); + + &:focus { + box-shadow: + inset 0 1px 2px rgba(#000, 0.1), + 0 1px 2px var(--shadow-color); + } + + &::placeholder { + color: var(--color-grey); + } +} + +button.advancedFilters { + position: relative; + box-shadow: none; + flex-shrink: 0; + &:focus, + &:hover { + transform: translate3d(0, 0, 0); + } +} + +.advancedFilters-badge { + position: absolute; + top: -4px; + right: -7px; + background: var(--color-primary); + border-radius: var(--border-radius); + padding: 0 4px 1px; + color: #fff; + font-size: 10px; +} + +@media (max-width: 440px) { + .filters-wrapper { + flex-direction: column; + } + + .queryForm, + .filters-content { + width: 100%; + margin-right: 0; + } +} diff --git a/webapp/app/pods/components/conflicts-list/styles.scss b/webapp/app/styles/components/conflicts-list.scss similarity index 56% rename from webapp/app/pods/components/conflicts-list/styles.scss rename to webapp/app/styles/components/conflicts-list.scss index 65422a437..23d758fa3 100644 --- a/webapp/app/pods/components/conflicts-list/styles.scss +++ b/webapp/app/styles/components/conflicts-list.scss @@ -1,9 +1,17 @@ -.conflicts-items { - position: relative; - margin-top: 20px; +.conflicts-header { + position: sticky; + padding: 14px 0 0 15px; + top: 0; + z-index: 4; + display: grid; + grid-template-columns: repeat(var(--group-columns-count, 1), 1fr); + font-size: 16px; + font-weight: bold; + margin: 0 0 10px -14px; + background: var(--content-background); } -.conflicts-items-version { +.conflicts-list-version { margin: 0 0 15px; padding: 15px; border-radius: var(--border-radius); @@ -16,14 +24,25 @@ color: var(--color-blue); } -.conflicts-items-version-tag { +.conflicts-list-version-tag { display: inline-flex; align-items: center; margin-left: 2px; font-size: 11px; font-family: var(--font-monospace); font-weight: 600; - text-transform: capitalize; + text-transform: none; +} + +.conflicts-header-item { + padding-bottom: 10px; +} + +.conflicts-header-item-slug { + font-weight: normal; + opacity: 0.6; + margin-left: 2px; + font-size: 12px; } .all-reviewed { @@ -43,7 +62,3 @@ font-size: 12px; color: var(--color-grey); } - -.empty-content { - margin-top: 40px; -} diff --git a/webapp/app/styles/components/conflicts-list/advanced-filters.scss b/webapp/app/styles/components/conflicts-list/advanced-filters.scss new file mode 100644 index 000000000..9e08e73f7 --- /dev/null +++ b/webapp/app/styles/components/conflicts-list/advanced-filters.scss @@ -0,0 +1,31 @@ +.conflicts-list-advanced-filters { + margin: 15px 0 0 0; + display: flex; + flex-direction: column; +} + +.title { + text-transform: uppercase; + color: var(--text-color-normal); + font-size: 11px; + font-weight: bold; +} + +.label { + display: flex; + align-items: flex-start; + width: 100%; + max-width: 250px; + margin-top: 10px; + font-size: 12px; + + input { + margin-right: 6px; + } +} + +.labels { + display: flex; + flex-wrap: wrap; + max-width: 960px; +} diff --git a/webapp/app/styles/components/conflicts-list/group.scss b/webapp/app/styles/components/conflicts-list/group.scss new file mode 100644 index 000000000..998b642e4 --- /dev/null +++ b/webapp/app/styles/components/conflicts-list/group.scss @@ -0,0 +1,111 @@ +.item { + --grid-item-reviewed-opacity: 0.5; + --grid-item-actions-opacity: 0; + border-bottom: 1px solid var(--background-light-highlight); + transition: 0.2s ease-in-out; + transition-property: border-color; + background: var(--content-background); + + :global(.lint-translations-item) { + display: none; + padding: 2px 6px; + border-radius: var(--border-radius); + margin: 0 5px 5px; + + background: var(--background-light); + } + + &.item--focus { + background: var(--background-light); + + :global(.lint-translations-item) { + display: block; + } + + .item-key { + transform: translateY(2px); + } + + --grid-item-reviewed-opacity: 1; + --grid-item-actions-opacity: 1; + } +} + +.item-grid { + display: grid; + grid-template-columns: repeat(var(--group-columns-count, 1), 1fr); +} + +.item-grid-item { + --border-color: var(--background-light-highlight); + position: relative; + padding: 0; + border-left: 2px solid var(--border-color); + opacity: 1; + + &.item-grid-item--translated { + --border-color: var(--color-warning); + opacity: 1; + } + + &.item-grid-item--reviewed { + --border-color: var(--color-green); + opacity: var(--grid-item-reviewed-opacity); + } + + &:hover { + opacity: 1; + } + + &::before { + position: absolute; + content: ''; + top: -24px; + left: -2px; + height: 32px; + width: 2px; + background: var(--border-color); + } +} + +.item-key-prefix { + display: inline-flex; + font-size: 11px; + color: #959595; + gap: 6px; + font-weight: 300; + flex-shrink: 0; + + &::before { + content: '/'; + } +} + +.item-key { + position: sticky; + top: 46px; + display: inline-flex; + align-items: flex-start; + gap: 5px; + background: var(--content-background); + padding: 2px 6px; + text-decoration: none; + font-family: var(--font-monospace); + box-shadow: 0 2px 4px var(--shadow-color); + border-radius: var(--border-radius); + font-weight: 600; + z-index: 2; + word-break: break-all; + transition: 0.2s ease-in-out; + transition-property: color; + margin-left: 4px; + line-height: 1.5; + font-size: 11px; + color: var(--text-color-normal); + transition: 0.2s ease-in-out; + transition-property: transform; +} + +.key { + text-decoration: none; +} diff --git a/webapp/app/styles/components/conflicts-list/item.scss b/webapp/app/styles/components/conflicts-list/item.scss new file mode 100644 index 000000000..edaca841d --- /dev/null +++ b/webapp/app/styles/components/conflicts-list/item.scss @@ -0,0 +1,142 @@ +.translation-item { + &:hover { + .form-helpers { + pointer-events: all; + opacity: 1; + } + .button-submit { + pointer-events: all; + opacity: 1; + } + } +} + +.revert-button { + position: absolute; + right: 8px; + top: -30px; + + :global(.label) { + padding-left: 3px; + padding-right: 3px; + } +} + +.item-details__column { + position: relative; +} + +.item-details { + display: flex; + flex-direction: column; + + &[data-dir='rtl'] { + .revert-button { + right: auto; + left: 8px; + } + + .item-details__column { + align-items: flex-end; + } + + .item-details__column:first-of-type { + margin-right: 0; + margin-left: 15px; + } + + .item-key { + margin-right: 0; + margin-left: 15px; + flex-direction: row-reverse; + } + + .item-key-prefix::before { + content: ''; + } + + .item-key-prefix::after { + content: '/'; + } + } +} + +.item-details__column { + display: flex; + flex-direction: column; + align-items: flex-start; +} + +.item-details__column:first-of-type { + margin-right: 15px; +} + +.translation-item.resolved { + background: var(--color-primary-opacity-10); +} + +.translation-item.errored { + .textInput { + border-color: var(--color-error); + } +} + +.error { + font-size: 12px; + font-weight: bold; + color: var(--color-error); +} + +.button-submit { + display: flex; + justify-content: flex-end; + position: absolute; + pointer-events: none; + opacity: var(--grid-item-actions-opacity); + gap: 0; + bottom: 15px; + right: 10px; + z-index: 3; + transition: 0.2s ease-in-out; + transition-property: opacity; + + &[data-dir='rtl'] { + right: auto; + left: 7px; + flex-direction: row-reverse; + } +} + +.textInput { + flex-grow: 1; + flex-shrink: 0; + width: 100%; + font-size: 13px; +} + +.item-text { + display: block; + width: 100%; + color: var(--color-black); + line-height: 1.4; + padding: 3px 10px 10px 0; + font-size: 13px; + line-height: 1.6; + cursor: pointer; + word-break: break-word; + + &:focus, + &:hover { + outline: none; + opacity: 0.8; + } +} + +.form-helpers { + pointer-events: none; + opacity: 0; + position: relative; + z-index: 1; + transition: 0.2s ease-in-out; + transition-property: opacity; +} diff --git a/webapp/app/pods/components/dashboard-revisions/styles.scss b/webapp/app/styles/components/dashboard-revisions.scss similarity index 93% rename from webapp/app/pods/components/dashboard-revisions/styles.scss rename to webapp/app/styles/components/dashboard-revisions.scss index 1faec244e..b7f603634 100644 --- a/webapp/app/pods/components/dashboard-revisions/styles.scss +++ b/webapp/app/styles/components/dashboard-revisions.scss @@ -37,7 +37,7 @@ align-items: center; justify-content: center; width: 100%; - margin: 20px auto 40px; + margin: 0 auto 10px; } .numberStat-reviewPercentage { @@ -92,17 +92,15 @@ .stats-title { display: flex; - justify-content: space-between; + justify-content: center; align-items: center; - margin-bottom: 0; + margin-bottom: 10px; padding-bottom: 2px; - margin-left: -8px; } .slaves { display: flex; - flex-wrap: wrap; - align-items: flex-start; + flex-direction: column; } .empty-slaves-button { @@ -111,7 +109,9 @@ max-width: 340px; padding: 15px; background: var(--content-background); - box-shadow: 0 1px 4px var(--shadow-color), 0 7px 12px var(--shadow-color); + box-shadow: + 0 1px 4px var(--shadow-color), + 0 7px 12px var(--shadow-color); border-radius: var(--border-radius); color: var(--color-primary); text-decoration: none; @@ -146,12 +146,13 @@ } .master { - margin: 0 0 10px; + margin: 25px 0 0; } .stats-title-links { display: flex; justify-content: flex-end; + gap: 5px; .button { margin-left: 10px; diff --git a/webapp/app/pods/components/dashboard-revisions/item/styles.scss b/webapp/app/styles/components/dashboard-revisions/item.scss similarity index 72% rename from webapp/app/pods/components/dashboard-revisions/item/styles.scss rename to webapp/app/styles/components/dashboard-revisions/item.scss index 9a320b1bc..bd89aa507 100644 --- a/webapp/app/pods/components/dashboard-revisions/item/styles.scss +++ b/webapp/app/styles/components/dashboard-revisions/item.scss @@ -2,18 +2,13 @@ transition: 0.2s ease-in-out; transition-property: box-shadow; position: relative; - flex: 1 1 calc(50% - 20px); justify-content: space-between; - max-width: 50%; - margin-bottom: 20px; - margin-right: 20px; + width: 100%; + margin-bottom: 10px; border-radius: var(--border-radius); - border: 1px solid var(--background-light-highlight); - - &:nth-child(even) { - flex: 1 1 50%; - margin-right: 0; - } + box-shadow: + 0 6px 25px var(--shadow-color), + 0 2px 7px var(--shadow-color); } .dashboard-revisions-item.low-percentage { @@ -62,8 +57,7 @@ .language { display: flex; - align-items: flex-end; - justify-content: space-between; + flex-direction: column; color: var(--color-grey); } @@ -82,21 +76,43 @@ } .language-reviewedPercentage { - display: block; - margin-top: 5px; + display: flex; + align-items: baseline; + gap: 4px; font-weight: normal; - font-size: 25px; + font-size: 22px; color: var(--color-grey); + letter-spacing: -2px; +} + +.language-reviewedPercentage-symbol { + font-size: 13px; } .reviewedStats { - padding: 2px 0 1px 7px; - border-radius: var(--border-radius); + display: flex; + align-items: baseline; + justify-content: space-between; + gap: 10px; color: var(--color-grey); font-size: 12px; font-family: var(--font-monospace); } +.reviewedStats-translatedCount, +.reviewedStats-translationsCount, +.reviewedStats-reviewedCount { + font-family: var(--font-primary); + color: var(--color-grey); + margin: 0 3px; + font-size: 11px; + text-decoration: none; +} + +.reviewedStats-reviewedCount { + font-weight: bold; +} + .actionsButton { transition: 0.2s ease-in-out; transition-property: opacity; @@ -114,17 +130,18 @@ &:focus, &:hover { .actionsButton-icon { - stroke: var(--color-primary); + opacity: 1; } } } .actionsButton-icon { stroke: var(--color-grey); + opacity: 0.4; transition: 0.2s ease-in-out; - transition-property: stroke; - width: 15px; - height: 15px; + transition-property: stroke, opacity; + width: 13px; + height: 13px; } .actions { @@ -132,6 +149,10 @@ display: flex; flex-direction: column; gap: 4px; + + :global(.button) { + justify-content: center; + } } .actionItem-text { diff --git a/webapp/app/pods/components/documents-list/styles.scss b/webapp/app/styles/components/documents-list.scss similarity index 100% rename from webapp/app/pods/components/documents-list/styles.scss rename to webapp/app/styles/components/documents-list.scss diff --git a/webapp/app/pods/components/documents-list/item/styles.scss b/webapp/app/styles/components/documents-list/item.scss similarity index 100% rename from webapp/app/pods/components/documents-list/item/styles.scss rename to webapp/app/styles/components/documents-list/item.scss diff --git a/webapp/app/pods/components/documents-machine-translations-button/styles.scss b/webapp/app/styles/components/documents-machine-translations-button.scss similarity index 100% rename from webapp/app/pods/components/documents-machine-translations-button/styles.scss rename to webapp/app/styles/components/documents-machine-translations-button.scss diff --git a/webapp/app/pods/components/dummy-login-form/styles.scss b/webapp/app/styles/components/dummy-login-form.scss similarity index 100% rename from webapp/app/pods/components/dummy-login-form/styles.scss rename to webapp/app/styles/components/dummy-login-form.scss diff --git a/webapp/app/pods/components/empty-content/styles.scss b/webapp/app/styles/components/empty-content.scss similarity index 100% rename from webapp/app/pods/components/empty-content/styles.scss rename to webapp/app/styles/components/empty-content.scss diff --git a/webapp/app/pods/components/error-section/styles.scss b/webapp/app/styles/components/error-section.scss similarity index 100% rename from webapp/app/pods/components/error-section/styles.scss rename to webapp/app/styles/components/error-section.scss diff --git a/webapp/app/pods/components/flash-messages-list/styles.scss b/webapp/app/styles/components/flash-messages-list.scss similarity index 100% rename from webapp/app/pods/components/flash-messages-list/styles.scss rename to webapp/app/styles/components/flash-messages-list.scss diff --git a/webapp/app/pods/components/google-login-form/styles.scss b/webapp/app/styles/components/google-login-form.scss similarity index 100% rename from webapp/app/pods/components/google-login-form/styles.scss rename to webapp/app/styles/components/google-login-form.scss diff --git a/webapp/app/pods/components/highlight-render/styles.scss b/webapp/app/styles/components/highlight-render.scss similarity index 100% rename from webapp/app/pods/components/highlight-render/styles.scss rename to webapp/app/styles/components/highlight-render.scss diff --git a/webapp/app/pods/components/improve-prompt/styles.scss b/webapp/app/styles/components/improve-prompt.scss similarity index 95% rename from webapp/app/pods/components/improve-prompt/styles.scss rename to webapp/app/styles/components/improve-prompt.scss index b6c176a51..9d3084c26 100644 --- a/webapp/app/pods/components/improve-prompt/styles.scss +++ b/webapp/app/styles/components/improve-prompt.scss @@ -64,13 +64,13 @@ button.button { opacity: 1; } .prompt-button-quick-access[data-rtl] { - left: 36px; + transform: translateX(36px); right: auto; } .prompt-button-quick-access { opacity: 1; - right: 36px; + transform: translateX(-36px); padding: 0 7px; top: -1px; pointer-events: all; @@ -87,6 +87,7 @@ button.button { background: var(--input-background); opacity: 0; pointer-events: none; + transform: translateX(0); right: 0; display: flex; gap: 4px; diff --git a/webapp/app/styles/components/inline-machine-translate.scss b/webapp/app/styles/components/inline-machine-translate.scss new file mode 100644 index 000000000..e69de29bb diff --git a/webapp/app/pods/components/jipt-back-to-translations/styles.scss b/webapp/app/styles/components/jipt-back-to-translations.scss similarity index 100% rename from webapp/app/pods/components/jipt-back-to-translations/styles.scss rename to webapp/app/styles/components/jipt-back-to-translations.scss diff --git a/webapp/app/styles/components/jipt-example.scss b/webapp/app/styles/components/jipt-example.scss new file mode 100644 index 000000000..36a93804e --- /dev/null +++ b/webapp/app/styles/components/jipt-example.scss @@ -0,0 +1,123 @@ +.browser-mockup { + position: relative; + border-top: 2em solid rgba(230, 230, 230, 0.5); + box-shadow: 0 0.1em 1em 0 rgba(0, 0, 0, 0.4); + position: relative; + border-radius: 8px; + margin: 70px auto; + flex: 1; + max-width: 800px; +} + +.browser-back-link { + position: absolute; + top: -24px; + right: 10px; + font-size: 11px; + color: #2a2d87; + text-shadow: 0 1px 1px rgba(#fff, 0.2); + opacity: 0.6; + text-decoration: none; +} + +.browser-mockup:before { + display: block; + position: absolute; + content: ''; + top: -1.25em; + left: 1em; + width: 0.5em; + height: 0.5em; + border-radius: 50%; + background-color: #f44; + box-shadow: + 0 0 0 2px #f44, + 1.5em 0 0 2px #9b3, + 3em 0 0 2px #fb5; +} + +.browser-content { + background: #fff; + padding: 30px; + border-radius: 0 0 8px 8px; +} + +.browser-content h1 { + font-weight: bold; + font-size: 25px; + padding-bottom: 8px; + border-bottom: 1px solid #eee; + margin-bottom: 20px; +} + +.browser-content h2 { + font-weight: bold; + font-size: 20px; + padding-bottom: 5px; + margin-top: 10px; + margin-bottom: 12px; + border-bottom: 1px solid #eee; +} + +.tag-inner-wrapper { + display: flex; + gap: 5px; + align-items: center; +} + +.tag-attributes-wrapper { + display: flex; + flex-direction: column; + align-items: flex-start; + gap: 5px; + margin-bottom: 25px; +} + +.tag { + display: inline-flex; + color: #888; + font-size: 11px; + font-family: var(--font-monospace); +} + +.browser-content pre { + background: #eee; + padding: 10px 14px; + opacity: 0.7; + border-radius: 4px; + font-size: 11px; + margin-bottom: 15px; + white-space: pre-line; +} + +.browser-content img { + width: 100%; + max-width: 200px; + display: block; + border-radius: 4px; +} + +.browser-content input[type='submit'] { + border-radius: 50px; + padding: 3px 10px; + border: 2px solid #555; + background: linear-gradient(to bottom, #e2e2e2, #bcbcbc); + box-shadow: inset 0 1px 0 rgba(#fff, 0.5); +} + +.browser-content input[type='text'] { + border-radius: 5px; + padding: 5px 10px; + border: 2px solid #555; + background: #fff; + max-width: 300px; + width: 100%; + box-shadow: inset 0 1px 1px rgba(#000, 0.1); +} + +.wrapper { + background: linear-gradient(to right, #4453ad, #2a2d87); + display: flex; + height: 100dvh; + align-items: flex-start; +} diff --git a/webapp/app/pods/components/jipt-header/styles.scss b/webapp/app/styles/components/jipt-header.scss similarity index 100% rename from webapp/app/pods/components/jipt-header/styles.scss rename to webapp/app/styles/components/jipt-header.scss diff --git a/webapp/app/pods/components/jipt-translation/styles.scss b/webapp/app/styles/components/jipt-translation.scss similarity index 100% rename from webapp/app/pods/components/jipt-translation/styles.scss rename to webapp/app/styles/components/jipt-translation.scss diff --git a/webapp/app/pods/components/jipt-translations-filtered-title/styles.scss b/webapp/app/styles/components/jipt-translations-filtered-title.scss similarity index 100% rename from webapp/app/pods/components/jipt-translations-filtered-title/styles.scss rename to webapp/app/styles/components/jipt-translations-filtered-title.scss diff --git a/webapp/app/pods/components/jipt-translations-list/styles.scss b/webapp/app/styles/components/jipt-translations-list.scss similarity index 82% rename from webapp/app/pods/components/jipt-translations-list/styles.scss rename to webapp/app/styles/components/jipt-translations-list.scss index f4f0c2946..5374b96d1 100644 --- a/webapp/app/pods/components/jipt-translations-list/styles.scss +++ b/webapp/app/styles/components/jipt-translations-list.scss @@ -1,8 +1,3 @@ -.jipt-translations-list { - margin: 20px 0; - border-top: 1px solid #eee; -} - .item { display: block; border-bottom: 1px solid #eee; diff --git a/webapp/app/pods/components/jipt-translations-list/item/styles.scss b/webapp/app/styles/components/jipt-translations-list/item.scss similarity index 58% rename from webapp/app/pods/components/jipt-translations-list/item/styles.scss rename to webapp/app/styles/components/jipt-translations-list/item.scss index 3038bd0d0..3a57e7c48 100644 --- a/webapp/app/pods/components/jipt-translations-list/item/styles.scss +++ b/webapp/app/styles/components/jipt-translations-list/item.scss @@ -1,5 +1,6 @@ .item-text { - display: block; + display: flex; + flex-direction: column; text-overflow: ellipsis; overflow-x: hidden; color: var(--text-color-normal); @@ -9,4 +10,10 @@ color: var(--color-grey); font-style: italic; } + + .item-key { + opacity: 0.5; + font-size: 11px; + font-family: var(--font-monospace); + } } diff --git a/webapp/app/pods/components/lint-options/styles.scss b/webapp/app/styles/components/lint-options.scss similarity index 85% rename from webapp/app/pods/components/lint-options/styles.scss rename to webapp/app/styles/components/lint-options.scss index 34ea70ed4..50970589b 100644 --- a/webapp/app/pods/components/lint-options/styles.scss +++ b/webapp/app/styles/components/lint-options.scss @@ -17,7 +17,9 @@ color: var(--color-black); &:focus { - box-shadow: inset 0 1px 2px rgba(#000, 0.1), 0 1px 2px var(--shadow-color); + box-shadow: + inset 0 1px 2px rgba(#000, 0.1), + 0 1px 2px var(--shadow-color); } &::placeholder { diff --git a/webapp/app/pods/components/lint-translations-page/styles.scss b/webapp/app/styles/components/lint-translations-page.scss similarity index 82% rename from webapp/app/pods/components/lint-translations-page/styles.scss rename to webapp/app/styles/components/lint-translations-page.scss index a7b86be5d..a79c0a651 100644 --- a/webapp/app/pods/components/lint-translations-page/styles.scss +++ b/webapp/app/styles/components/lint-translations-page.scss @@ -23,6 +23,12 @@ .item { margin-bottom: 20px; + transition: 0.2s ease-in-out; + transition-property: opacity; +} + +.item--fixing { + opacity: 0.5; } .icon-warning { @@ -46,10 +52,9 @@ .stats-item { display: inline-flex; gap: 4px; - padding: 2px 12px; + padding: 4px 12px; border-radius: 30px; - border: 1px solid var(--background-light-highlight); - background: var(--background-light); + background: var(--background-light-highlight); color: var(--text-color-normal); font-size: 12px; cursor: default; diff --git a/webapp/app/pods/components/lint-translations-page/item/styles.scss b/webapp/app/styles/components/lint-translations-page/item.scss similarity index 86% rename from webapp/app/pods/components/lint-translations-page/item/styles.scss rename to webapp/app/styles/components/lint-translations-page/item.scss index 63c1687b2..261e14bf6 100644 --- a/webapp/app/pods/components/lint-translations-page/item/styles.scss +++ b/webapp/app/styles/components/lint-translations-page/item.scss @@ -13,11 +13,26 @@ flex-direction: column; } +.messages-item { + display: flex; + gap: 4px; + align-items: center; +} + .description { color: var(--color-error); font-size: 12px; } +.button-fix { + width: max-content; + + &:focus, + &:hover { + --button-text-hover-color: var(--color-green); + } +} + .item-language { font-size: 11px; font-weight: bold; @@ -94,6 +109,15 @@ } } +.item-diff-text { + font-size: 13px; + color: var(--text-color-normal); + + > div { + display: inline; + } +} + .item-text { strong { color: var(--color-green); diff --git a/webapp/app/pods/components/loading-content/styles.scss b/webapp/app/styles/components/loading-content.scss similarity index 100% rename from webapp/app/pods/components/loading-content/styles.scss rename to webapp/app/styles/components/loading-content.scss diff --git a/webapp/app/pods/components/login-forms/styles.scss b/webapp/app/styles/components/login-forms.scss similarity index 95% rename from webapp/app/pods/components/login-forms/styles.scss rename to webapp/app/styles/components/login-forms.scss index dfe12e818..373f29688 100644 --- a/webapp/app/pods/components/login-forms/styles.scss +++ b/webapp/app/styles/components/login-forms.scss @@ -7,7 +7,9 @@ padding: 30px 30px 20px 30px; border-radius: 6px; background: var(--content-background); - box-shadow: 0 1px 4px var(--shadow-color), 0 7px 12px var(--shadow-color); + box-shadow: + 0 1px 4px var(--shadow-color), + 0 7px 12px var(--shadow-color); .text { max-width: 300px; @@ -218,6 +220,13 @@ a.loginButton { color: #03a5f0; background: lighten(#03a5f0, 48%); } + + &.loginButton--oidc { + border-color: #f7931e; + text-shadow: none; + color: #f7931e; + background: lighten(#f7931e, 48%); + } } .loginButton-logo { diff --git a/webapp/app/pods/components/machine-translations-document-translate/styles.scss b/webapp/app/styles/components/machine-translations-document-translate.scss similarity index 100% rename from webapp/app/pods/components/machine-translations-document-translate/styles.scss rename to webapp/app/styles/components/machine-translations-document-translate.scss diff --git a/webapp/app/pods/components/machine-translations-translate-upload-form/styles.scss b/webapp/app/styles/components/machine-translations-translate-upload-form.scss similarity index 100% rename from webapp/app/pods/components/machine-translations-translate-upload-form/styles.scss rename to webapp/app/styles/components/machine-translations-translate-upload-form.scss diff --git a/webapp/app/pods/components/operations-peek/styles.scss b/webapp/app/styles/components/operations-peek.scss similarity index 100% rename from webapp/app/pods/components/operations-peek/styles.scss rename to webapp/app/styles/components/operations-peek.scss diff --git a/webapp/app/pods/components/operations-peek/item/styles.scss b/webapp/app/styles/components/operations-peek/item.scss similarity index 93% rename from webapp/app/pods/components/operations-peek/item/styles.scss rename to webapp/app/styles/components/operations-peek/item.scss index 68cf9d5c9..b83efd6b9 100644 --- a/webapp/app/pods/components/operations-peek/item/styles.scss +++ b/webapp/app/styles/components/operations-peek/item.scss @@ -11,7 +11,9 @@ color: var(--color-black); &:focus { - box-shadow: inset 0 1px 2px rgba(#000, 0.1), 0 1px 2px var(--shadow-color); + box-shadow: + inset 0 1px 2px rgba(#000, 0.1), + 0 1px 2px var(--shadow-color); } &::placeholder { diff --git a/webapp/app/pods/components/page-title/styles.scss b/webapp/app/styles/components/page-title.scss similarity index 100% rename from webapp/app/pods/components/page-title/styles.scss rename to webapp/app/styles/components/page-title.scss diff --git a/webapp/app/pods/components/page-title/end/styles.scss b/webapp/app/styles/components/page-title/end.scss similarity index 100% rename from webapp/app/pods/components/page-title/end/styles.scss rename to webapp/app/styles/components/page-title/end.scss diff --git a/webapp/app/pods/components/project-activities-filter/styles.scss b/webapp/app/styles/components/project-activities-filter.scss similarity index 100% rename from webapp/app/pods/components/project-activities-filter/styles.scss rename to webapp/app/styles/components/project-activities-filter.scss diff --git a/webapp/app/pods/components/project-activities-list/styles.scss b/webapp/app/styles/components/project-activities-list.scss similarity index 100% rename from webapp/app/pods/components/project-activities-list/styles.scss rename to webapp/app/styles/components/project-activities-list.scss diff --git a/webapp/app/pods/components/project-activity/styles.scss b/webapp/app/styles/components/project-activity.scss similarity index 95% rename from webapp/app/pods/components/project-activity/styles.scss rename to webapp/app/styles/components/project-activity.scss index 5c255bd57..4e64c8082 100644 --- a/webapp/app/pods/components/project-activity/styles.scss +++ b/webapp/app/styles/components/project-activity.scss @@ -68,12 +68,6 @@ color: var(--color-primary); } -.details-associations-overflow { - max-height: 400px; - overflow-y: scroll; - margin-top: -12px; -} - .details-associations-pagination { border-top: 1px solid var(--content-background-border); } @@ -96,7 +90,9 @@ margin-right: 20px; padding: 13px; background: var(--content-background); - box-shadow: 0 1px 2px var(--shadow-color), 0 5px 10px var(--shadow-color); + box-shadow: + 0 1px 2px var(--shadow-color), + 0 5px 10px var(--shadow-color); border-radius: var(--border-radius); } diff --git a/webapp/app/pods/components/project-comments-list/styles.scss b/webapp/app/styles/components/project-comments-list.scss similarity index 100% rename from webapp/app/pods/components/project-comments-list/styles.scss rename to webapp/app/styles/components/project-comments-list.scss diff --git a/webapp/app/pods/components/project-comments-list/item/styles.scss b/webapp/app/styles/components/project-comments-list/item.scss similarity index 100% rename from webapp/app/pods/components/project-comments-list/item/styles.scss rename to webapp/app/styles/components/project-comments-list/item.scss diff --git a/webapp/app/pods/components/project-create-form/styles.scss b/webapp/app/styles/components/project-create-form.scss similarity index 98% rename from webapp/app/pods/components/project-create-form/styles.scss rename to webapp/app/styles/components/project-create-form.scss index a3dfa51c8..05aecd27d 100644 --- a/webapp/app/pods/components/project-create-form/styles.scss +++ b/webapp/app/styles/components/project-create-form.scss @@ -41,10 +41,10 @@ .colorInput { @extend %textInput; - width: 48px; margin-right: 5px; + width: 48px; height: 40px; - padding: 10px 14px; + padding: 5px 11px; cursor: pointer; } diff --git a/webapp/app/pods/components/project-file-operation/styles.scss b/webapp/app/styles/components/project-file-operation.scss similarity index 100% rename from webapp/app/pods/components/project-file-operation/styles.scss rename to webapp/app/styles/components/project-file-operation.scss diff --git a/webapp/app/pods/components/project-header/styles.scss b/webapp/app/styles/components/project-header.scss similarity index 100% rename from webapp/app/pods/components/project-header/styles.scss rename to webapp/app/styles/components/project-header.scss diff --git a/webapp/app/pods/components/project-navigation/styles.scss b/webapp/app/styles/components/project-navigation.scss similarity index 87% rename from webapp/app/pods/components/project-navigation/styles.scss rename to webapp/app/styles/components/project-navigation.scss index 9c7ec920d..092ae92ea 100644 --- a/webapp/app/pods/components/project-navigation/styles.scss +++ b/webapp/app/styles/components/project-navigation.scss @@ -40,16 +40,16 @@ } .project { - padding: 20px 0 20px 16px; + padding: 20px 10px 20px 16px; margin-left: 0; position: relative; display: flex; - align-items: flex-start; - font-size: 18px; + align-items: center; + font-size: 15px; font-weight: 700; color: var(--color-black); text-decoration: none; - line-height: 1; + line-height: 1.2; } .project-logo { @@ -59,13 +59,11 @@ top: 2px; font-size: 22px; margin-right: 10px; - line-height: 1.1; + line-height: 1.2; svg { - position: relative; - top: -2px; - width: 25px; - height: 25px; + width: 22px; + height: 22px; circle { fill: var(--logo-background); diff --git a/webapp/app/pods/components/project-navigation/list/styles.scss b/webapp/app/styles/components/project-navigation/list.scss similarity index 95% rename from webapp/app/pods/components/project-navigation/list/styles.scss rename to webapp/app/styles/components/project-navigation/list.scss index af1ac5c7b..c98a82277 100644 --- a/webapp/app/pods/components/project-navigation/list/styles.scss +++ b/webapp/app/styles/components/project-navigation/list.scss @@ -34,7 +34,7 @@ } :global([data-theme='dark']) .list-item-link:global(.active) { - color: rgba(255, 255, 255, 0.4); + color: var(--color-primary); } .list-item-link { @@ -43,11 +43,10 @@ position: relative; gap: 10px; left: 1px; - padding: 5px 12px 4px; + padding: 7px 12px 6px; text-decoration: none; font-size: 13px; border-radius: var(--border-radius); - margin-bottom: 3px; color: var(--text-color-normal); &:hover, diff --git a/webapp/app/pods/components/project-settings/api-token/styles.scss b/webapp/app/styles/components/project-settings/api-token.scss similarity index 100% rename from webapp/app/pods/components/project-settings/api-token/styles.scss rename to webapp/app/styles/components/project-settings/api-token.scss diff --git a/webapp/app/pods/components/project-settings/api-token/item/styles.scss b/webapp/app/styles/components/project-settings/api-token/item.scss similarity index 100% rename from webapp/app/pods/components/project-settings/api-token/item/styles.scss rename to webapp/app/styles/components/project-settings/api-token/item.scss diff --git a/webapp/app/pods/components/project-settings/back-link/styles.scss b/webapp/app/styles/components/project-settings/back-link.scss similarity index 100% rename from webapp/app/pods/components/project-settings/back-link/styles.scss rename to webapp/app/styles/components/project-settings/back-link.scss diff --git a/webapp/app/pods/components/project-settings/badges/styles.scss b/webapp/app/styles/components/project-settings/badges.scss similarity index 100% rename from webapp/app/pods/components/project-settings/badges/styles.scss rename to webapp/app/styles/components/project-settings/badges.scss diff --git a/webapp/app/pods/components/project-settings/collaborators/styles.scss b/webapp/app/styles/components/project-settings/collaborators.scss similarity index 88% rename from webapp/app/pods/components/project-settings/collaborators/styles.scss rename to webapp/app/styles/components/project-settings/collaborators.scss index c9e842803..cf8503fb1 100644 --- a/webapp/app/pods/components/project-settings/collaborators/styles.scss +++ b/webapp/app/styles/components/project-settings/collaborators.scss @@ -28,8 +28,15 @@ border-left: 1px solid var(--background-light-highlight); } +.rolesList-icon { + width: 13px; + height: 13px; +} + .rolesList-title { - display: block; + display: flex; + align-items: center; + gap: 5px; padding-bottom: 5px; margin: 15px 0 0; color: var(--color-primary); diff --git a/webapp/app/pods/components/project-settings/collaborators/create-form/styles.scss b/webapp/app/styles/components/project-settings/collaborators/create-form.scss similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/create-form/styles.scss rename to webapp/app/styles/components/project-settings/collaborators/create-form.scss diff --git a/webapp/app/pods/components/project-settings/collaborators/list/item/styles.scss b/webapp/app/styles/components/project-settings/collaborators/list/item.scss similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/list/item/styles.scss rename to webapp/app/styles/components/project-settings/collaborators/list/item.scss diff --git a/webapp/app/pods/components/project-settings/delete-form/styles.scss b/webapp/app/styles/components/project-settings/delete-form.scss similarity index 100% rename from webapp/app/pods/components/project-settings/delete-form/styles.scss rename to webapp/app/styles/components/project-settings/delete-form.scss diff --git a/webapp/app/pods/components/project-settings/form/styles.scss b/webapp/app/styles/components/project-settings/form.scss similarity index 96% rename from webapp/app/pods/components/project-settings/form/styles.scss rename to webapp/app/styles/components/project-settings/form.scss index 9d9b947e9..9c8acc882 100644 --- a/webapp/app/pods/components/project-settings/form/styles.scss +++ b/webapp/app/styles/components/project-settings/form.scss @@ -6,7 +6,6 @@ @extend %textInput; max-width: 350px; width: 100%; - margin-right: 5px; flex-grow: 0; flex-shrink: 1; padding: 10px; @@ -16,15 +15,15 @@ .colorInput { @extend %textInput; - margin-right: 5px; width: 48px; height: 40px; - padding: 10px 14px; + padding: 5px 11px; } .field { display: flex; margin-bottom: 15px; + gap: 8px; } .lock { diff --git a/webapp/app/pods/components/project-settings/integrations/styles.scss b/webapp/app/styles/components/project-settings/integrations.scss similarity index 87% rename from webapp/app/pods/components/project-settings/integrations/styles.scss rename to webapp/app/styles/components/project-settings/integrations.scss index 418615754..da8d07888 100644 --- a/webapp/app/pods/components/project-settings/integrations/styles.scss +++ b/webapp/app/styles/components/project-settings/integrations.scss @@ -23,7 +23,7 @@ .empty-description { display: grid; - grid-template-columns: 1fr 1fr 1fr; + grid-template-columns: 1fr 1fr 1fr 1fr; gap: 30px; margin-top: 20px; } @@ -35,6 +35,7 @@ text-align: center; border: 1px solid var(--background-light-highlight); background: var(--background-light); + color: var(--text-color-normal); border-radius: var(--border-radius); padding: 20px 30px; gap: 2px; @@ -53,7 +54,9 @@ &:focus, &:hover { - box-shadow: 0 1px 6px var(--shadow-color), 0 2px 19px var(--shadow-color); + box-shadow: + 0 1px 6px var(--shadow-color), + 0 2px 19px var(--shadow-color); } } @@ -70,7 +73,7 @@ color: var(--color-black); } -@media (max-width: 940px) { +@media (max-width: 1140px) { .empty-description { grid-template-columns: 1fr 1fr; gap: 15px; diff --git a/webapp/app/pods/components/project-settings/integrations/form/styles.scss b/webapp/app/styles/components/project-settings/integrations/form.scss similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/form/styles.scss rename to webapp/app/styles/components/project-settings/integrations/form.scss diff --git a/webapp/app/styles/components/project-settings/integrations/form/aws-s3.scss b/webapp/app/styles/components/project-settings/integrations/form/aws-s3.scss new file mode 100644 index 000000000..7984b260b --- /dev/null +++ b/webapp/app/styles/components/project-settings/integrations/form/aws-s3.scss @@ -0,0 +1,32 @@ +.container { + display: flex; + align-items: flex-start; + gap: 15px; +} + +.form { + width: 100%; + max-width: 400px; +} + +.policy { + width: 100%; +} + +.policy-title { + display: block; + font-weight: bold; + font-size: 13px; + margin-bottom: 6px; +} +.policy-render { + background: var(--content-background); + border: 1px solid var(--background-light-highlight); + border-radius: var(--border-radius); +} + +@media (max-width: 600px) { + .container { + flex-wrap: wrap; + } +} diff --git a/webapp/app/pods/components/project-settings/integrations/form/data-control-checkboxes/styles.scss b/webapp/app/styles/components/project-settings/integrations/form/data-control-checkboxes.scss similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/form/data-control-checkboxes/styles.scss rename to webapp/app/styles/components/project-settings/integrations/form/data-control-checkboxes.scss diff --git a/webapp/app/pods/components/project-settings/integrations/form/data-control-text/styles.scss b/webapp/app/styles/components/project-settings/integrations/form/data-control-text.scss similarity index 97% rename from webapp/app/pods/components/project-settings/integrations/form/data-control-text/styles.scss rename to webapp/app/styles/components/project-settings/integrations/form/data-control-text.scss index 1b6c57e51..3e5c5cfff 100644 --- a/webapp/app/pods/components/project-settings/integrations/form/data-control-text/styles.scss +++ b/webapp/app/styles/components/project-settings/integrations/form/data-control-text.scss @@ -1,5 +1,5 @@ .data-control { - margin-bottom: 15px; + margin-bottom: 10px; } .data-title { diff --git a/webapp/app/pods/components/project-settings/integrations/list/styles.scss b/webapp/app/styles/components/project-settings/integrations/list.scss similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/list/styles.scss rename to webapp/app/styles/components/project-settings/integrations/list.scss diff --git a/webapp/app/pods/components/project-settings/integrations/list/item/styles.scss b/webapp/app/styles/components/project-settings/integrations/list/item.scss similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/list/item/styles.scss rename to webapp/app/styles/components/project-settings/integrations/list/item.scss diff --git a/webapp/app/styles/components/project-settings/integrations/list/item/execute/aws-s3.scss b/webapp/app/styles/components/project-settings/integrations/list/item/execute/aws-s3.scss new file mode 100644 index 000000000..f199956c1 --- /dev/null +++ b/webapp/app/styles/components/project-settings/integrations/list/item/execute/aws-s3.scss @@ -0,0 +1,103 @@ +.aws-push-form { + padding: 20px; + background: var(--content-background); +} + +.title { + margin-bottom: 20px; + text-align: center; + line-height: 1.2; + font-size: 27px; + font-weight: 300; + color: var(--color-primary); +} + +.info { + display: flex; + flex-direction: column; + gap: 10px; + margin-bottom: 10px; + padding-bottom: 16px; + border-bottom: 1px solid var(--background-light-highlight); + font-size: 13px; + + div { + display: flex; + flex-direction: column; + gap: 2px; + } + + span { + font-size: 12px; + font-family: var(--font-monospace); + opacity: 0.6; + } +} + +.text { + font-size: 13px; + margin-bottom: 20px; + color: #555; +} + +.textInput { + @extend %textInput; + flex-grow: 1; + flex-shrink: 1; + padding: 10px; + min-width: 250px; + width: 100%; + font-size: 12px; + font-family: var(--font-primary); +} + +.errors { + margin-bottom: 15px; + padding-bottom: 5px; +} + +.error { + margin-bottom: 5px; + color: var(--color-error); + font-size: 13px; + font-weight: bold; +} + +.formActions { + display: flex; + justify-content: flex-end; + padding-top: 20px; +} + +.data-control { + .radio { + display: inline-flex; + align-items: center; + margin-right: 10px; + border: 1px solid var(--background-light-highlight); + padding: 4px 6px; + border-radius: var(--border-radius); + background: var(--input-background); + cursor: pointer; + font-size: 12px; + transition: 0.2s ease-in-out; + transition-property: background; + + &:hover, + &:focus { + background: var(--background-light); + } + + input { + margin-right: 5px; + cursor: pointer; + } + } +} + +.data-title { + display: block; + margin-bottom: 5px; + font-size: 13px; + font-weight: bold; +} diff --git a/webapp/app/pods/components/project-settings/integrations/list/item/execute/azure-storage-container/styles.scss b/webapp/app/styles/components/project-settings/integrations/list/item/execute/azure-storage-container.scss similarity index 100% rename from webapp/app/pods/components/project-settings/integrations/list/item/execute/azure-storage-container/styles.scss rename to webapp/app/styles/components/project-settings/integrations/list/item/execute/azure-storage-container.scss diff --git a/webapp/app/pods/components/project-settings/jipt/styles.scss b/webapp/app/styles/components/project-settings/jipt.scss similarity index 90% rename from webapp/app/pods/components/project-settings/jipt/styles.scss rename to webapp/app/styles/components/project-settings/jipt.scss index 9dd93c893..4e39008b8 100644 --- a/webapp/app/pods/components/project-settings/jipt/styles.scss +++ b/webapp/app/styles/components/project-settings/jipt.scss @@ -1,5 +1,12 @@ .project-settings-jipt { margin-top: 30px; + + hr { + margin: 20px 0; + border: 0; + height: 1px; + background: var(--background-light-highlight); + } } .text { diff --git a/webapp/app/pods/components/project-settings/links-list/styles.scss b/webapp/app/styles/components/project-settings/links-list.scss similarity index 85% rename from webapp/app/pods/components/project-settings/links-list/styles.scss rename to webapp/app/styles/components/project-settings/links-list.scss index 5271851e7..81fc2a22a 100644 --- a/webapp/app/pods/components/project-settings/links-list/styles.scss +++ b/webapp/app/styles/components/project-settings/links-list.scss @@ -19,12 +19,15 @@ background: var(--content-background); border-radius: var(--border-radius); color: var(--color-black); - box-shadow: 0 1px 4px var(--shadow-color), 0 9px 19px var(--shadow-color); + box-shadow: + 0 1px 4px var(--shadow-color), + 0 9px 19px var(--shadow-color); text-decoration: none; font-weight: 600; font-size: 13px; transition: 0.2s ease-in-out; transition-property: background, box-shadow, color, transform; + overflow: hidden; strong { padding: 10px 20px 5px; @@ -33,9 +36,10 @@ p { display: block; width: 100%; - padding: 10px 20px; + padding: 10px 20px 10px 19px; margin-top: 10px; font-size: 11px; + font-weight: 400; border-top: 1px solid var(--content-background-border); color: var(--color-grey); background: var(--background-light); @@ -46,7 +50,9 @@ &:focus, &:hover { color: var(--color-primary); - box-shadow: 0 3px 8px var(--shadow-color), 0 9px 12px var(--shadow-color); + box-shadow: + 0 3px 8px var(--shadow-color), + 0 9px 12px var(--shadow-color); transform: scale(1.02); .link-check { @@ -60,6 +66,10 @@ height: 14px; } +.link-check-text { + font-size: 11px; +} + .link-check { position: absolute; top: 7px; @@ -75,12 +85,12 @@ } .link-icon { - width: 20px; - height: 20px; + width: 30px; + height: 30px; margin: 20px 20px 0; transition: 0.2s ease-in-out; transition-property: stroke; - opacity: 0.6; + opacity: 0.8; stroke: var(--color-black); transition: 0.2s ease-in-out; transition-property: stroke; diff --git a/webapp/app/pods/components/project-settings/machine-translations/styles.scss b/webapp/app/styles/components/project-settings/machine-translations.scss similarity index 100% rename from webapp/app/pods/components/project-settings/machine-translations/styles.scss rename to webapp/app/styles/components/project-settings/machine-translations.scss diff --git a/webapp/app/pods/components/project-settings/manage-languages/styles.scss b/webapp/app/styles/components/project-settings/manage-languages.scss similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/styles.scss rename to webapp/app/styles/components/project-settings/manage-languages.scss diff --git a/webapp/app/pods/components/project-settings/manage-languages/create-form/styles.scss b/webapp/app/styles/components/project-settings/manage-languages/create-form.scss similarity index 83% rename from webapp/app/pods/components/project-settings/manage-languages/create-form/styles.scss rename to webapp/app/styles/components/project-settings/manage-languages/create-form.scss index d9ab3ecbd..8eec7d755 100644 --- a/webapp/app/pods/components/project-settings/manage-languages/create-form/styles.scss +++ b/webapp/app/styles/components/project-settings/manage-languages/create-form.scss @@ -34,7 +34,11 @@ } .optionLabel-icon { - width: 15px; + width: 22px; margin-left: 3px; transform: rotate(45deg); + background: var(--color-primary-opacity-25); + border-radius: 50%; + padding: 3px; + color: var(--color-primary); } diff --git a/webapp/app/pods/components/project-settings/manage-languages/overview/styles.scss b/webapp/app/styles/components/project-settings/manage-languages/overview.scss similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/overview/styles.scss rename to webapp/app/styles/components/project-settings/manage-languages/overview.scss diff --git a/webapp/app/pods/components/project-settings/manage-languages/overview/item/styles.scss b/webapp/app/styles/components/project-settings/manage-languages/overview/item.scss similarity index 100% rename from webapp/app/pods/components/project-settings/manage-languages/overview/item/styles.scss rename to webapp/app/styles/components/project-settings/manage-languages/overview/item.scss diff --git a/webapp/app/pods/components/project-settings/prompts/styles.scss b/webapp/app/styles/components/project-settings/prompts.scss similarity index 78% rename from webapp/app/pods/components/project-settings/prompts/styles.scss rename to webapp/app/styles/components/project-settings/prompts.scss index d29a65808..c165870d8 100644 --- a/webapp/app/pods/components/project-settings/prompts/styles.scss +++ b/webapp/app/styles/components/project-settings/prompts.scss @@ -1,11 +1,11 @@ .wrapper { display: flex; + flex-direction: column; margin-top: 25px; } .content { - flex: 1 1 auto; - padding-right: 30px; + margin-top: 20px; } .list { @@ -13,10 +13,10 @@ flex-direction: column; gap: 10px; margin-bottom: 20px; + max-width: 450px; } .config { - flex: 0 1 auto; max-width: 450px; width: 100%; } @@ -30,5 +30,5 @@ .text { display: block; font-size: 13px; - padding-bottom: 22px; + margin-bottom: 12px; } diff --git a/webapp/app/pods/components/project-settings/prompts/config/styles.scss b/webapp/app/styles/components/project-settings/prompts/config.scss similarity index 83% rename from webapp/app/pods/components/project-settings/prompts/config/styles.scss rename to webapp/app/styles/components/project-settings/prompts/config.scss index 54fe10c81..04b3d1ce4 100644 --- a/webapp/app/pods/components/project-settings/prompts/config/styles.scss +++ b/webapp/app/styles/components/project-settings/prompts/config.scss @@ -1,5 +1,4 @@ .form { - margin-top: 25px; padding: 16px; border-radius: var(--border-radius); background: var(--background-light); @@ -31,7 +30,6 @@ } .select select { - background: transparent; border: 1px solid var(--background-light-highlight); padding: 7px 10px; font-weight: bold; @@ -67,9 +65,25 @@ font-size: 12px; } +.options { + margin: 10px 0 0; + display: flex; + flex-direction: column; + gap: 5px; +} + +.option-checkbox { + display: flex; + align-items: center; + width: 100%; + gap: 8px; + font-size: 12px; + font-weight: bold; +} + .actions { display: flex; justify-content: flex-end; gap: 12px; - margin-top: 20px; + margin-top: 6px; } diff --git a/webapp/app/pods/components/project-settings/prompts/item/styles.scss b/webapp/app/styles/components/project-settings/prompts/item.scss similarity index 100% rename from webapp/app/pods/components/project-settings/prompts/item/styles.scss rename to webapp/app/styles/components/project-settings/prompts/item.scss diff --git a/webapp/app/pods/components/project-settings/title/styles.scss b/webapp/app/styles/components/project-settings/title.scss similarity index 100% rename from webapp/app/pods/components/project-settings/title/styles.scss rename to webapp/app/styles/components/project-settings/title.scss diff --git a/webapp/app/pods/components/projects-filters/styles.scss b/webapp/app/styles/components/projects-filters.scss similarity index 97% rename from webapp/app/pods/components/projects-filters/styles.scss rename to webapp/app/styles/components/projects-filters.scss index 8f332d19a..374eb1733 100644 --- a/webapp/app/pods/components/projects-filters/styles.scss +++ b/webapp/app/styles/components/projects-filters.scss @@ -1,4 +1,4 @@ -div.filters { +.filters:global(.filters) { margin: 20px auto 0; padding: 0 20px; max-width: var(--screen-lg); diff --git a/webapp/app/pods/components/projects-header/styles.scss b/webapp/app/styles/components/projects-header.scss similarity index 100% rename from webapp/app/pods/components/projects-header/styles.scss rename to webapp/app/styles/components/projects-header.scss diff --git a/webapp/app/pods/components/projects-list/styles.scss b/webapp/app/styles/components/projects-list.scss similarity index 91% rename from webapp/app/pods/components/projects-list/styles.scss rename to webapp/app/styles/components/projects-list.scss index 2c924a8b9..025a0c18d 100644 --- a/webapp/app/pods/components/projects-list/styles.scss +++ b/webapp/app/styles/components/projects-list.scss @@ -8,8 +8,8 @@ } .item { - flex: 1 1 calc(50% - 10px); - max-width: 50%; + flex: 1 1 calc(33.3% - 10px); + max-width: 33.3%; } .item-link { diff --git a/webapp/app/pods/components/projects-list/item/styles.scss b/webapp/app/styles/components/projects-list/item.scss similarity index 58% rename from webapp/app/pods/components/projects-list/item/styles.scss rename to webapp/app/styles/components/projects-list/item.scss index b6599678e..4a2b93c67 100644 --- a/webapp/app/pods/components/projects-list/item/styles.scss +++ b/webapp/app/styles/components/projects-list/item.scss @@ -1,37 +1,7 @@ -.projects-list-item.low-percentage { - .language-reviewedPercentage { - color: var(--color-error); - } - - .progress { - color: var(--color-error); - } -} - -.projects-list-item.medium-percentage { - .language-reviewedPercentage { - color: var(--color-warning); - } - - .progress { - color: var(--color-warning); - } -} - -.projects-list-item.high-percentage { - .language-reviewedPercentage { - color: var(--color-success); - } - - .progress { - color: var(--color-success); - } -} - .projects-list-item { padding: 12px 15px; transition: 0.2s ease-in-out; - transition-property: box-shadow; + transition-property: box-shadow, border-color; border-radius: var(--border-radius); border: 1px solid var(--content-background-border); background: var(--content-background); @@ -39,6 +9,14 @@ &:focus, &:hover { + box-shadow: 0 4px 15px var(--shadow-color); + border-color: color-mix(in srgb, var(--color-primary) 20%, transparent); + border-bottom-color: color-mix( + in srgb, + var(--color-primary) 25%, + transparent + ); + .projectName { color: var(--color-primary); } @@ -87,12 +65,4 @@ display: flex; justify-content: space-between; align-items: center; - margin-bottom: 6px; -} - -.numberStat-totals { - padding: 2px 0 1px 7px; - color: var(--color-black); - font-size: 12px; - font-family: var(--font-monospace); } diff --git a/webapp/app/pods/components/prompt-create-form/styles.scss b/webapp/app/styles/components/prompt-create-form.scss similarity index 93% rename from webapp/app/pods/components/prompt-create-form/styles.scss rename to webapp/app/styles/components/prompt-create-form.scss index 10cab4a86..1aa312bb1 100644 --- a/webapp/app/pods/components/prompt-create-form/styles.scss +++ b/webapp/app/styles/components/prompt-create-form.scss @@ -25,7 +25,8 @@ .quickAccess-picker-empty { width: 23px; - opacity: 0.2; + opacity: 0.7; + color: var(--text-color-normal); } .quickAccess-picker-remove { @@ -37,6 +38,7 @@ top: 0; opacity: 0.5; padding: 0; + color: var(--text-color-normal); } .quickAccess-picker-remove-icon { width: 13px; diff --git a/webapp/app/pods/components/prompt-update-form/styles.scss b/webapp/app/styles/components/prompt-update-form.scss similarity index 93% rename from webapp/app/pods/components/prompt-update-form/styles.scss rename to webapp/app/styles/components/prompt-update-form.scss index 6fabcc981..9f38c2d8c 100644 --- a/webapp/app/pods/components/prompt-update-form/styles.scss +++ b/webapp/app/styles/components/prompt-update-form.scss @@ -25,7 +25,8 @@ .quickAccess-picker-empty { width: 23px; - opacity: 0.2; + opacity: 0.7; + color: var(--text-color-normal); } .quickAccess-picker-remove { @@ -37,6 +38,7 @@ top: 0; opacity: 0.5; padding: 0; + color: var(--text-color-normal); } .quickAccess-picker-remove-icon { width: 13px; diff --git a/webapp/app/pods/components/recent-projects-list/styles.scss b/webapp/app/styles/components/recent-projects-list.scss similarity index 89% rename from webapp/app/pods/components/recent-projects-list/styles.scss rename to webapp/app/styles/components/recent-projects-list.scss index 88c714bf1..de696c494 100644 --- a/webapp/app/pods/components/recent-projects-list/styles.scss +++ b/webapp/app/styles/components/recent-projects-list.scss @@ -23,12 +23,11 @@ .item-link { display: block; - padding: 4px 10px 5px; - background: var(--background-light); + padding: 3px 8px 4px; + border: 1px solid var(--background-light-highlight); border-radius: var(--border-radius); text-decoration: none; font-weight: 600; - box-shadow: 0 1px 2px var(--shadow-color); transition: 0.2s ease-in-out; transition-property: background, box-shadow; diff --git a/webapp/app/pods/components/recent-projects-list/item/styles.scss b/webapp/app/styles/components/recent-projects-list/item.scss similarity index 100% rename from webapp/app/pods/components/recent-projects-list/item/styles.scss rename to webapp/app/styles/components/recent-projects-list/item.scss diff --git a/webapp/app/pods/components/related-translations-list/styles.scss b/webapp/app/styles/components/related-translations-list.scss similarity index 100% rename from webapp/app/pods/components/related-translations-list/styles.scss rename to webapp/app/styles/components/related-translations-list.scss diff --git a/webapp/app/pods/components/related-translations-list/item/styles.scss b/webapp/app/styles/components/related-translations-list/item.scss similarity index 100% rename from webapp/app/pods/components/related-translations-list/item/styles.scss rename to webapp/app/styles/components/related-translations-list/item.scss diff --git a/webapp/app/pods/components/removed-translation-edit/styles.scss b/webapp/app/styles/components/removed-translation-edit.scss similarity index 100% rename from webapp/app/pods/components/removed-translation-edit/styles.scss rename to webapp/app/styles/components/removed-translation-edit.scss diff --git a/webapp/app/pods/components/resource-pagination/styles.scss b/webapp/app/styles/components/resource-pagination.scss similarity index 100% rename from webapp/app/pods/components/resource-pagination/styles.scss rename to webapp/app/styles/components/resource-pagination.scss diff --git a/webapp/app/pods/components/review-progress-bar/styles.scss b/webapp/app/styles/components/review-progress-bar.scss similarity index 100% rename from webapp/app/pods/components/review-progress-bar/styles.scss rename to webapp/app/styles/components/review-progress-bar.scss diff --git a/webapp/app/pods/components/revision-export-options/styles.scss b/webapp/app/styles/components/revision-export-options.scss similarity index 97% rename from webapp/app/pods/components/revision-export-options/styles.scss rename to webapp/app/styles/components/revision-export-options.scss index 7d1a7d65b..77ffd3c5c 100644 --- a/webapp/app/pods/components/revision-export-options/styles.scss +++ b/webapp/app/styles/components/revision-export-options.scss @@ -7,6 +7,7 @@ .exportOptions { display: flex; justify-content: space-between; + width: 100%; } .exportOptions-advancedFilters { diff --git a/webapp/app/pods/components/revision-export-options/advanced-filters/styles.scss b/webapp/app/styles/components/revision-export-options/advanced-filters.scss similarity index 100% rename from webapp/app/pods/components/revision-export-options/advanced-filters/styles.scss rename to webapp/app/styles/components/revision-export-options/advanced-filters.scss diff --git a/webapp/app/pods/components/revision-selector/styles.scss b/webapp/app/styles/components/revision-selector.scss similarity index 88% rename from webapp/app/pods/components/revision-selector/styles.scss rename to webapp/app/styles/components/revision-selector.scss index dd7a96127..fecd744ca 100644 --- a/webapp/app/pods/components/revision-selector/styles.scss +++ b/webapp/app/styles/components/revision-selector.scss @@ -2,6 +2,14 @@ position: relative; margin-bottom: 15px; + &.revision-selector--jipt { + margin-bottom: 0; + + .overlay { + border-radius: 0; + } + } + select { padding: 10px 10px 10px 7px; padding-bottom: 25px; diff --git a/webapp/app/pods/components/revision-update-form/styles.scss b/webapp/app/styles/components/revision-update-form.scss similarity index 100% rename from webapp/app/pods/components/revision-update-form/styles.scss rename to webapp/app/styles/components/revision-update-form.scss diff --git a/webapp/app/pods/components/skeleton-ui/styles.scss b/webapp/app/styles/components/skeleton-ui.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/styles.scss rename to webapp/app/styles/components/skeleton-ui.scss diff --git a/webapp/app/pods/components/skeleton-ui/activities-list/styles.scss b/webapp/app/styles/components/skeleton-ui/activities-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/activities-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/activities-list.scss diff --git a/webapp/app/pods/components/skeleton-ui/conflicts-items/styles.scss b/webapp/app/styles/components/skeleton-ui/conflicts-items.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/conflicts-items/styles.scss rename to webapp/app/styles/components/skeleton-ui/conflicts-items.scss diff --git a/webapp/app/pods/components/skeleton-ui/documents-list/styles.scss b/webapp/app/styles/components/skeleton-ui/documents-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/documents-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/documents-list.scss diff --git a/webapp/app/pods/components/skeleton-ui/progress-line/styles.scss b/webapp/app/styles/components/skeleton-ui/progress-line.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/progress-line/styles.scss rename to webapp/app/styles/components/skeleton-ui/progress-line.scss diff --git a/webapp/app/pods/components/skeleton-ui/project-activities-filter/styles.scss b/webapp/app/styles/components/skeleton-ui/project-activities-filter.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/project-activities-filter/styles.scss rename to webapp/app/styles/components/skeleton-ui/project-activities-filter.scss diff --git a/webapp/app/pods/components/skeleton-ui/project-comments-list/styles.scss b/webapp/app/styles/components/skeleton-ui/project-comments-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/project-comments-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/project-comments-list.scss diff --git a/webapp/app/pods/components/skeleton-ui/project-navigation/styles.scss b/webapp/app/styles/components/skeleton-ui/project-navigation.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/project-navigation/styles.scss rename to webapp/app/styles/components/skeleton-ui/project-navigation.scss diff --git a/webapp/app/pods/components/skeleton-ui/translation-comments-list/styles.scss b/webapp/app/styles/components/skeleton-ui/translation-comments-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/translation-comments-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/translation-comments-list.scss diff --git a/webapp/app/pods/components/skeleton-ui/translation-splash-title/styles.scss b/webapp/app/styles/components/skeleton-ui/translation-splash-title.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/translation-splash-title/styles.scss rename to webapp/app/styles/components/skeleton-ui/translation-splash-title.scss diff --git a/webapp/app/pods/components/skeleton-ui/translations-list/styles.scss b/webapp/app/styles/components/skeleton-ui/translations-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/translations-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/translations-list.scss diff --git a/webapp/app/pods/components/skeleton-ui/versions-list/styles.scss b/webapp/app/styles/components/skeleton-ui/versions-list.scss similarity index 100% rename from webapp/app/pods/components/skeleton-ui/versions-list/styles.scss rename to webapp/app/styles/components/skeleton-ui/versions-list.scss diff --git a/webapp/app/pods/components/translation-activities-list/styles.scss b/webapp/app/styles/components/translation-activities-list.scss similarity index 100% rename from webapp/app/pods/components/translation-activities-list/styles.scss rename to webapp/app/styles/components/translation-activities-list.scss diff --git a/webapp/app/pods/components/translation-comment-delete/styles.scss b/webapp/app/styles/components/translation-comment-delete.scss similarity index 100% rename from webapp/app/pods/components/translation-comment-delete/styles.scss rename to webapp/app/styles/components/translation-comment-delete.scss diff --git a/webapp/app/pods/components/translation-comment-form/styles.scss b/webapp/app/styles/components/translation-comment-form.scss similarity index 100% rename from webapp/app/pods/components/translation-comment-form/styles.scss rename to webapp/app/styles/components/translation-comment-form.scss diff --git a/webapp/app/pods/components/translation-comments-list/styles.scss b/webapp/app/styles/components/translation-comments-list.scss similarity index 89% rename from webapp/app/pods/components/translation-comments-list/styles.scss rename to webapp/app/styles/components/translation-comments-list.scss index 0184ffacc..6bac29eb4 100644 --- a/webapp/app/pods/components/translation-comments-list/styles.scss +++ b/webapp/app/styles/components/translation-comments-list.scss @@ -1,6 +1,8 @@ .translation-comments-list { position: relative; - box-shadow: 0 1px 4px var(--shadow-color), 0 9px 19px var(--shadow-color); + box-shadow: + 0 1px 4px var(--shadow-color), + 0 9px 19px var(--shadow-color); background: var(--background-light); border-radius: var(--border-radius); } diff --git a/webapp/app/pods/components/translation-comments-list/item/styles.scss b/webapp/app/styles/components/translation-comments-list/item.scss similarity index 100% rename from webapp/app/pods/components/translation-comments-list/item/styles.scss rename to webapp/app/styles/components/translation-comments-list/item.scss diff --git a/webapp/app/pods/components/translation-comments-subscriptions/styles.scss b/webapp/app/styles/components/translation-comments-subscriptions.scss similarity index 100% rename from webapp/app/pods/components/translation-comments-subscriptions/styles.scss rename to webapp/app/styles/components/translation-comments-subscriptions.scss diff --git a/webapp/app/pods/components/translation-comments-subscriptions/item/styles.scss b/webapp/app/styles/components/translation-comments-subscriptions/item.scss similarity index 100% rename from webapp/app/pods/components/translation-comments-subscriptions/item/styles.scss rename to webapp/app/styles/components/translation-comments-subscriptions/item.scss diff --git a/webapp/app/pods/components/translation-conversation/styles.scss b/webapp/app/styles/components/translation-conversation.scss similarity index 100% rename from webapp/app/pods/components/translation-conversation/styles.scss rename to webapp/app/styles/components/translation-conversation.scss diff --git a/webapp/app/pods/components/translation-edit/styles.scss b/webapp/app/styles/components/translation-edit.scss similarity index 98% rename from webapp/app/pods/components/translation-edit/styles.scss rename to webapp/app/styles/components/translation-edit.scss index 8e0b398a9..09264e300 100644 --- a/webapp/app/pods/components/translation-edit/styles.scss +++ b/webapp/app/styles/components/translation-edit.scss @@ -2,7 +2,7 @@ width: 100%; } -.full-width { +.jipt { padding-right: 15px; padding-left: 15px; } diff --git a/webapp/app/pods/components/translation-edit/form/styles.scss b/webapp/app/styles/components/translation-edit/form.scss similarity index 88% rename from webapp/app/pods/components/translation-edit/form/styles.scss rename to webapp/app/styles/components/translation-edit/form.scss index 4273fd00d..66962adfd 100644 --- a/webapp/app/pods/components/translation-edit/form/styles.scss +++ b/webapp/app/styles/components/translation-edit/form.scss @@ -74,12 +74,21 @@ width: 100%; height: 100%; padding: 10px 130px 10px 10px; - font-size: 12px; &[dir='rtl'] { padding: 10px 10px 10px 130px; } + &.inputText--borderless { + border-color: transparent; + background: transparent; + + &:focus { + border-color: transparent; + background: transparent; + } + } + &::placeholder { opacity: 0.2; font-style: italic; @@ -122,7 +131,9 @@ font-size: 12px; color: var(--text-color-normal); opacity: 0.8; - transition: color 0.2s ease-in-out, opacity 0.2s ease-in-out; + transition: + color 0.2s ease-in-out, + opacity 0.2s ease-in-out; &.placeholders-item--warning { opacity: 1; @@ -176,3 +187,12 @@ margin-right: 6px; stroke: var(--color-black); } + +.lint-messages { + transition: 0.2s ease-in-out; + transition-property: opacity; +} + +.lint-messages--loading { + opacity: 0.5; +} diff --git a/webapp/app/pods/components/translation-edit/helpers/styles.scss b/webapp/app/styles/components/translation-edit/helpers.scss similarity index 100% rename from webapp/app/pods/components/translation-edit/helpers/styles.scss rename to webapp/app/styles/components/translation-edit/helpers.scss diff --git a/webapp/app/pods/components/translation-editions-list/styles.scss b/webapp/app/styles/components/translation-editions-list.scss similarity index 50% rename from webapp/app/pods/components/translation-editions-list/styles.scss rename to webapp/app/styles/components/translation-editions-list.scss index b3381f679..2ba260253 100644 --- a/webapp/app/pods/components/translation-editions-list/styles.scss +++ b/webapp/app/styles/components/translation-editions-list.scss @@ -2,13 +2,6 @@ margin-top: 20px; } -.icon { - display: block; - width: 50px; - height: 50px; - margin-bottom: 10px; -} - .empty-content { margin-top: 20px; } diff --git a/webapp/app/pods/components/translation-editions-list/item/styles.scss b/webapp/app/styles/components/translation-editions-list/item.scss similarity index 100% rename from webapp/app/pods/components/translation-editions-list/item/styles.scss rename to webapp/app/styles/components/translation-editions-list/item.scss diff --git a/webapp/app/pods/components/translation-index/styles.scss b/webapp/app/styles/components/translation-index.scss similarity index 96% rename from webapp/app/pods/components/translation-index/styles.scss rename to webapp/app/styles/components/translation-index.scss index 2c0bc943c..8e3a140ca 100644 --- a/webapp/app/pods/components/translation-index/styles.scss +++ b/webapp/app/styles/components/translation-index.scss @@ -1,6 +1,6 @@ .wrapper { display: flex; - margin-top: 30px; + margin-top: 10px; & > div:first-of-type { margin-right: 40px; diff --git a/webapp/app/pods/components/translation-navigation/styles.scss b/webapp/app/styles/components/translation-navigation.scss similarity index 79% rename from webapp/app/pods/components/translation-navigation/styles.scss rename to webapp/app/styles/components/translation-navigation.scss index c2116f793..cc9e99242 100644 --- a/webapp/app/pods/components/translation-navigation/styles.scss +++ b/webapp/app/styles/components/translation-navigation.scss @@ -1,19 +1,15 @@ .navigation--alt { - border-bottom: 1px solid var(--input-border-color); - .navigation-list-item { margin-right: 6px; } .navigation-list-item-link { - position: relative; - top: 1px; - padding: 10px 15px 8px 10px; - border-bottom: 2px solid transparent; + padding: 2px 11px 2px 10px; + border-radius: var(--border-radius); &:global(.active) { - border-bottom-color: var(--color-primary); - color: var(--color-primary); + background: var(--background-light-highlight); + opacity: 0.7; } } } @@ -21,7 +17,6 @@ .navigation-list { display: flex; align-items: stretch; - min-height: 40px; position: relative; left: -6px; z-index: 10; @@ -45,21 +40,19 @@ position: relative; padding: 5px 5px 6px 3px; text-decoration: none; - font-size: 14px; + font-size: 13px; font-weight: 600; color: var(--color-black); &:focus, &:hover, :global(&.active) { - color: var(--color-primary); - .navigation-list-item-link-text { opacity: 1; } .navigation-list-item-link-icon { - stroke: var(--color-primary); + stroke: currentColor; } } } @@ -73,9 +66,9 @@ .navigation-list-item-link-icon { transition: 0.2s ease-in-out; transition-property: stroke; - width: 16px; - height: 16px; - margin-right: 7px; + width: 12px; + height: 12px; + margin-right: 5px; opacity: 0.6; stroke: var(--color-black); } diff --git a/webapp/app/pods/components/translation-splash-title/styles.scss b/webapp/app/styles/components/translation-splash-title.scss similarity index 100% rename from webapp/app/pods/components/translation-splash-title/styles.scss rename to webapp/app/styles/components/translation-splash-title.scss diff --git a/webapp/app/pods/components/translations-filter/styles.scss b/webapp/app/styles/components/translations-filter.scss similarity index 81% rename from webapp/app/pods/components/translations-filter/styles.scss rename to webapp/app/styles/components/translations-filter.scss index 6c111128d..525aa8d8d 100644 --- a/webapp/app/pods/components/translations-filter/styles.scss +++ b/webapp/app/styles/components/translations-filter.scss @@ -4,11 +4,6 @@ align-items: flex-start; } -.filters-content { - flex-grow: 1; - margin-right: 15px; -} - .queryForm { position: relative; } @@ -32,7 +27,9 @@ color: var(--color-black); &:focus { - box-shadow: inset 0 1px 2px rgba(#000, 0.1), 0 1px 2px var(--shadow-color); + box-shadow: + inset 0 1px 2px rgba(#000, 0.1), + 0 1px 2px var(--shadow-color); } &::placeholder { @@ -43,6 +40,7 @@ button.advancedFilters { position: relative; box-shadow: none; + flex-shrink: 0; &:focus, &:hover { transform: translate3d(0, 0, 0); @@ -60,13 +58,6 @@ button.advancedFilters { font-size: 10px; } -.totalEntries { - flex-shrink: 0; - margin-top: 10px; - color: var(--color-grey); - font-size: 12px; -} - @media (max-width: 440px) { .filters-wrapper { flex-direction: column; diff --git a/webapp/app/pods/components/translations-filter/advanced-filters/styles.scss b/webapp/app/styles/components/translations-filter/advanced-filters.scss similarity index 100% rename from webapp/app/pods/components/translations-filter/advanced-filters/styles.scss rename to webapp/app/styles/components/translations-filter/advanced-filters.scss diff --git a/webapp/app/pods/components/translations-list/styles.scss b/webapp/app/styles/components/translations-list.scss similarity index 100% rename from webapp/app/pods/components/translations-list/styles.scss rename to webapp/app/styles/components/translations-list.scss diff --git a/webapp/app/pods/components/translations-list/item/styles.scss b/webapp/app/styles/components/translations-list/item.scss similarity index 98% rename from webapp/app/pods/components/translations-list/item/styles.scss rename to webapp/app/styles/components/translations-list/item.scss index 45b2b6cbc..87b81155d 100644 --- a/webapp/app/pods/components/translations-list/item/styles.scss +++ b/webapp/app/styles/components/translations-list/item.scss @@ -160,6 +160,7 @@ .item-meta { display: flex; align-items: center; + gap: 3px; } .item-text { @@ -193,7 +194,7 @@ .item-updatedAt { opacity: 0; - transform: translateX(-10px); + transform: translateX(-5px); margin-left: 5px; color: var(--color-grey); font-size: 11px; diff --git a/webapp/app/pods/components/version-create-form/styles.scss b/webapp/app/styles/components/version-create-form.scss similarity index 83% rename from webapp/app/pods/components/version-create-form/styles.scss rename to webapp/app/styles/components/version-create-form.scss index c9e05c208..ea3535572 100644 --- a/webapp/app/pods/components/version-create-form/styles.scss +++ b/webapp/app/styles/components/version-create-form.scss @@ -51,6 +51,18 @@ font-size: 13px; } +.formItem-checkbox { + margin-bottom: 4px; + display: flex; + align-items: center; + gap: 6px; +} + +.formItem-help { + font-size: 11px; + opacity: 0.6; +} + .formActions { padding-top: 10px; } diff --git a/webapp/app/pods/components/version-update-form/styles.scss b/webapp/app/styles/components/version-update-form.scss similarity index 83% rename from webapp/app/pods/components/version-update-form/styles.scss rename to webapp/app/styles/components/version-update-form.scss index 292153c88..e3f609623 100644 --- a/webapp/app/pods/components/version-update-form/styles.scss +++ b/webapp/app/styles/components/version-update-form.scss @@ -51,6 +51,18 @@ font-size: 13px; } +.formItem-checkbox { + margin-bottom: 4px; + display: flex; + align-items: center; + gap: 6px; +} + +.formItem-help { + font-size: 11px; + opacity: 0.6; +} + .formActions { padding-top: 10px; } diff --git a/webapp/app/pods/components/versions-list/styles.scss b/webapp/app/styles/components/versions-list.scss similarity index 100% rename from webapp/app/pods/components/versions-list/styles.scss rename to webapp/app/styles/components/versions-list.scss diff --git a/webapp/app/pods/components/versions-list/item/styles.scss b/webapp/app/styles/components/versions-list/item.scss similarity index 100% rename from webapp/app/pods/components/versions-list/item/styles.scss rename to webapp/app/styles/components/versions-list/item.scss diff --git a/webapp/app/pods/components/welcome-project/styles.scss b/webapp/app/styles/components/welcome-project.scss similarity index 88% rename from webapp/app/pods/components/welcome-project/styles.scss rename to webapp/app/styles/components/welcome-project.scss index 2f44273ef..182b190e7 100644 --- a/webapp/app/pods/components/welcome-project/styles.scss +++ b/webapp/app/styles/components/welcome-project.scss @@ -51,7 +51,9 @@ padding: 20px; margin-bottom: 10px; margin-right: 10px; - box-shadow: 0 4px 8px var(--shadow-color), 0 5px 15px var(--shadow-color); + box-shadow: + 0 4px 8px var(--shadow-color), + 0 5px 15px var(--shadow-color); border-radius: var(--border-radius); color: var(--color-primary); text-decoration: none; @@ -62,7 +64,9 @@ &:focus, &:hover { background: var(--color-primary-opacity-10); - box-shadow: 0 1px 6px var(--shadow-color), 0 5px 18px var(--shadow-color); + box-shadow: + 0 1px 6px var(--shadow-color), + 0 5px 18px var(--shadow-color); } } diff --git a/webapp/app/styles/html-components/button.scss b/webapp/app/styles/html-components/button.scss index 934c24cad..ad7c3b0fb 100644 --- a/webapp/app/styles/html-components/button.scss +++ b/webapp/app/styles/html-components/button.scss @@ -76,6 +76,18 @@ } } +.button--blue { + --button-text-color: var(--color-blue); +} + +.button--green { + --button-text-color: var(--color-green); +} + +.button--grey { + --button-text-color: var(--color-grey); +} + .button--grey, .button--success, .button--black, @@ -104,7 +116,9 @@ padding: 5px 12px; background: var(--button-background-color); border: 1px solid var(--button-border-color); - box-shadow: 0 1px 2px var(--shadow-color), 0 2px 6px var(--shadow-color); + box-shadow: + 0 1px 2px var(--shadow-color), + 0 2px 6px var(--shadow-color); text-shadow: 0 1px 1px var(--shadow-color); color: var(--button-text-color); @@ -172,8 +186,8 @@ } &.button--grey { - --button-text-hover-color: #fff; - --button-text-color: #fff; + --button-text-hover-color: var(--color-black); + --button-text-color: var(--color-black); --button-background-color: var(--background-light); --button-background-hover-color: darken(#ccc, 4%); --button-border-color: darken(#ccc, 4%); @@ -235,20 +249,13 @@ } } -.button--grey { - color: #000; - border-color: var(--background-light); - - &.button--loading svg.loading { - fill: #8a8a8a; - } +.button--borderLess { + border-color: transparent; +} - &:hover, - &:focus { - color: darken(#8a8a8a, 10%); - background: lighten(#8a8a8a, 43%); - border-color: var(--background-light-highlight); - } +.button--grey { + --button-text-color: var(--color-grey); + --button-text-hover-color: var(--color-grey); } .button--green { diff --git a/webapp/app/styles/html-components/emoji-picker.scss b/webapp/app/styles/html-components/emoji-picker.scss deleted file mode 100644 index f0377a011..000000000 --- a/webapp/app/styles/html-components/emoji-picker.scss +++ /dev/null @@ -1,59 +0,0 @@ -div.emoji-picker__wrapper { - z-index: 3001; - border: 0; - box-shadow: 0 3px 15px var(--shadow-color); - overflow: hidden; - - .emoji-picker__tab-body h2 { - display: none; - } - - .emoji-picker__variant-overlay { - background: rgba(#fff, 0.7); - } - - .emoji-picker__variant-popup-close-button { - display: none; - } - - .emoji-picker__variant-popup { - border-radius: var(--border-radius); - box-shadow: 0 3px 10px rgba(#3a3a3a, 0.1); - } - - .emoji-picker__search { - border-color: transparent; - box-shadow: 0 3px 10px rgba(#3a3a3a, 0.1); - } - - .emoji-picker__search:focus { - box-shadow: 0 2px 4px rgba(#3a3a3a, 0.1); - } - - .emoji-picker__emoji:focus, - .emoji-picker__emoji:hover, - .emoji-picker__tab:hover { - background: #eee; - } - - .emoji-picker__tab.active { - background: #bbb; - } - - .emoji-picker__search { - padding-left: 0.6em; - } - - .emoji-picker__tab svg, - .emoji-picker__tab path { - box-sizing: content-box; - } - - .emoji-picker__search-container { - margin: 0.5em 0.5em 0; - } - - .emoji-picker__search-icon { - top: 5px; - } -} diff --git a/webapp/app/styles/html-components/filters.scss b/webapp/app/styles/html-components/filters.scss index 1dd37ea14..75aa58464 100644 --- a/webapp/app/styles/html-components/filters.scss +++ b/webapp/app/styles/html-components/filters.scss @@ -1,19 +1,23 @@ .filters { padding: 15px; - border-radius: 0 0 3px 3px; background: var(--background-light); border-radius: var(--border-radius); border: 1px solid var(--background-light-highlight); margin-top: -2px; + &.filters--jipt { + border-radius: 0; + } + &.filters--transparent { background: none; border-color: transparent; box-shadow: none; + padding: 0; } .ember-power-select-trigger { - padding: 5px 10px; + padding: 5px 27px 5px 5px; background: var(--content-background); box-shadow: none; border: 1px solid transparent; @@ -30,13 +34,28 @@ .filters-wrapper { position: relative; + display: flex; + justify-content: space-between; + align-items: flex-start; +} + +.filters-content { + display: flex; + flex-direction: column; + gap: 10px; + flex-grow: 1; +} + +.queryForm { + position: relative; + display: flex; + gap: 10px; } .queryForm-filters { display: flex; justify-content: space-between; align-items: center; - margin-top: 6px; } .queryForm-filters-column { diff --git a/webapp/app/styles/html-components/power-select.scss b/webapp/app/styles/html-components/power-select.scss index f2e86b76c..4519dd9cf 100644 --- a/webapp/app/styles/html-components/power-select.scss +++ b/webapp/app/styles/html-components/power-select.scss @@ -70,6 +70,65 @@ } } +.ember-power-select-multiple-options + .ember-power-select-multiple-option:first-of-type { + span[role='button'] { + display: none; + } +} + +.ember-power-select-multiple-options { + margin: 0; + padding: 0; + display: flex; + flex-wrap: wrap; + gap: 4px; + align-items: center; + list-style: none; + + & li.ember-power-select-trigger-multiple-input-container { + flex-grow: 1; + display: flex; + + & input { + flex-grow: 1; + } + } +} + +.ember-power-select-trigger-multiple-input { + font-family: inherit; + font-size: inherit; + border: none; + line-height: inherit; + -webkit-appearance: none; + outline: none; + padding: 0; + background-color: transparent; + text-indent: 2px; +} + +html[data-theme='dark'] .ember-power-select-multiple-option { + color: var(--color-primary); +} + +.ember-power-select-multiple-option { + display: inline-block; + background: var(--color-primary-opacity-25); + color: var(--color-primary-darken-50); + border-radius: var(--border-radius); + + padding: 3px 9px; + font-size: 14px; +} + +.ember-power-select-multiple-remove-btn { + cursor: pointer; + &:not(:hover) { + opacity: 0.5; + } +} + .ember-power-select-status-icon { position: absolute; right: 0; diff --git a/webapp/app/pods/application/template.hbs b/webapp/app/templates/application.hbs similarity index 100% rename from webapp/app/pods/application/template.hbs rename to webapp/app/templates/application.hbs diff --git a/webapp/app/pods/components/acc-avatar-img/template.hbs b/webapp/app/templates/components/acc-avatar-img.hbs similarity index 100% rename from webapp/app/pods/components/acc-avatar-img/template.hbs rename to webapp/app/templates/components/acc-avatar-img.hbs diff --git a/webapp/app/pods/components/acc-badge/template.hbs b/webapp/app/templates/components/acc-badge.hbs similarity index 100% rename from webapp/app/pods/components/acc-badge/template.hbs rename to webapp/app/templates/components/acc-badge.hbs diff --git a/webapp/app/templates/components/acc-emoji-picker.hbs b/webapp/app/templates/components/acc-emoji-picker.hbs new file mode 100644 index 000000000..1833a0035 --- /dev/null +++ b/webapp/app/templates/components/acc-emoji-picker.hbs @@ -0,0 +1,16 @@ + + +{{#if this.picker}} + +
    +
    +
    + {{this.picker}} +
    +
    +
    +{{/if}} \ No newline at end of file diff --git a/webapp/app/pods/components/acc-flash-message/template.hbs b/webapp/app/templates/components/acc-flash-message.hbs similarity index 100% rename from webapp/app/pods/components/acc-flash-message/template.hbs rename to webapp/app/templates/components/acc-flash-message.hbs diff --git a/webapp/app/pods/components/acc-modal/template.hbs b/webapp/app/templates/components/acc-modal.hbs similarity index 100% rename from webapp/app/pods/components/acc-modal/template.hbs rename to webapp/app/templates/components/acc-modal.hbs diff --git a/webapp/app/pods/components/acc-select/template.hbs b/webapp/app/templates/components/acc-select.hbs similarity index 80% rename from webapp/app/pods/components/acc-select/template.hbs rename to webapp/app/templates/components/acc-select.hbs index 4ccd83171..88d9851fd 100644 --- a/webapp/app/pods/components/acc-select/template.hbs +++ b/webapp/app/templates/components/acc-select.hbs @@ -17,6 +17,10 @@ {{option.label}} +{{else if @multi}} + + {{option.label}} + {{else}}
    + + {{#if @onChangeAdvancedFilterBoolean}} + + {{/if}} +
    + + {{#if this.showSomeFilters}} +
    +
    + {{#if this.showDocumentsSelect}} +
    +
    + +
    +
    + {{/if}} + + {{#if this.showVersionsSelect}} +
    +
    + +
    +
    + {{/if}} +
    +
    + {{/if}} + + {{#if this.displayAdvancedFilters}} + + {{/if}} +
    + + + \ No newline at end of file diff --git a/webapp/app/templates/components/conflicts-list.hbs b/webapp/app/templates/components/conflicts-list.hbs new file mode 100644 index 000000000..b8503f572 --- /dev/null +++ b/webapp/app/templates/components/conflicts-list.hbs @@ -0,0 +1,47 @@ +{{#if this.currentVersion}} +
    + + {{t 'components.conflicts_list.translations_version_notice'}} + {{this.currentVersion.tag}} +
    +{{/if}} + + + + \ No newline at end of file diff --git a/webapp/app/templates/components/conflicts-list/advanced-filters.hbs b/webapp/app/templates/components/conflicts-list/advanced-filters.hbs new file mode 100644 index 000000000..30947477d --- /dev/null +++ b/webapp/app/templates/components/conflicts-list/advanced-filters.hbs @@ -0,0 +1,42 @@ +
    + + {{t 'components.conflicts_filters.advanced_filters_title'}} + + +
    + + + + + + + + + +
    +
    \ No newline at end of file diff --git a/webapp/app/templates/components/conflicts-list/group.hbs b/webapp/app/templates/components/conflicts-list/group.hbs new file mode 100644 index 000000000..e2d1e4e76 --- /dev/null +++ b/webapp/app/templates/components/conflicts-list/group.hbs @@ -0,0 +1,37 @@ +
  • + + {{this.translationKey.value}} + + + {{#if this.translationKey.prefix}} + {{this.translationKey.prefix}} + {{else}} + {{@groupedTranslation.document.path}} + {{/if}} + + + + +
  • \ No newline at end of file diff --git a/webapp/app/templates/components/conflicts-list/item.hbs b/webapp/app/templates/components/conflicts-list/item.hbs new file mode 100644 index 000000000..8d4da5a35 --- /dev/null +++ b/webapp/app/templates/components/conflicts-list/item.hbs @@ -0,0 +1,77 @@ +
  • +
    +
    + {{#if this.error}} +
    + {{t 'components.translation_item.correct_error_text'}} +
    + {{/if}} +
    +
    +
    + + {{#component form.submit}} +
    + {{#if this.showOriginalButton}} + + {{inline-svg '/assets/revert.svg' class='button-icon'}} + + {{/if}} + +
    + +
    + + {{#if @translation.isConflicted}} + {{#if (get @permissions 'correct_translation')}} + + {{inline-svg '/assets/check.svg' class='button-icon'}} + + {{else if (get @permissions 'update_translation')}} + + {{inline-svg '/assets/pencil.svg' class='button-icon'}} + + {{/if}} + {{else}} + {{#if (get @permissions 'uncorrect_translation')}} + + {{inline-svg '/assets/revert.svg' class='button-icon'}} + + {{else if (get @permissions 'update_translation')}} + + {{inline-svg '/assets/pencil.svg' class='button-icon'}} + + {{/if}} + {{/if}} +
    + {{/component}} +
    +
    +
    +
    +
  • \ No newline at end of file diff --git a/webapp/app/pods/components/dashboard-revisions/template.hbs b/webapp/app/templates/components/dashboard-revisions.hbs similarity index 88% rename from webapp/app/pods/components/dashboard-revisions/template.hbs rename to webapp/app/templates/components/dashboard-revisions.hbs index 97566eb37..ea1a58c8f 100644 --- a/webapp/app/pods/components/dashboard-revisions/template.hbs +++ b/webapp/app/templates/components/dashboard-revisions.hbs @@ -30,7 +30,7 @@

    {{#if (get @permissions 'sync')}} - + {{inline-svg '/assets/sync.svg' class='button-icon'}} {{t 'components.documents_list.sync'}} @@ -59,17 +59,6 @@
    {{#if this.slaveRevisions}} -

    -
    - {{#if (get @permissions 'create_slave')}} - - {{inline-svg 'assets/language.svg' class='button-icon'}} - {{t 'components.dashboard_revisions.manage_languages_link_title'}} - - {{/if}} -
    -

    -
    {{#each this.slaveRevisions key='id' as |revision|}} - {{#if (get @permissions 'index_translations')}} - - {{this.languageName}} - {{#if this.rtl}} - - {{t 'components.dashboard_revisions.item.rtl_badge'}} - - {{/if}} - - {{this.correctedKeysPercentage}} - % - - - {{else}} - - {{this.languageName}} - - {{this.correctedKeysPercentage}} - % - - - {{/if}} + + {{this.languageName}} + {{#if this.rtl}} + + {{t 'components.dashboard_revisions.item.rtl_badge'}} + + {{/if}} + - - {{this.reviewsCount}} - - / - - {{@revision.translationsCount}} + + {{this.correctedKeysPercentage}}% + +
    + + {{this.toReviewCount}} + {{t 'components.dashboard_revisions.item.stats_to_review'}} + + + {{#unless @revision.isMaster}} + + {{@revision.translatedCount}} + {{t 'components.dashboard_revisions.item.stats_translated'}} + + {{/unless}} +
    diff --git a/webapp/app/templates/components/date-tag.hbs b/webapp/app/templates/components/date-tag.hbs new file mode 100644 index 000000000..00744688c --- /dev/null +++ b/webapp/app/templates/components/date-tag.hbs @@ -0,0 +1,5 @@ + + + \ No newline at end of file diff --git a/webapp/app/pods/components/documents-add-button/template.hbs b/webapp/app/templates/components/documents-add-button.hbs similarity index 100% rename from webapp/app/pods/components/documents-add-button/template.hbs rename to webapp/app/templates/components/documents-add-button.hbs diff --git a/webapp/app/pods/components/documents-list/template.hbs b/webapp/app/templates/components/documents-list.hbs similarity index 100% rename from webapp/app/pods/components/documents-list/template.hbs rename to webapp/app/templates/components/documents-list.hbs diff --git a/webapp/app/pods/components/documents-list/item/template.hbs b/webapp/app/templates/components/documents-list/item.hbs similarity index 78% rename from webapp/app/pods/components/documents-list/item/template.hbs rename to webapp/app/templates/components/documents-list/item.hbs index 2c9ec0ed2..b8a8bc98f 100644 --- a/webapp/app/pods/components/documents-list/item/template.hbs +++ b/webapp/app/templates/components/documents-list/item.hbs @@ -81,45 +81,28 @@ {{#unless this.empty}}
    {{#if (get @permissions 'sync')}} - + {{inline-svg '/assets/sync.svg' class='button-icon'}} + {{t 'components.documents_list.sync'}} {{/if}} {{#if this.multipleRevisions}} {{#if (get @permissions 'merge')}} - + {{inline-svg '/assets/merge.svg' class='button-icon'}} + {{t 'components.documents_list.merge'}} {{/if}} {{/if}} {{#if (get @permissions 'machine_translations_translate')}} - + {{inline-svg '/assets/language.svg' class='button-icon'}} + {{t 'components.documents_list.machine_translations'}} {{/if}} - + {{inline-svg '/assets/export.svg' class='button-icon'}} + {{t 'components.documents_list.export'}}
    diff --git a/webapp/app/pods/components/documents-machine-translations-button/template.hbs b/webapp/app/templates/components/documents-machine-translations-button.hbs similarity index 100% rename from webapp/app/pods/components/documents-machine-translations-button/template.hbs rename to webapp/app/templates/components/documents-machine-translations-button.hbs diff --git a/webapp/app/pods/components/dummy-login-form/template.hbs b/webapp/app/templates/components/dummy-login-form.hbs similarity index 100% rename from webapp/app/pods/components/dummy-login-form/template.hbs rename to webapp/app/templates/components/dummy-login-form.hbs diff --git a/webapp/app/pods/components/empty-content/template.hbs b/webapp/app/templates/components/empty-content.hbs similarity index 100% rename from webapp/app/pods/components/empty-content/template.hbs rename to webapp/app/templates/components/empty-content.hbs diff --git a/webapp/app/pods/components/error-section/template.hbs b/webapp/app/templates/components/error-section.hbs similarity index 100% rename from webapp/app/pods/components/error-section/template.hbs rename to webapp/app/templates/components/error-section.hbs diff --git a/webapp/app/pods/components/file-export-all/template.hbs b/webapp/app/templates/components/file-export-all.hbs similarity index 100% rename from webapp/app/pods/components/file-export-all/template.hbs rename to webapp/app/templates/components/file-export-all.hbs diff --git a/webapp/app/pods/components/file-export/template.hbs b/webapp/app/templates/components/file-export.hbs similarity index 100% rename from webapp/app/pods/components/file-export/template.hbs rename to webapp/app/templates/components/file-export.hbs diff --git a/webapp/app/pods/components/file-input/template.hbs b/webapp/app/templates/components/file-input.hbs similarity index 100% rename from webapp/app/pods/components/file-input/template.hbs rename to webapp/app/templates/components/file-input.hbs diff --git a/webapp/app/pods/components/flash-messages-list/template.hbs b/webapp/app/templates/components/flash-messages-list.hbs similarity index 100% rename from webapp/app/pods/components/flash-messages-list/template.hbs rename to webapp/app/templates/components/flash-messages-list.hbs diff --git a/webapp/app/pods/components/google-login-form/template.hbs b/webapp/app/templates/components/google-login-form.hbs similarity index 100% rename from webapp/app/pods/components/google-login-form/template.hbs rename to webapp/app/templates/components/google-login-form.hbs diff --git a/webapp/app/pods/components/highlight-render/template.hbs b/webapp/app/templates/components/highlight-render.hbs similarity index 100% rename from webapp/app/pods/components/highlight-render/template.hbs rename to webapp/app/templates/components/highlight-render.hbs diff --git a/webapp/app/pods/components/html-textarea/template.hbs b/webapp/app/templates/components/html-textarea.hbs similarity index 100% rename from webapp/app/pods/components/html-textarea/template.hbs rename to webapp/app/templates/components/html-textarea.hbs diff --git a/webapp/app/pods/components/improve-prompt/template.hbs b/webapp/app/templates/components/improve-prompt.hbs similarity index 100% rename from webapp/app/pods/components/improve-prompt/template.hbs rename to webapp/app/templates/components/improve-prompt.hbs diff --git a/webapp/app/templates/components/inline-machine-translate.hbs b/webapp/app/templates/components/inline-machine-translate.hbs new file mode 100644 index 000000000..bf5e9c1ae --- /dev/null +++ b/webapp/app/templates/components/inline-machine-translate.hbs @@ -0,0 +1,9 @@ + + {{inline-svg '/assets/language.svg' class='button-icon'}} + \ No newline at end of file diff --git a/webapp/app/pods/components/jipt-back-to-translations/template.hbs b/webapp/app/templates/components/jipt-back-to-translations.hbs similarity index 100% rename from webapp/app/pods/components/jipt-back-to-translations/template.hbs rename to webapp/app/templates/components/jipt-back-to-translations.hbs diff --git a/webapp/app/templates/components/jipt-example.hbs b/webapp/app/templates/components/jipt-example.hbs new file mode 100644 index 000000000..92723f816 --- /dev/null +++ b/webapp/app/templates/components/jipt-example.hbs @@ -0,0 +1,60 @@ +
    +
    + + ← Back to project + + +
    + {{#unless @loading}} +
    + <head> +
    + + + + +
    <script>
    +          {{{this.scriptContent}}}
    +          </script> <script async src="{{this.scriptSrc}}"></script>
    +        
    + +
    + <body> +
    + +

    {{@project.name}}’s Just-in-place translations

    + +

    Node inner HTML

    + +
    + <p>{^key@document_path}</p> + {{this.translationKey}} +
    + +
    + <strong>{^key@document_path}</strong> + {{this.translationKey}} +
    + +

    Node attributes

    + +
    + <input type="text" placeholder="{^key@document_path}" /> + +
    + +
    + <input type="submit" value="{^key@document_path}" /> + +
    + +
    + <img title="{^key@document_path}" src="…"/> + +
    + {{/unless}} +
    +
    +
    \ No newline at end of file diff --git a/webapp/app/pods/components/jipt-export/template.hbs b/webapp/app/templates/components/jipt-export.hbs similarity index 100% rename from webapp/app/pods/components/jipt-export/template.hbs rename to webapp/app/templates/components/jipt-export.hbs diff --git a/webapp/app/pods/components/jipt-header/template.hbs b/webapp/app/templates/components/jipt-header.hbs similarity index 100% rename from webapp/app/pods/components/jipt-header/template.hbs rename to webapp/app/templates/components/jipt-header.hbs diff --git a/webapp/app/pods/components/jipt-translation/template.hbs b/webapp/app/templates/components/jipt-translation.hbs similarity index 100% rename from webapp/app/pods/components/jipt-translation/template.hbs rename to webapp/app/templates/components/jipt-translation.hbs diff --git a/webapp/app/pods/components/jipt-translations-filtered-title/template.hbs b/webapp/app/templates/components/jipt-translations-filtered-title.hbs similarity index 100% rename from webapp/app/pods/components/jipt-translations-filtered-title/template.hbs rename to webapp/app/templates/components/jipt-translations-filtered-title.hbs diff --git a/webapp/app/pods/components/jipt-translations-list/template.hbs b/webapp/app/templates/components/jipt-translations-list.hbs similarity index 100% rename from webapp/app/pods/components/jipt-translations-list/template.hbs rename to webapp/app/templates/components/jipt-translations-list.hbs diff --git a/webapp/app/pods/components/jipt-translations-list/item/template.hbs b/webapp/app/templates/components/jipt-translations-list/item.hbs similarity index 51% rename from webapp/app/pods/components/jipt-translations-list/item/template.hbs rename to webapp/app/templates/components/jipt-translations-list/item.hbs index 35cb528ed..9f405cf0d 100644 --- a/webapp/app/pods/components/jipt-translations-list/item/template.hbs +++ b/webapp/app/templates/components/jipt-translations-list/item.hbs @@ -1,5 +1,8 @@ {{#if this.isTextEmpty}} {{t 'components.translations_list.empty_text'}} {{else}} - {{@translation.correctedText}} + + {{@translation.correctedText}} + {{@translation.key}} + {{/if}} \ No newline at end of file diff --git a/webapp/app/templates/components/lint-options.hbs b/webapp/app/templates/components/lint-options.hbs new file mode 100644 index 000000000..9da5d72a2 --- /dev/null +++ b/webapp/app/templates/components/lint-options.hbs @@ -0,0 +1,40 @@ +
    +
    +
    +
    +
    + {{inline-svg '/assets/search.svg' local-class='search-icon'}} + + +
    + + {{#if this.showSomeFilters}} +
    + {{#if this.showDocuments}} +
    +
    + +
    +
    + {{/if}} + + {{#if this.showVersions}} +
    +
    + +
    +
    + {{/if}} +
    + {{/if}} +
    +
    +
    +
    \ No newline at end of file diff --git a/webapp/app/pods/components/lint-translations-page/template.hbs b/webapp/app/templates/components/lint-translations-page.hbs similarity index 80% rename from webapp/app/pods/components/lint-translations-page/template.hbs rename to webapp/app/templates/components/lint-translations-page.hbs index e41722bbc..f1263429e 100644 --- a/webapp/app/pods/components/lint-translations-page/template.hbs +++ b/webapp/app/templates/components/lint-translations-page.hbs @@ -15,8 +15,8 @@
    {{#each @lintTranslations key='id' as |lintTranslation|}} -
    - +
    +
    {{/each}}
    diff --git a/webapp/app/templates/components/lint-translations-page/item.hbs b/webapp/app/templates/components/lint-translations-page/item.hbs new file mode 100644 index 000000000..dfcc94931 --- /dev/null +++ b/webapp/app/templates/components/lint-translations-page/item.hbs @@ -0,0 +1,62 @@ +{{#if @lintTranslation.messages}} +
    + + + {{#if @project}} +
    + + + {{this.translationKey.value}} + + {{#if this.translationKey.prefix}} + {{this.translationKey.prefix}} + {{else}} + {{@lintTranslation.translation.document.path}} + {{/if}} + + + +
    + {{/if}} + + {{#if this.allReplacable}} + + {{else}} +
    {{{this.annotatedText}}}
    + {{/if}} +
    +{{/if}} \ No newline at end of file diff --git a/webapp/app/pods/components/loading-content/template.hbs b/webapp/app/templates/components/loading-content.hbs similarity index 100% rename from webapp/app/pods/components/loading-content/template.hbs rename to webapp/app/templates/components/loading-content.hbs diff --git a/webapp/app/pods/components/login-forms/template.hbs b/webapp/app/templates/components/login-forms.hbs similarity index 92% rename from webapp/app/pods/components/login-forms/template.hbs rename to webapp/app/templates/components/login-forms.hbs index 5a2d0d9a0..72cc21402 100644 --- a/webapp/app/pods/components/login-forms/template.hbs +++ b/webapp/app/templates/components/login-forms.hbs @@ -92,6 +92,13 @@ {{t 'components.login_forms.auth0'}} {{/if}} + + {{#if this.oidcLoginEnabled}} + + + {{t 'components.login_forms.oidc'}} + + {{/if}} {{/if}} diff --git a/webapp/app/pods/components/machine-translations-document-translate/template.hbs b/webapp/app/templates/components/machine-translations-document-translate.hbs similarity index 100% rename from webapp/app/pods/components/machine-translations-document-translate/template.hbs rename to webapp/app/templates/components/machine-translations-document-translate.hbs diff --git a/webapp/app/pods/components/machine-translations-translate-upload-form/template.hbs b/webapp/app/templates/components/machine-translations-translate-upload-form.hbs similarity index 100% rename from webapp/app/pods/components/machine-translations-translate-upload-form/template.hbs rename to webapp/app/templates/components/machine-translations-translate-upload-form.hbs diff --git a/webapp/app/pods/components/operations-peek/template.hbs b/webapp/app/templates/components/operations-peek.hbs similarity index 100% rename from webapp/app/pods/components/operations-peek/template.hbs rename to webapp/app/templates/components/operations-peek.hbs diff --git a/webapp/app/pods/components/operations-peek/item/template.hbs b/webapp/app/templates/components/operations-peek/item.hbs similarity index 100% rename from webapp/app/pods/components/operations-peek/item/template.hbs rename to webapp/app/templates/components/operations-peek/item.hbs diff --git a/webapp/app/pods/components/page-title/end/template.hbs b/webapp/app/templates/components/page-title.hbs similarity index 100% rename from webapp/app/pods/components/page-title/end/template.hbs rename to webapp/app/templates/components/page-title.hbs diff --git a/webapp/app/pods/components/page-title/template.hbs b/webapp/app/templates/components/page-title/end.hbs similarity index 100% rename from webapp/app/pods/components/page-title/template.hbs rename to webapp/app/templates/components/page-title/end.hbs diff --git a/webapp/app/pods/components/phoenix-channel-listener/template.hbs b/webapp/app/templates/components/phoenix-channel-listener.hbs similarity index 100% rename from webapp/app/pods/components/phoenix-channel-listener/template.hbs rename to webapp/app/templates/components/phoenix-channel-listener.hbs diff --git a/webapp/app/pods/components/project-activities-filter/template.hbs b/webapp/app/templates/components/project-activities-filter.hbs similarity index 100% rename from webapp/app/pods/components/project-activities-filter/template.hbs rename to webapp/app/templates/components/project-activities-filter.hbs diff --git a/webapp/app/pods/components/project-activities-list/template.hbs b/webapp/app/templates/components/project-activities-list.hbs similarity index 100% rename from webapp/app/pods/components/project-activities-list/template.hbs rename to webapp/app/templates/components/project-activities-list.hbs diff --git a/webapp/app/pods/components/project-activity/template.hbs b/webapp/app/templates/components/project-activity.hbs similarity index 94% rename from webapp/app/pods/components/project-activity/template.hbs rename to webapp/app/templates/components/project-activity.hbs index 53f6fd1ab..b4bdf2bae 100644 --- a/webapp/app/pods/components/project-activity/template.hbs +++ b/webapp/app/templates/components/project-activity.hbs @@ -208,18 +208,16 @@ {{t 'components.project_activity.operations_label'}} -
    - {{#each this.operations.entries key='id' as |activity|}} - - {{/each}} -
    + {{#each this.operations.entries key='id' as |activity|}} + + {{/each}} {{#if this.operations.meta.nextPage}}
    diff --git a/webapp/app/pods/components/project-comments-list/template.hbs b/webapp/app/templates/components/project-comments-list.hbs similarity index 100% rename from webapp/app/pods/components/project-comments-list/template.hbs rename to webapp/app/templates/components/project-comments-list.hbs diff --git a/webapp/app/pods/components/project-comments-list/item/template.hbs b/webapp/app/templates/components/project-comments-list/item.hbs similarity index 100% rename from webapp/app/pods/components/project-comments-list/item/template.hbs rename to webapp/app/templates/components/project-comments-list/item.hbs diff --git a/webapp/app/pods/components/project-create-form/template.hbs b/webapp/app/templates/components/project-create-form.hbs similarity index 100% rename from webapp/app/pods/components/project-create-form/template.hbs rename to webapp/app/templates/components/project-create-form.hbs diff --git a/webapp/app/pods/components/project-file-operation/template.hbs b/webapp/app/templates/components/project-file-operation.hbs similarity index 100% rename from webapp/app/pods/components/project-file-operation/template.hbs rename to webapp/app/templates/components/project-file-operation.hbs diff --git a/webapp/app/pods/components/project-header/template.hbs b/webapp/app/templates/components/project-header.hbs similarity index 100% rename from webapp/app/pods/components/project-header/template.hbs rename to webapp/app/templates/components/project-header.hbs diff --git a/webapp/app/pods/components/project-logo/template.hbs b/webapp/app/templates/components/project-logo.hbs similarity index 100% rename from webapp/app/pods/components/project-logo/template.hbs rename to webapp/app/templates/components/project-logo.hbs diff --git a/webapp/app/pods/components/project-navigation/template.hbs b/webapp/app/templates/components/project-navigation.hbs similarity index 100% rename from webapp/app/pods/components/project-navigation/template.hbs rename to webapp/app/templates/components/project-navigation.hbs diff --git a/webapp/app/pods/components/project-navigation/list/template.hbs b/webapp/app/templates/components/project-navigation/list.hbs similarity index 90% rename from webapp/app/pods/components/project-navigation/list/template.hbs rename to webapp/app/templates/components/project-navigation/list.hbs index c7e588777..5e9de0412 100644 --- a/webapp/app/pods/components/project-navigation/list/template.hbs +++ b/webapp/app/templates/components/project-navigation/list.hbs @@ -26,13 +26,22 @@ {{#if (get @permissions 'correct_translation')}}
  • - + {{inline-svg '/assets/check.svg' local-class='list-item-link-icon'}} {{t 'components.project_navigation.conflicts_link_title'}}
  • + {{else if (get @permissions 'update_translation')}} +
  • + + {{inline-svg '/assets/pencil.svg' local-class='list-item-link-icon'}} + + {{t 'components.project_navigation.translate_link_title'}} + + +
  • {{/if}} {{#if (get @permissions 'lint')}} diff --git a/webapp/app/pods/components/project-settings/api-token/template.hbs b/webapp/app/templates/components/project-settings/api-token.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/api-token/template.hbs rename to webapp/app/templates/components/project-settings/api-token.hbs diff --git a/webapp/app/pods/components/project-settings/api-token/item/template.hbs b/webapp/app/templates/components/project-settings/api-token/item.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/api-token/item/template.hbs rename to webapp/app/templates/components/project-settings/api-token/item.hbs diff --git a/webapp/app/pods/components/project-settings/back-link/template.hbs b/webapp/app/templates/components/project-settings/back-link.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/back-link/template.hbs rename to webapp/app/templates/components/project-settings/back-link.hbs diff --git a/webapp/app/pods/components/project-settings/badges/template.hbs b/webapp/app/templates/components/project-settings/badges.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/badges/template.hbs rename to webapp/app/templates/components/project-settings/badges.hbs diff --git a/webapp/app/pods/components/project-settings/collaborators/template.hbs b/webapp/app/templates/components/project-settings/collaborators.hbs similarity index 72% rename from webapp/app/pods/components/project-settings/collaborators/template.hbs rename to webapp/app/templates/components/project-settings/collaborators.hbs index d0de5f7d1..547d72591 100644 --- a/webapp/app/pods/components/project-settings/collaborators/template.hbs +++ b/webapp/app/templates/components/project-settings/collaborators.hbs @@ -17,6 +17,7 @@
    + {{inline-svg 'assets/tool.svg' local-class='rolesList-icon'}} {{t 'general.roles.OWNER'}}

    @@ -24,6 +25,7 @@

    + {{inline-svg 'assets/users.svg' local-class='rolesList-icon'}} {{t 'general.roles.ADMIN'}}

    @@ -31,6 +33,7 @@

    + {{inline-svg 'assets/code.svg' local-class='rolesList-icon'}} {{t 'general.roles.DEVELOPER'}}

    @@ -38,11 +41,20 @@

    + {{inline-svg 'assets/check.svg' local-class='rolesList-icon'}} {{t 'general.roles.REVIEWER'}}

    {{t 'components.project_settings.collaborators.reviewer_text'}}

    + + + {{inline-svg 'assets/machine-translations.svg' local-class='rolesList-icon'}} + {{t 'general.roles.TRANSLATOR'}} + +

    + {{t 'components.project_settings.collaborators.translator_text'}} +

    \ No newline at end of file diff --git a/webapp/app/pods/components/project-settings/collaborators/create-form/template.hbs b/webapp/app/templates/components/project-settings/collaborators/create-form.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/create-form/template.hbs rename to webapp/app/templates/components/project-settings/collaborators/create-form.hbs diff --git a/webapp/app/pods/components/project-settings/collaborators/list/template.hbs b/webapp/app/templates/components/project-settings/collaborators/list.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/list/template.hbs rename to webapp/app/templates/components/project-settings/collaborators/list.hbs diff --git a/webapp/app/pods/components/project-settings/collaborators/list/item/template.hbs b/webapp/app/templates/components/project-settings/collaborators/list/item.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/collaborators/list/item/template.hbs rename to webapp/app/templates/components/project-settings/collaborators/list/item.hbs diff --git a/webapp/app/pods/components/project-settings/delete-form/template.hbs b/webapp/app/templates/components/project-settings/delete-form.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/delete-form/template.hbs rename to webapp/app/templates/components/project-settings/delete-form.hbs diff --git a/webapp/app/pods/components/project-settings/form/template.hbs b/webapp/app/templates/components/project-settings/form.hbs similarity index 100% rename from webapp/app/pods/components/project-settings/form/template.hbs rename to webapp/app/templates/components/project-settings/form.hbs diff --git a/webapp/app/pods/components/project-settings/integrations/template.hbs b/webapp/app/templates/components/project-settings/integrations.hbs similarity index 86% rename from webapp/app/pods/components/project-settings/integrations/template.hbs rename to webapp/app/templates/components/project-settings/integrations.hbs index a7ac62842..917847c8e 100644 --- a/webapp/app/pods/components/project-settings/integrations/template.hbs +++ b/webapp/app/templates/components/project-settings/integrations.hbs @@ -38,6 +38,12 @@

    {{t 'components.project_settings.integrations.empty_description.azure_storage_container.text'}}

    + +