diff --git a/.dockerignore b/.dockerignore index 065d2a7e6..a332c2ee7 100644 --- a/.dockerignore +++ b/.dockerignore @@ -1,3 +1,4 @@ +**/__pycache__ /src-ui/.vscode /src-ui/node_modules /src-ui/dist @@ -10,3 +11,9 @@ .pytest_cache /dist /scripts +/resources +**/tests +**/*.spec.ts +**/htmlcov +/src/.pytest_cache +.idea diff --git a/.env b/.env index 511a1386d..0bd5be6ff 100644 --- a/.env +++ b/.env @@ -1 +1 @@ -COMPOSE_PROJECT_NAME=paperless \ No newline at end of file +COMPOSE_PROJECT_NAME=paperless diff --git a/.github/ISSUE_TEMPLATE/bug_report.md b/.github/ISSUE_TEMPLATE/bug_report.md new file mode 100644 index 000000000..9a4b9bc2f --- /dev/null +++ b/.github/ISSUE_TEMPLATE/bug_report.md @@ -0,0 +1,48 @@ +--- +name: Bug report +about: Something is not working +title: "[BUG] Concise description of the issue" +labels: '' +assignees: '' + +--- + + + +**Describe the bug** +A clear and concise description of what the bug is. + +**To Reproduce** +Steps to reproduce the behavior: +1. Go to '...' +2. Click on '....' +3. Scroll down to '....' +4. See error + +**Expected behavior** +A clear and concise description of what you expected to happen. + +**Screenshots** +If applicable, add screenshots to help explain your problem. + +**Webserver logs** +``` +If available, post any logs from the web server related to your issue. +``` + +**Relevant information** + - Host OS of the machine running paperless: [e.g. Archlinux / Ubuntu 20.04] + - Browser [e.g. chrome, safari] + - Version [e.g. 1.0.0] + - Installation method: [docker / bare metal] + - Any configuration changes you made in `docker-compose.yml`, `docker-compose.env` or `paperless.conf`. diff --git a/.github/ISSUE_TEMPLATE/other.md b/.github/ISSUE_TEMPLATE/other.md new file mode 100644 index 000000000..8bb40d9d1 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/other.md @@ -0,0 +1,20 @@ +--- +name: Other +about: Anything that is not a feature request or bug. +title: "[Other] Title of your issue" +labels: '' +assignees: '' + +--- + + diff --git a/.github/workflow-scripts/check-trailing-newline b/.github/workflow-scripts/check-trailing-newline new file mode 100755 index 000000000..6973a5e3e --- /dev/null +++ b/.github/workflow-scripts/check-trailing-newline @@ -0,0 +1,37 @@ +#!/bin/bash + +# Verify that all text files end in a trailing newline. + +# Exit on first failing command. +set -e + +# Exit on unset variable. +set -u + +success=0 + +function is_plaintext_file() { + local file="$1" + if [[ $file == *.svg ]]; then + echo "" + return + fi + file --brief "${file}" | grep text +} + +# Split strings on newlines. +IFS=' +' +for file in $(git ls-files) +do + if [[ -z $(is_plaintext_file "${file}") ]]; then + continue + fi + + if ! [[ -z "$(tail -c 1 "${file}")" ]]; then + printf "File must end in a trailing newline: %s\n" "${file}" >&2 + success=255 + fi +done + +exit "${success}" diff --git a/.github/workflow-scripts/check-trailing-whitespace b/.github/workflow-scripts/check-trailing-whitespace new file mode 100755 index 000000000..fe7a2bdd6 --- /dev/null +++ b/.github/workflow-scripts/check-trailing-whitespace @@ -0,0 +1,26 @@ +#!/bin/bash + +# Check for trailing whitespace at end of lines. + +# Exit on first failing command. +set -e +# Exit on unset variable. +set -u + +FOUND_TRAILING_WHITESPACE=0 + +while read -r line; do + if grep \ + "\s$" \ + --line-number \ + --with-filename \ + --binary-files=without-match \ + --exclude="*.svg" \ + --exclude="*.eps" \ + "${line}"; then + echo "ERROR: Found trailing whitespace" >&2; + FOUND_TRAILING_WHITESPACE=1 + fi +done < <(git ls-files) + +exit "${FOUND_TRAILING_WHITESPACE}" diff --git a/.github/workflows/ansible.yml b/.github/workflows/ansible.yml new file mode 100644 index 000000000..7467806c9 --- /dev/null +++ b/.github/workflows/ansible.yml @@ -0,0 +1,78 @@ +--- +name: Ansible Role + +on: + push: + branches-ignore: + - 'translations**' + pull_request: + branches-ignore: + - 'translations**' + +jobs: + # https://molecule.readthedocs.io/en/latest/ci.html#github-actions + test: + runs-on: ubuntu-latest + steps: + - name: Check out the codebase + uses: actions/checkout@v2 + if: github.event_name != 'pull_request' + with: + path: "${{ github.repository }}" + - name: Check out the codebase + uses: actions/checkout@v2 + if: github.event_name == 'pull_request' + with: + # merge commit is not available from tree at this point in time + # https://github.com/actions/checkout#checkout-pull-request-head-commit-instead-of-merge-commit + ref: "${{ github.event.pull_request.head.sha }}" + path: "${{ github.repository }}" + - name: Set up Python + uses: actions/setup-python@v2 + - name: Set up Docker + uses: docker-practice/actions-setup-docker@master + - name: Install dependencies + run: | + python3 -m pip install --upgrade pip + python3 -m pip install molecule[ansible,docker] jmespath + ansible --version + docker --version + molecule --version + python --version + - name: Test installation/build/upgrade with molecule + run: | + cd ansible + molecule create + molecule verify + molecule converge + molecule idempotence + molecule verify + working-directory: "${{ github.repository }}" + env: + TARGET_GITHUB_SHA: "${{ github.event.pull_request.head.sha }}" + # # https://galaxy.ansible.com/docs/contributing/importing.html + # release: + # runs-on: ubuntu-latest + # needs: + # - test + # # https://docs.github.com/en/free-pro-team@latest/actions/reference/context-and-expression-syntax-for-github-actions#github-context + # if: contains(github.ref, 'refs/tags/') + # steps: + # - name: Check out the codebase + # uses: actions/checkout@v2 + # with: + # path: "${{ github.repository }}" + # - name: Set up Python + # uses: actions/setup-python@v2 + # - name: Install dependencies + # run: | + # python3 -m pip install --upgrade ansible-base + # ansible --version + # python --version + # - name: Trigger a new import on Galaxy + # # TODO Check if source if pulled from cwd or imported from github + # # https://github.com/ansible/ansible/blob/devel/lib/ansible/cli/galaxy.py + # run: | + # cd ansible + # ansible-galaxy role import --api-key ${{ secrets.GALAXY_API_KEY }} $(echo ${{ github.repository }} | cut -d/ -f1) $(echo ${{ github.repository }} | cut -d/ -f2) + # working-directory: "${{ github.repository }}" diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml new file mode 100644 index 000000000..8e297cda2 --- /dev/null +++ b/.github/workflows/ci.yml @@ -0,0 +1,346 @@ +name: ci + +on: + push: + tags: ng-* + branches-ignore: + - 'translations**' + pull_request: + branches-ignore: + - 'translations**' + +jobs: + documentation: + runs-on: ubuntu-20.04 + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: 3.9 + - + name: Get pip cache dir + id: pip-cache + run: | + echo "::set-output name=dir::$(pip cache dir)" + - + name: Persistent Github pip cache + uses: actions/cache@v2 + with: + path: ${{ steps.pip-cache.outputs.dir }} + key: ${{ runner.os }}-pip3.8} + - + name: Install dependencies + run: | + pip install --upgrade pipenv + pipenv install --system --dev --ignore-pipfile + - + name: Make documentation + run: | + cd docs/ + make html + - + name: Upload artifact + uses: actions/upload-artifact@v2 + with: + name: documentation + path: docs/_build/html/ + + codestyle: + runs-on: ubuntu-20.04 + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: 3.9 + - + name: Get pip cache dir + id: pip-cache + run: | + echo "::set-output name=dir::$(pip cache dir)" + - + name: Persistent Github pip cache + uses: actions/cache@v2 + with: + path: ${{ steps.pip-cache.outputs.dir }} + key: ${{ runner.os }}-pip${{ matrix.python-version }} + - + name: Install dependencies + run: | + pip install --upgrade pipenv + pipenv install --system --dev --ignore-pipfile + - + name: Codestyle + run: | + cd src/ + pycodestyle + whitespace: + runs-on: ubuntu-20.04 + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Ensure there are no trailing spaces + run: | + .github/workflow-scripts/check-trailing-whitespace + - + name: Ensure all text files end with a trailing newline + run: | + .github/workflow-scripts/check-trailing-whitespace + + tests: + runs-on: ubuntu-20.04 + strategy: + matrix: + python-version: ['3.7', '3.8', '3.9'] + fail-fast: false + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: "${{ matrix.python-version }}" + - + name: Get pip cache dir + id: pip-cache + run: | + echo "::set-output name=dir::$(pip cache dir)" + - + name: Persistent Github pip cache + uses: actions/cache@v2 + with: + path: ${{ steps.pip-cache.outputs.dir }} + key: ${{ runner.os }}-pip${{ matrix.python-version }} + - + name: Install dependencies + run: | + sudo apt-get update -qq + sudo apt-get install -qq --no-install-recommends unpaper tesseract-ocr imagemagick ghostscript optipng + pip install --upgrade pipenv + pipenv install --system --dev --ignore-pipfile + - + name: Tests + run: | + cd src/ + pytest + - + name: Publish coverage results + if: matrix.python-version == '3.9' + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + # https://github.com/coveralls-clients/coveralls-python/issues/251 + run: | + cd src/ + coveralls --service=github + + frontend: + runs-on: ubuntu-20.04 + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + uses: actions/setup-node@v2 + with: + node-version: '15' + - + name: Configure version on dev branches + if: startsWith(github.ref, 'refs/heads/feature-') || github.ref == 'refs/heads/dev' + run: | + git_hash=$(git rev-parse --short "$GITHUB_SHA") + git_branch=${GITHUB_REF#refs/heads/} + sed -i -E "s/version: \"(.*)\"/version: \"${git_branch} ${git_hash}\"/g" src-ui/src/environments/environment.prod.ts + - + name: Build frontend + run: ./compile-frontend.sh + - + name: Upload artifact + uses: actions/upload-artifact@v2 + with: + name: frontend-compiled + path: src/documents/static/frontend/ + + build-release: + needs: [frontend, documentation, tests, whitespace, codestyle] + runs-on: ubuntu-20.04 + steps: + - + name: Checkout + uses: actions/checkout@v2 + - + name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: 3.9 + - + name: Install dependencies + run: | + sudo apt-get update -qq + sudo apt-get install -qq --no-install-recommends gettext liblept5 + pip3 install -r requirements.txt + - + name: Download frontend artifact + uses: actions/download-artifact@v2 + with: + name: frontend-compiled + path: src/documents/static/frontend/ + - + name: Download documentation artifact + uses: actions/download-artifact@v2 + with: + name: documentation + path: docs/_build/html/ + - + name: Move files + run: | + mkdir dist + mkdir dist/paperless-ng + mkdir dist/paperless-ng/scripts + cp .dockerignore .env Dockerfile Pipfile Pipfile.lock LICENSE README.md requirements.txt dist/paperless-ng/ + cp paperless.conf.example dist/paperless-ng/paperless.conf + cp gunicorn.conf.py dist/paperless-ng/gunicorn.conf.py + cp docker/ dist/paperless-ng/docker -r + cp scripts/*.service scripts/*.sh dist/paperless-ng/scripts/ + cp src/ dist/paperless-ng/src -r + cp docs/_build/html/ dist/paperless-ng/docs -r + - + name: Compile messages + run: | + cd dist/paperless-ng/src + python3 manage.py compilemessages + - + name: Collect static files + run: | + cd dist/paperless-ng/src + python3 manage.py collectstatic --no-input + - + name: Make release package + run: | + cd dist + find . -name __pycache__ | xargs rm -r + tar -cJf paperless-ng.tar.xz paperless-ng/ + - + name: Upload release artifact + uses: actions/upload-artifact@v2 + with: + name: release + path: dist/paperless-ng.tar.xz + + publish-release: + runs-on: ubuntu-latest + needs: build-release + if: contains(github.ref, 'refs/tags/ng-') + steps: + - + name: Download release artifact + uses: actions/download-artifact@v2 + with: + name: release + path: ./ + - + name: Get version + id: get_version + run: | + echo ::set-output name=version::${GITHUB_REF#refs/tags/ng-} + - + name: Create release + id: create_release + uses: actions/create-release@v1 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + with: + tag_name: ng-${{ steps.get_version.outputs.version }} + release_name: Paperless-ng ${{ steps.get_version.outputs.version }} + draft: false + prerelease: false + body: | + For a complete list of changes, see the changelog at https://paperless-ng.readthedocs.io/en/latest/changelog.html. + - + name: Upload release archive + id: upload-release-asset + uses: actions/upload-release-asset@v1 + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + with: + upload_url: ${{ steps.create_release.outputs.upload_url }} # This pulls from the CREATE RELEASE step above, referencing it's ID to get its outputs object, which include a `upload_url`. See this blog post for more info: https://jasonet.co/posts/new-features-of-github-actions/#passing-data-to-future-steps + asset_path: ./paperless-ng.tar.xz + asset_name: paperless-ng-${{ steps.get_version.outputs.version }}.tar.xz + asset_content_type: application/x-xz + + # build and push image to docker hub. + build-docker-image: + if: github.event_name == 'push' && (startsWith(github.ref, 'refs/heads/feature-') || github.ref == 'refs/heads/dev' || startsWith(github.ref, 'refs/tags/ng-')) + runs-on: ubuntu-latest + needs: [frontend, tests, whitespace, codestyle] + steps: + - + name: Prepare + id: prepare + run: | + IMAGE_NAME=jonaswinkler/paperless-ng + if [[ $GITHUB_REF == refs/tags/ng-* ]]; then + TAGS=${IMAGE_NAME}:${GITHUB_REF#refs/tags/ng-},${IMAGE_NAME}:latest + INSPECT_TAG=${IMAGE_NAME}:latest + elif [[ $GITHUB_REF == refs/heads/* ]]; then + TAGS=${IMAGE_NAME}:${GITHUB_REF#refs/heads/} + INSPECT_TAG=${TAGS} + else + exit 1 + fi + echo ::set-output name=tags::${TAGS} + echo ::set-output name=inspect_tag::${INSPECT_TAG} + - + name: Checkout + uses: actions/checkout@v2 + - + name: Download frontend artifact + uses: actions/download-artifact@v2 + with: + name: frontend-compiled + path: src/documents/static/frontend/ + - + name: Set up Docker Buildx + uses: docker/setup-buildx-action@v1 + - + name: Set up QEMU + uses: docker/setup-qemu-action@v1 + - + name: Cache Docker layers + uses: actions/cache@v2 + with: + path: /tmp/.buildx-cache + key: ${{ runner.os }}-buildx-${{ github.sha }} + restore-keys: | + ${{ runner.os }}-buildx- + - + name: Login to DockerHub + uses: docker/login-action@v1 + with: + username: ${{ secrets.DOCKER_USERNAME }} + password: ${{ secrets.DOCKER_PASSWORD }} + - + name: Build and push + uses: docker/build-push-action@v2 + with: + context: . + file: ./Dockerfile + platforms: linux/amd64,linux/arm/v7,linux/arm64 + push: true + tags: ${{ steps.prepare.outputs.tags }} + cache-from: type=local,src=/tmp/.buildx-cache + cache-to: type=local,dest=/tmp/.buildx-cache + - + name: Inspect image + run: | + docker buildx imagetools inspect ${{ steps.prepare.outputs.inspect_tag }} diff --git a/.gitignore b/.gitignore index d63794fb3..b427d5c5c 100644 --- a/.gitignore +++ b/.gitignore @@ -65,8 +65,8 @@ target/ .virtualenv virtualenv /venv -docker-compose.env -docker-compose.yml +/docker-compose.env +/docker-compose.yml # Used for development scripts/import-for-development @@ -85,3 +85,4 @@ scripts/nuke # this is where the compiled frontend is moved to. /src/documents/static/frontend/ +/docs/.vscode/settings.json diff --git a/.readthedocs.yml b/.readthedocs.yml new file mode 100644 index 000000000..05a8e0c62 --- /dev/null +++ b/.readthedocs.yml @@ -0,0 +1,16 @@ +# .readthedocs.yml +# Read the Docs configuration file +# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details + +# Required +version: 2 + +# Build documentation in the docs/ directory with Sphinx +sphinx: + configuration: docs/conf.py + +# Optionally set the version of Python and requirements required to build your docs +python: + version: 3.7 + install: + - requirements: docs/requirements.txt diff --git a/.travis.yml b/.travis.yml deleted file mode 100644 index b745d6bd7..000000000 --- a/.travis.yml +++ /dev/null @@ -1,51 +0,0 @@ -language: python - -dist: focal -os: linux - -jobs: - include: - - name: "Paperless on Python 3.6" - python: "3.6" - - - name: "Paperless on Python 3.7" - python: "3.7" - - - name: "Paperless on Python 3.8" - python: "3.8" - - - name: "Documentation" - script: - - cd docs/ - - make html - after_success: true - - - name: "Front end" - language: node_js - node_js: - - 15 - before_install: true - install: - - cd src-ui/ - - npm install -g @angular/cli - - npm install - script: - - ng build --prod - after_success: true - - -before_install: - - sudo apt-get update -qq - - sudo apt-get install -qq libpoppler-cpp-dev unpaper tesseract-ocr imagemagick ghostscript optipng - -install: - - pip install --upgrade pipenv - - pipenv install --system --dev - -script: - - cd src/ - - pipenv run pytest --cov - - pipenv run pycodestyle - -after_success: - - pipenv run coveralls diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index a8fb1f8e9..0bbd6c8ea 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -9,11 +9,11 @@ If you want to implement something big: Please start a discussion about that in ## Python -Use python 3.6 for development. Paperless supports python 3.6, 3.7 and 3.8. +Paperless supports python 3.6, 3.7, 3.8 and 3.9. ## Branches -master always reflects the latest release. +master always reflects the latest release. Apart from changes to the documentation or readme, absolutely no functional changes on this branch in between releases. dev contains all changes that will be part of the next release. Use this branch to start making your changes. diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 000000000..8ddb56479 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,104 @@ +FROM ubuntu:20.04 AS jbig2enc + +WORKDIR /usr/src/jbig2enc + +RUN apt-get update && apt-get install -y --no-install-recommends build-essential automake libtool libleptonica-dev zlib1g-dev git ca-certificates + +RUN git clone https://github.com/agl/jbig2enc . +RUN ./autogen.sh +RUN ./configure && make + +FROM python:3.9-slim-bullseye + +# Binary dependencies +RUN apt-get update \ + && apt-get -y --no-install-recommends install \ + # Basic dependencies + curl \ + gnupg \ + imagemagick \ + gettext \ + tzdata \ + gosu \ + # fonts for text file thumbnail generation + fonts-liberation \ + # for Numpy + libatlas-base-dev \ + libxslt1-dev \ + # thumbnail size reduction + optipng \ + libxml2 \ + pngquant \ + unpaper \ + zlib1g \ + ghostscript \ + icc-profiles-free \ + # Mime type detection + file \ + libmagic-dev \ + media-types \ + # OCRmyPDF dependencies + liblept5 \ + qpdf \ + tesseract-ocr \ + tesseract-ocr-eng \ + tesseract-ocr-deu \ + tesseract-ocr-fra \ + tesseract-ocr-ita \ + tesseract-ocr-spa \ + && rm -rf /var/lib/apt/lists/* + +# copy jbig2enc +COPY --from=jbig2enc /usr/src/jbig2enc/src/.libs/libjbig2enc* /usr/local/lib/ +COPY --from=jbig2enc /usr/src/jbig2enc/src/jbig2 /usr/local/bin/ +COPY --from=jbig2enc /usr/src/jbig2enc/src/*.h /usr/local/include/ + +WORKDIR /usr/src/paperless/src/ + +COPY requirements.txt ../ + +# Python dependencies +RUN apt-get update \ + && apt-get -y --no-install-recommends install \ + build-essential \ + libpq-dev \ + libqpdf-dev \ + && python3 -m pip install --upgrade --no-cache-dir supervisor \ + && python3 -m pip install --no-cache-dir -r ../requirements.txt \ + && apt-get -y purge build-essential libqpdf-dev \ + && apt-get -y autoremove --purge \ + && rm -rf /var/lib/apt/lists/* + +# setup docker-specific things +COPY docker/ ./docker/ + +RUN cd docker \ + && cp imagemagick-policy.xml /etc/ImageMagick-6/policy.xml \ + && mkdir /var/log/supervisord /var/run/supervisord \ + && cp supervisord.conf /etc/supervisord.conf \ + && cp docker-entrypoint.sh /sbin/docker-entrypoint.sh \ + && cp docker-prepare.sh /sbin/docker-prepare.sh \ + && chmod 755 /sbin/docker-entrypoint.sh \ + && chmod +x install_management_commands.sh \ + && ./install_management_commands.sh \ + && cd .. \ + && rm docker -rf + +COPY gunicorn.conf.py ../ + +# copy app +COPY src/ ./ + +# add users, setup scripts +RUN addgroup --gid 1000 paperless \ + && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ + && chown -R paperless:paperless ../ \ + && gosu paperless python3 manage.py collectstatic --clear --no-input \ + && gosu paperless python3 manage.py compilemessages + +VOLUME ["/usr/src/paperless/data", "/usr/src/paperless/media", "/usr/src/paperless/consume", "/usr/src/paperless/export"] +ENTRYPOINT ["/sbin/docker-entrypoint.sh"] +EXPOSE 8000 +CMD ["/usr/local/bin/supervisord", "-c", "/etc/supervisord.conf"] + +LABEL maintainer="Jonas Winkler " diff --git a/Pipfile b/Pipfile index 48759307c..1a5d6f035 100644 --- a/Pipfile +++ b/Pipfile @@ -8,40 +8,49 @@ url = "https://www.piwheels.org/simple" verify_ssl = true name = "piwheels" -[requires] -python_version = "3.6" - [packages] -dateparser = "~=0.7.6" -django = "~=3.1.3" +dateparser = "~=1.0.0" +django = "~=3.2" django-cors-headers = "*" django-extensions = "*" django-filter = "~=2.4.0" django-q = "~=1.3.4" djangorestframework = "~=3.12.2" filelock = "*" -fuzzywuzzy = "*" +fuzzywuzzy = {extras = ["speedup"], version = "*"} gunicorn = "*" imap-tools = "*" langdetect = "*" -pdftotext = "*" +numpy = "~=1.20.0" pathvalidate = "*" -pillow = "*" -pikepdf = "*" +pillow = "~=8.1" +pikepdf = "~=2.5" python-gnupg = "*" python-dotenv = "*" python-dateutil = "*" -python-Levenshtein = "*" python-magic = "*" psycopg2-binary = "*" redis = "*" -scikit-learn="~=0.23.2" -whitenoise = "~=5.2.0" -watchdog = "*" +# Pinned because aarch64 wheels and updates cause warnings when loading the classifier model. +scikit-learn="==0.24.0" +whitenoise = "~=5.3.0" +watchdog = "~=2.1.0" whoosh="~=2.7.4" inotifyrecursive = "~=0.3.4" -ocrmypdf = "*" +ocrmypdf = "~=12.3" tqdm = "*" +tika = "*" +# TODO: This will sadly also install daphne+dependencies, +# which an ASGI server we don't need. Adds about 15MB image size. +channels = "~=3.0" +channels-redis = "*" +uvicorn = {extras = ["standard"], version = "*"} +concurrent-log-handler = "*" +# uvloop 0.15+ incompatible with python 3.6 +uvloop = "~=0.15" +cryptography = "~=3.4" +"pdfminer.six" = "*" +"backports.zoneinfo" = "*" [dev-packages] coveralls = "*" @@ -53,6 +62,6 @@ pytest-django = "*" pytest-env = "*" pytest-sugar = "*" pytest-xdist = "*" -sphinx = "~=3.3" +sphinx = "~=3.4.2" sphinx_rtd_theme = "*" tox = "*" diff --git a/Pipfile.lock b/Pipfile.lock index 1cfccb8ff..4f297844d 100644 --- a/Pipfile.lock +++ b/Pipfile.lock @@ -1,12 +1,10 @@ { "_meta": { "hash": { - "sha256": "3d576f289958226a7583e4c471c7f8c11bff6933bf093185f623cfb381a92412" + "sha256": "e574568592f15d69f1126bd8057bddee0fc7d41df38a81eda24658628cfbb8fd" }, "pipfile-spec": 6, - "requires": { - "python_version": "3.6" - }, + "requires": {}, "sources": [ { "name": "pypi", @@ -21,150 +19,280 @@ ] }, "default": { + "aioredis": { + "hashes": [ + "sha256:15f8af30b044c771aee6787e5ec24694c048184c7b9e54c3b60c750a4b93273a", + "sha256:b61808d7e97b7cd5a92ed574937a079c9387fdadd22bfbfa7ad2fd319ecc26e3" + ], + "version": "==1.3.1" + }, "arrow": { "hashes": [ - "sha256:e098abbd9af3665aea81bdd6c869e93af4feb078e98468dd351c383af187aac5", - "sha256:ff08d10cda1d36c68657d6ad20d74fbea493d980f8b2d45344e00d6ed2bf6ed4" + "sha256:77a60a4db5766d900a2085ce9074c5c7b8e2c99afeaa98ad627637ff6f292510", + "sha256:dee7602f6c60e3ec510095b5e301441bc56288cb8f51def14dcb3079f623823a" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==0.17.0" + "markers": "python_version >= '3.6'", + "version": "==1.1.1" }, "asgiref": { "hashes": [ - "sha256:5ee950735509d04eb673bd7f7120f8fa1c9e2df495394992c73234d526907e17", - "sha256:7162a3cb30ab0609f1a4c95938fd73e8604f63bdba516a7f7d64b83ff09478f0" + "sha256:4ef1ab46b484e3c706329cedeff284a5d40824200638503f5768edb6de7d58e9", + "sha256:ffc141aa908e6f175673e7b1b3b7af4fdb0ecb738fc5c8b88f69f055c2415214" ], - "markers": "python_version >= '3.5'", - "version": "==3.3.1" + "markers": "python_version >= '3.6'", + "version": "==3.4.1" + }, + "async-timeout": { + "hashes": [ + "sha256:0c3c816a028d47f659d6ff5c745cb2acf1f966da1fe5c19c77a70282b25f4c5f", + "sha256:4291ca197d287d274d0b6cb5d6f8f8f82d434ed288f962539ff18cc9012f9ea3" + ], + "markers": "python_full_version >= '3.5.3'", + "version": "==3.0.1" + }, + "attrs": { + "hashes": [ + "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1", + "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb" + ], + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==21.2.0" + }, + "autobahn": { + "hashes": [ + "sha256:9195df8af03b0ff29ccd4b7f5abbde957ee90273465942205f9a1bad6c3f07ac", + "sha256:e126c1f583e872fb59e79d36977cfa1f2d0a8a79f90ae31f406faae7664b8e03" + ], + "markers": "python_version >= '3.7'", + "version": "==21.3.1" + }, + "automat": { + "hashes": [ + "sha256:7979803c74610e11ef0c0d68a2942b152df52da55336e0c9d58daf1831cbdf33", + "sha256:b6feb6455337df834f6c9962d6ccf771515b7d939bca142b29c20c2376bc6111", + "sha256:d6d976cf8da698fc85fa7def46e2544493f78cb7ee72d2f4acd1a5c759a3060e" + ], + "version": "==20.2.0" + }, + "backports.zoneinfo": { + "hashes": [ + "sha256:17746bd546106fa389c51dbea67c8b7c8f0d14b5526a579ca6ccf5ed72c526cf", + "sha256:1b13e654a55cd45672cb54ed12148cd33628f672548f373963b0bff67b217328", + "sha256:1c5742112073a563c81f786e77514969acb58649bcdf6cdf0b4ed31a348d4546", + "sha256:4a0f800587060bf8880f954dbef70de6c11bbe59c673c3d818921f042f9954a6", + "sha256:5c144945a7752ca544b4b78c8c41544cdfaf9786f25fe5ffb10e838e19a27570", + "sha256:7b0a64cda4145548fed9efc10322770f929b944ce5cee6c0dfe0c87bf4c0c8c9", + "sha256:8439c030a11780786a2002261569bdf362264f605dfa4d65090b64b05c9f79a7", + "sha256:8961c0f32cd0336fb8e8ead11a1f8cd99ec07145ec2931122faaac1c8f7fd987", + "sha256:89a48c0d158a3cc3f654da4c2de1ceba85263fafb861b98b59040a5086259722", + "sha256:a76b38c52400b762e48131494ba26be363491ac4f9a04c1b7e92483d169f6582", + "sha256:da6013fd84a690242c310d77ddb8441a559e9cb3d3d59ebac9aca1a57b2e18bc", + "sha256:dff179a489c57d2c9ccd60720f1ec13173bc1aa1c15c2714649581b32eab56ca", + "sha256:e34cdbc646438368c17100e3dafeb88626d3b5467b89681d20ef6731a29e2895", + "sha256:e55b384612d93be96506932a786bbcde5a2db7a9e6a4bb4bffe8b733f5b9036b", + "sha256:e81b76cace8eda1fca50e345242ba977f9be6ae3945af8d46326d776b4cf78d1", + "sha256:e8236383a20872c0cdf5a62b554b27538db7fa1bbec52429d8d106effbaeca08", + "sha256:f04e857b59d9d1ccc39ce2da1021d196e47234873820cbeaad210724b1ee28ac", + "sha256:fadbfe37f74051d024037f223b8e001611eac868b5c5b06144ef4d8b799862f2" + ], + "index": "pypi", + "version": "==0.2.1" }, "blessed": { "hashes": [ - "sha256:0a74a8d3f0366db600d061273df77d44f0db07daade7bb7a4d49c8bc22ed9f74", - "sha256:580429e7e0c6f6a42ea81b0ae5a4993b6205c6ccbb635d034b4277af8175753e" + "sha256:8b09936def6bc06583db99b65636b980075733e13550cb6af262ce724a55da23", + "sha256:dd7c0d33db9a2e7f597b446996484d0ed46e1586239db064fb5025008937dcae" ], - "version": "==1.17.12" + "markers": "python_version >= '2.7'", + "version": "==1.18.1" + }, + "certifi": { + "hashes": [ + "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee", + "sha256:50b1e4f8446b06f41be7dd6338db18e0990601dce795c2b1686458aa7e8fa7d8" + ], + "version": "==2021.5.30" }, "cffi": { "hashes": [ - "sha256:00a1ba5e2e95684448de9b89888ccd02c98d512064b4cb987d48f4b40aa0421e", - "sha256:00e28066507bfc3fe865a31f325c8391a1ac2916219340f87dfad602c3e48e5d", - "sha256:045d792900a75e8b1e1b0ab6787dd733a8190ffcf80e8c8ceb2fb10a29ff238a", - "sha256:0638c3ae1a0edfb77c6765d487fee624d2b1ee1bdfeffc1f0b58c64d149e7eec", - "sha256:105abaf8a6075dc96c1fe5ae7aae073f4696f2905fde6aeada4c9d2926752362", - "sha256:155136b51fd733fa94e1c2ea5211dcd4c8879869008fc811648f16541bf99668", - "sha256:1a465cbe98a7fd391d47dce4b8f7e5b921e6cd805ef421d04f5f66ba8f06086c", - "sha256:1d2c4994f515e5b485fd6d3a73d05526aa0fcf248eb135996b088d25dfa1865b", - "sha256:23f318bf74b170c6e9adb390e8bd282457f6de46c19d03b52f3fd042b5e19654", - "sha256:2c24d61263f511551f740d1a065eb0212db1dbbbbd241db758f5244281590c06", - "sha256:51a8b381b16ddd370178a65360ebe15fbc1c71cf6f584613a7ea08bfad946698", - "sha256:594234691ac0e9b770aee9fcdb8fa02c22e43e5c619456efd0d6c2bf276f3eb2", - "sha256:5cf4be6c304ad0b6602f5c4e90e2f59b47653ac1ed9c662ed379fe48a8f26b0c", - "sha256:64081b3f8f6f3c3de6191ec89d7dc6c86a8a43911f7ecb422c60e90c70be41c7", - "sha256:6bc25fc545a6b3d57b5f8618e59fc13d3a3a68431e8ca5fd4c13241cd70d0009", - "sha256:798caa2a2384b1cbe8a2a139d80734c9db54f9cc155c99d7cc92441a23871c03", - "sha256:7c6b1dece89874d9541fc974917b631406233ea0440d0bdfbb8e03bf39a49b3b", - "sha256:840793c68105fe031f34d6a086eaea153a0cd5c491cde82a74b420edd0a2b909", - "sha256:8d6603078baf4e11edc4168a514c5ce5b3ba6e3e9c374298cb88437957960a53", - "sha256:9cc46bc107224ff5b6d04369e7c595acb700c3613ad7bcf2e2012f62ece80c35", - "sha256:9f7a31251289b2ab6d4012f6e83e58bc3b96bd151f5b5262467f4bb6b34a7c26", - "sha256:9ffb888f19d54a4d4dfd4b3f29bc2c16aa4972f1c2ab9c4ab09b8ab8685b9c2b", - "sha256:a5ed8c05548b54b998b9498753fb9cadbfd92ee88e884641377d8a8b291bcc01", - "sha256:a7711edca4dcef1a75257b50a2fbfe92a65187c47dab5a0f1b9b332c5919a3fb", - "sha256:af5c59122a011049aad5dd87424b8e65a80e4a6477419c0c1015f73fb5ea0293", - "sha256:b18e0a9ef57d2b41f5c68beefa32317d286c3d6ac0484efd10d6e07491bb95dd", - "sha256:b4e248d1087abf9f4c10f3c398896c87ce82a9856494a7155823eb45a892395d", - "sha256:ba4e9e0ae13fc41c6b23299545e5ef73055213e466bd107953e4a013a5ddd7e3", - "sha256:be8661bcee1bc2fc4b033a6ab65bd1f87ce5008492601695d0b9a4e820c3bde5", - "sha256:c6332685306b6417a91b1ff9fae889b3ba65c2292d64bd9245c093b1b284809d", - "sha256:d5ff0621c88ce83a28a10d2ce719b2ee85635e85c515f12bac99a95306da4b2e", - "sha256:d9efd8b7a3ef378dd61a1e77367f1924375befc2eba06168b6ebfa903a5e59ca", - "sha256:df5169c4396adc04f9b0a05f13c074df878b6052430e03f50e68adf3a57aa28d", - "sha256:ebb253464a5d0482b191274f1c8bf00e33f7e0b9c66405fbffc61ed2c839c775", - "sha256:ec80dc47f54e6e9a78181ce05feb71a0353854cc26999db963695f950b5fb375", - "sha256:f032b34669220030f905152045dfa27741ce1a6db3324a5bc0b96b6c7420c87b", - "sha256:f60567825f791c6f8a592f3c6e3bd93dd2934e3f9dac189308426bd76b00ef3b", - "sha256:f803eaa94c2fcda012c047e62bc7a51b0bdabda1cad7a92a522694ea2d76e49f" + "sha256:06c54a68935738d206570b20da5ef2b6b6d92b38ef3ec45c5422c0ebaf338d4d", + "sha256:0c0591bee64e438883b0c92a7bed78f6290d40bf02e54c5bf0978eaf36061771", + "sha256:19ca0dbdeda3b2615421d54bef8985f72af6e0c47082a8d26122adac81a95872", + "sha256:1f53e2a841e57da2546cd033668485feaf5dec20ed730e90765e138a3117a8ce", + "sha256:22b9c3c320171c108e903d61a3723b51e37aaa8c81255b5e7ce102775bd01e2c", + "sha256:26bb2549b72708c833f5abe62b756176022a7b9a7f689b571e74c8478ead51dc", + "sha256:33791e8a2dc2953f28b8d8d300dde42dd929ac28f974c4b4c6272cb2955cb762", + "sha256:3c8d896becff2fa653dc4438b54a5a25a971d1f4110b32bd3068db3722c80202", + "sha256:4373612d59c404baeb7cbd788a18b2b2a8331abcc84c3ba40051fcd18b17a4d5", + "sha256:487d63e1454627c8e47dd230025780e91869cfba4c753a74fda196a1f6ad6548", + "sha256:48916e459c54c4a70e52745639f1db524542140433599e13911b2f329834276a", + "sha256:4922cd707b25e623b902c86188aca466d3620892db76c0bdd7b99a3d5e61d35f", + "sha256:55af55e32ae468e9946f741a5d51f9896da6b9bf0bbdd326843fec05c730eb20", + "sha256:57e555a9feb4a8460415f1aac331a2dc833b1115284f7ded7278b54afc5bd218", + "sha256:5d4b68e216fc65e9fe4f524c177b54964af043dde734807586cf5435af84045c", + "sha256:64fda793737bc4037521d4899be780534b9aea552eb673b9833b01f945904c2e", + "sha256:6d6169cb3c6c2ad50db5b868db6491a790300ade1ed5d1da29289d73bbe40b56", + "sha256:7bcac9a2b4fdbed2c16fa5681356d7121ecabf041f18d97ed5b8e0dd38a80224", + "sha256:80b06212075346b5546b0417b9f2bf467fea3bfe7352f781ffc05a8ab24ba14a", + "sha256:818014c754cd3dba7229c0f5884396264d51ffb87ec86e927ef0be140bfdb0d2", + "sha256:8eb687582ed7cd8c4bdbff3df6c0da443eb89c3c72e6e5dcdd9c81729712791a", + "sha256:99f27fefe34c37ba9875f224a8f36e31d744d8083e00f520f133cab79ad5e819", + "sha256:9f3e33c28cd39d1b655ed1ba7247133b6f7fc16fa16887b120c0c670e35ce346", + "sha256:a8661b2ce9694ca01c529bfa204dbb144b275a31685a075ce123f12331be790b", + "sha256:a9da7010cec5a12193d1af9872a00888f396aba3dc79186604a09ea3ee7c029e", + "sha256:aedb15f0a5a5949ecb129a82b72b19df97bbbca024081ed2ef88bd5c0a610534", + "sha256:b315d709717a99f4b27b59b021e6207c64620790ca3e0bde636a6c7f14618abb", + "sha256:ba6f2b3f452e150945d58f4badd92310449876c4c954836cfb1803bdd7b422f0", + "sha256:c33d18eb6e6bc36f09d793c0dc58b0211fccc6ae5149b808da4a62660678b156", + "sha256:c993e9eefcd5d3b6ff6bf16e85a3c5c331ec8ac47d4b1a942da58aaf46f16519", + "sha256:c9a875ce9d7fe32887784274dd533c57909b7b1dcadcc128a2ac21331a9765dd", + "sha256:c9e005e9bd57bc987764c32a1bee4364c44fdc11a3cc20a40b93b444984f2b87", + "sha256:d2ad4d668a5c0645d281dcd17aff2be3212bc109b33814bbb15c4939f44181cc", + "sha256:d950695ae4381ecd856bcaf2b1e866720e4ab9a1498cba61c602e56630ca7195", + "sha256:e22dcb48709fc51a7b58a927391b23ab37eb3737a98ac4338e2448bef8559b33", + "sha256:e8c6a99be100371dbb046880e7a282152aa5d6127ae01783e37662ef73850d8f", + "sha256:e9dc245e3ac69c92ee4c167fbdd7428ec1956d4e754223124991ef29eb57a09d", + "sha256:eb687a11f0a7a1839719edd80f41e459cc5366857ecbed383ff376c4e3cc6afd", + "sha256:eb9e2a346c5238a30a746893f23a9535e700f8192a68c07c0258e7ece6ff3728", + "sha256:ed38b924ce794e505647f7c331b22a693bee1538fdf46b0222c4717b42f744e7", + "sha256:f0010c6f9d1a4011e429109fda55a225921e3206e7f62a0c22a35344bfd13cca", + "sha256:f0c5d1acbfca6ebdd6b1e3eded8d261affb6ddcf2186205518f1428b8569bb99", + "sha256:f10afb1004f102c7868ebfe91c28f4a712227fe4cb24974350ace1f90e1febbf", + "sha256:f174135f5609428cc6e1b9090f9268f5c8935fddb1b25ccb8255a2d50de6789e", + "sha256:f3ebe6e73c319340830a9b2825d32eb6d8475c1dac020b4f0aa774ee3b898d1c", + "sha256:f627688813d0a4140153ff532537fbe4afea5a3dffce1f9deb7f91f848a832b5", + "sha256:fd4305f86f53dfd8cd3522269ed7fc34856a8ee3709a5e28b2836b2db9d4cd69", + "sha256:ff33cea1d8d00288900efac48b20257d1efe9df77ede3a3fcc397a343337407d" ], - "version": "==1.14.4" + "version": "==1.14.6" + }, + "channels": { + "hashes": [ + "sha256:0ff0422b4224d10efac76e451575517f155fe7c97d369b5973b116f22eeaf86c", + "sha256:fdd9a94987a23d8d7ebd97498ed8b8cc83163f37e53fc6c85098aba7a3bb8b75" + ], + "index": "pypi", + "version": "==3.0.4" + }, + "channels-redis": { + "hashes": [ + "sha256:0a18ce279c15ba79b7985bb12b2d6dd0ac8a14e4ad6952681f4422a4cc4a5ea9", + "sha256:1abd5820ff1ed4ac627f8a219ad389e4c87e52e47a230929a7a474e95dd2c6c2" + ], + "index": "pypi", + "version": "==3.3.0" }, "chardet": { "hashes": [ - "sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae", - "sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691" + "sha256:0d6f53a15db4120f2b08c94f11e7d93d2c911ee118b6b30a04ec3ee8310179fa", + "sha256:f864054d66fd9118f2e67044ac8981a54775ec5b67aed0441892edb553d21da5" ], "markers": "python_version >= '3.1'", - "version": "==3.0.4" + "version": "==4.0.0" + }, + "charset-normalizer": { + "hashes": [ + "sha256:0c8911edd15d19223366a194a513099a302055a962bca2cec0f54b8b63175d8b", + "sha256:762c76ab955f265f68d4683641bb10d7f2723dbc10091505886e357e0b42760f", + "sha256:f23667ebe1084be45f6ae0538e4a5a865206544097e4e8bbcacf42cd02a348f3" + ], + "markers": "python_version >= '3'", + "version": "==2.0.4" + }, + "click": { + "hashes": [ + "sha256:8c04c11192119b1ef78ea049e0a6f0463e4c48ef00a30160c704337586f3ad7a", + "sha256:fba402a4a47334742d782209a7c79bc448911afe1149d07bdabdf480b3e2f4b6" + ], + "markers": "python_version >= '3.6'", + "version": "==8.0.1" }, "coloredlogs": { "hashes": [ - "sha256:346f58aad6afd48444c2468618623638dadab76e4e70d5e10822676f2d32226a", - "sha256:a1fab193d2053aa6c0a97608c4342d031f1f93a3d1218432c59322441d31a505", - "sha256:b0c2124367d4f72bd739f48e1f61491b4baf145d6bda33b606b4a53cb3f96a97" + "sha256:612ee75c546f53e92e70049c9dbfcc18c935a2b9a53b66085ce9ef6a6e5c0934", + "sha256:7c991aa71a4577af2f82600d8f8f3a89f936baeaf9b50a9c197da014e5bf16b0" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==14.0" + "version": "==15.0.1" + }, + "concurrent-log-handler": { + "hashes": [ + "sha256:00d5ca24d463a7013c3479b026f34b76da4b50df8d76194132b8d8403c014379", + "sha256:b12f79abed3f94121c25ce9c24cdb57d889282ec6ff61f5535ab2068dc37d409" + ], + "index": "pypi", + "version": "==0.9.19" + }, + "constantly": { + "hashes": [ + "sha256:586372eb92059873e29eba4f9dec8381541b4d3834660707faf8ba59146dfc35", + "sha256:dd2fa9d6b1a51a83f0d7dd76293d734046aa176e384bf6e33b7e44880eb37c5d" + ], + "version": "==15.1.0" }, "cryptography": { "hashes": [ - "sha256:07ca431b788249af92764e3be9a488aa1d39a0bc3be313d826bbec690417e538", - "sha256:13b88a0bd044b4eae1ef40e265d006e34dbcde0c2f1e15eb9896501b2d8f6c6f", - "sha256:257dab4f368fae15f378ea9a4d2799bf3696668062de0e9fa0ebb7a738a6917d", - "sha256:32434673d8505b42c0de4de86da8c1620651abd24afe91ae0335597683ed1b77", - "sha256:3cd75a683b15576cfc822c7c5742b3276e50b21a06672dc3a800a2d5da4ecd1b", - "sha256:4e7268a0ca14536fecfdf2b00297d4e407da904718658c1ff1961c713f90fd33", - "sha256:545a8550782dda68f8cdc75a6e3bf252017aa8f75f19f5a9ca940772fc0cb56e", - "sha256:55d0b896631412b6f0c7de56e12eb3e261ac347fbaa5d5e705291a9016e5f8cb", - "sha256:5849d59358547bf789ee7e0d7a9036b2d29e9a4ddf1ce5e06bb45634f995c53e", - "sha256:59f7d4cfea9ef12eb9b14b83d79b432162a0a24a91ddc15c2c9bf76a68d96f2b", - "sha256:6dc59630ecce8c1f558277ceb212c751d6730bd12c80ea96b4ac65637c4f55e7", - "sha256:7117319b44ed1842c617d0a452383a5a052ec6aa726dfbaffa8b94c910444297", - "sha256:75e8e6684cf0034f6bf2a97095cb95f81537b12b36a8fedf06e73050bb171c2d", - "sha256:7b8d9d8d3a9bd240f453342981f765346c87ade811519f98664519696f8e6ab7", - "sha256:a035a10686532b0587d58a606004aa20ad895c60c4d029afa245802347fab57b", - "sha256:a4e27ed0b2504195f855b52052eadcc9795c59909c9d84314c5408687f933fc7", - "sha256:a733671100cd26d816eed39507e585c156e4498293a907029969234e5e634bc4", - "sha256:a75f306a16d9f9afebfbedc41c8c2351d8e61e818ba6b4c40815e2b5740bb6b8", - "sha256:bd717aa029217b8ef94a7d21632a3bb5a4e7218a4513d2521c2a2fd63011e98b", - "sha256:d25cecbac20713a7c3bc544372d42d8eafa89799f492a43b79e1dfd650484851", - "sha256:d26a2557d8f9122f9bf445fc7034242f4375bd4e95ecda007667540270965b13", - "sha256:d3545829ab42a66b84a9aaabf216a4dce7f16dbc76eb69be5c302ed6b8f4a29b", - "sha256:d3d5e10be0cf2a12214ddee45c6bd203dab435e3d83b4560c03066eda600bfe3", - "sha256:efe15aca4f64f3a7ea0c09c87826490e50ed166ce67368a68f315ea0807a20df" + "sha256:0f1212a66329c80d68aeeb39b8a16d54ef57071bf22ff4e521657b27372e327d", + "sha256:1e056c28420c072c5e3cb36e2b23ee55e260cb04eee08f702e0edfec3fb51959", + "sha256:240f5c21aef0b73f40bb9f78d2caff73186700bf1bc6b94285699aff98cc16c6", + "sha256:26965837447f9c82f1855e0bc8bc4fb910240b6e0d16a664bb722df3b5b06873", + "sha256:2f258145b6ff52bfe4b8f4c8a36705012f449b4bc966ff53b405103e018d6dbc", + "sha256:37340614f8a5d2fb9aeea67fd159bfe4f5f4ed535b1090ce8ec428b2f15a11f2", + "sha256:3d10de8116d25649631977cb37da6cbdd2d6fa0e0281d014a5b7d337255ca713", + "sha256:3d8427734c781ea5f1b41d6589c293089704d4759e34597dce91014ac125aad1", + "sha256:7ec5d3b029f5fa2b179325908b9cd93db28ab7b85bb6c1db56b10e0b54235177", + "sha256:8e56e16617872b0957d1c9742a3f94b43533447fd78321514abbe7db216aa250", + "sha256:a79fef41233d4c535d28133627ce6c7ac73d5cc0eb7316331a5905bf01411f08", + "sha256:b01fd6f2737816cb1e08ed4807ae194404790eac7ad030b34f2ce72b332f5586", + "sha256:bf40af59ca2465b24e54f671b2de2c59257ddc4f7e5706dbd6930e26823668d3", + "sha256:de4e5f7f68220d92b7637fc99847475b59154b7a1b3868fb7385337af54ac9ca", + "sha256:eb8cc2afe8b05acbd84a43905832ec78e7b3873fb124ca190f574dca7389a87d", + "sha256:ee77aa129f481be46f8d92a1a7db57269a2f23052d5f2433b4621bb457081cc9" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==3.2.1" + "index": "pypi", + "version": "==3.4.7" + }, + "daphne": { + "hashes": [ + "sha256:76ffae916ba3aa66b46996c14fa713e46004788167a4873d647544e750e0e99f", + "sha256:a9af943c79717bc52fe64a3c236ae5d3adccc8b5be19c881b442d2c3db233393" + ], + "markers": "python_version >= '3.6'", + "version": "==3.0.2" }, "dateparser": { "hashes": [ - "sha256:7552c994f893b5cb8fcf103b4cd2ff7f57aab9bfd2619fdf0cf571c0740fd90b", - "sha256:e875efd8c57c85c2d02b238239878db59ff1971f5a823457fcc69e493bf6ebfa" + "sha256:159cc4e01a593706a15cd4e269a0b3345edf3aef8bf9278a57dac8adf5bf1e4a", + "sha256:17202df32c7a36e773136ff353aa3767e987f8b3e27374c39fd21a30a803d6f8" ], "index": "pypi", - "version": "==0.7.6" + "version": "==1.0.0" }, "django": { "hashes": [ - "sha256:5c866205f15e7a7123f1eec6ab939d22d5bde1416635cab259684af66d8e48a2", - "sha256:edb10b5c45e7e9c0fb1dc00b76ec7449aca258a39ffd613dbd078c51d19c9f03" + "sha256:1ef5b9e9d5d41990fa877fd5a205c72076f5edfd7157109238c810a60eadeda8", + "sha256:7f92413529aa0e291f3be78ab19be31aefb1e1c9a52cd59e130f505f27a51f13", + "sha256:f27f8544c9d4c383bbe007c57e3235918e258364577373d4920e9162837be022" ], "index": "pypi", - "version": "==3.1.4" + "version": "==3.2.6" }, "django-cors-headers": { "hashes": [ - "sha256:9322255c296d5f75089571f29e520c83ff9693df17aa3cf9f6a4bea7c6740169", - "sha256:db82b2840f667d47872ae3e4a4e0a0d72fbecb42779b8aa233fa8bb965f7836a" + "sha256:3777bdb1f8472e898b3cd8a015862e334474f35791d9619f125eb65a83fee1e3", + "sha256:425c20ceffa42b9ac11b02611eece4ae6c5fef2ff0f039c14c1df20e00c80df8", + "sha256:4b8e13bf8d3df50ac4b986bd87085c3073dd56402ede109222ea34a774f9ec1b" ], "index": "pypi", - "version": "==3.5.0" + "version": "==3.8.0" }, "django-extensions": { "hashes": [ - "sha256:7cd002495ff0a0e5eb6cdd6be759600905b4e4079232ea27618fc46bdd853651", - "sha256:c7f88625a53f631745d4f2bef9ec4dcb999ed59476393bdbbe99db8596778846" + "sha256:50de8977794a66a91575dd40f87d5053608f679561731845edbd325ceeb387e3", + "sha256:5f0fea7bf131ca303090352577a9e7f8bfbf5489bd9d9c8aea9401db28db34a0" ], "index": "pypi", - "version": "==3.1.0" + "version": "==3.1.3" }, "django-filter": { "hashes": [ @@ -184,18 +312,19 @@ }, "django-q": { "hashes": [ - "sha256:523d54dcf1b66152c1b658f914f00ed3b518a3432a9decd4898738ca8dbbe10f", - "sha256:7e5c5c021a15cff6807044a3aa48f5757789ccfef839d71c575f5512931a3e33" + "sha256:1b74ce3a8931990b136903e3a7bc9b07243282a2b5355117246f05ed5d076e68", + "sha256:5c6b4d530aa3aabf9c6aa57376da1ca2abf89a1562b77038b7a04e52a4a0a91b" ], "index": "pypi", - "version": "==1.3.4" + "version": "==1.3.9" }, "djangorestframework": { "hashes": [ - "sha256:0209bafcb7b5010fdfec784034f059d512256424de2a0f084cb82b096d6dd6a7" + "sha256:6d1d59f623a5ad0509fe0d6bfe93cbdfe17b8116ebc8eda86d45f6e16e819aaf", + "sha256:f747949a8ddac876e879190df194b925c177cdeb725a099db1460872f7c0a7f2" ], "index": "pypi", - "version": "==3.12.2" + "version": "==3.12.4" }, "filelock": { "hashes": [ @@ -206,6 +335,9 @@ "version": "==3.0.12" }, "fuzzywuzzy": { + "extras": [ + "speedup" + ], "hashes": [ "sha256:45016e92264780e58972dca1b3d939ac864b78437422beecebb3095f8efd00e8", "sha256:928244b28db720d1e0ee7587acf660ea49d7e4c632569cad4f1cd7e68a5f0993" @@ -215,42 +347,137 @@ }, "gunicorn": { "hashes": [ - "sha256:1904bb2b8a43658807108d59c3f3d56c2b6121a701161de0ddf9ad140073c626", - "sha256:cd4a810dd51bf497552cf3f863b575dabd73d6ad6a91075b65936b151cbf4f9c" + "sha256:8d737657cdd62f483d482ee65cf9cc6a64d5b2199674f8b174803be9065e5d60", + "sha256:9dcc4547dbb1cb284accfb15ab5667a0e5d1881cc443e0677b4882a4067a807e", + "sha256:e0a968b5ba15f8a328fdfd7ab1fcb5af4470c28aaf7e55df02a99bc13138e6e8" ], "index": "pypi", - "version": "==20.0.4" + "version": "==20.1.0" + }, + "h11": { + "hashes": [ + "sha256:36a3cb8c0a032f56e2da7084577878a035d3b61d104230d4bd49c0c6b555a9c6", + "sha256:47222cb6067e4a307d535814917cd98fd0a57b6788ce715755fa2b6c28b56042" + ], + "markers": "python_version >= '3.6'", + "version": "==0.12.0" + }, + "hiredis": { + "hashes": [ + "sha256:04026461eae67fdefa1949b7332e488224eac9e8f2b5c58c98b54d29af22093e", + "sha256:04927a4c651a0e9ec11c68e4427d917e44ff101f761cd3b5bc76f86aaa431d27", + "sha256:07bbf9bdcb82239f319b1f09e8ef4bdfaec50ed7d7ea51a56438f39193271163", + "sha256:09004096e953d7ebd508cded79f6b21e05dff5d7361771f59269425108e703bc", + "sha256:0adea425b764a08270820531ec2218d0508f8ae15a448568109ffcae050fee26", + "sha256:0b39ec237459922c6544d071cdcf92cbb5bc6685a30e7c6d985d8a3e3a75326e", + "sha256:0d5109337e1db373a892fdcf78eb145ffb6bbd66bb51989ec36117b9f7f9b579", + "sha256:0f41827028901814c709e744060843c77e78a3aca1e0d6875d2562372fcb405a", + "sha256:11d119507bb54e81f375e638225a2c057dda748f2b1deef05c2b1a5d42686048", + "sha256:1233e303645f468e399ec906b6b48ab7cd8391aae2d08daadbb5cad6ace4bd87", + "sha256:139705ce59d94eef2ceae9fd2ad58710b02aee91e7fa0ccb485665ca0ecbec63", + "sha256:1f03d4dadd595f7a69a75709bc81902673fa31964c75f93af74feac2f134cc54", + "sha256:240ce6dc19835971f38caf94b5738092cb1e641f8150a9ef9251b7825506cb05", + "sha256:294a6697dfa41a8cba4c365dd3715abc54d29a86a40ec6405d677ca853307cfb", + "sha256:3d55e36715ff06cdc0ab62f9591607c4324297b6b6ce5b58cb9928b3defe30ea", + "sha256:3dddf681284fe16d047d3ad37415b2e9ccdc6c8986c8062dbe51ab9a358b50a5", + "sha256:3f5f7e3a4ab824e3de1e1700f05ad76ee465f5f11f5db61c4b297ec29e692b2e", + "sha256:508999bec4422e646b05c95c598b64bdbef1edf0d2b715450a078ba21b385bcc", + "sha256:5d2a48c80cf5a338d58aae3c16872f4d452345e18350143b3bf7216d33ba7b99", + "sha256:5dc7a94bb11096bc4bffd41a3c4f2b958257085c01522aa81140c68b8bf1630a", + "sha256:65d653df249a2f95673976e4e9dd7ce10de61cfc6e64fa7eeaa6891a9559c581", + "sha256:7492af15f71f75ee93d2a618ca53fea8be85e7b625e323315169977fae752426", + "sha256:7f0055f1809b911ab347a25d786deff5e10e9cf083c3c3fd2dd04e8612e8d9db", + "sha256:807b3096205c7cec861c8803a6738e33ed86c9aae76cac0e19454245a6bbbc0a", + "sha256:81d6d8e39695f2c37954d1011c0480ef7cf444d4e3ae24bc5e89ee5de360139a", + "sha256:87c7c10d186f1743a8fd6a971ab6525d60abd5d5d200f31e073cd5e94d7e7a9d", + "sha256:8b42c0dc927b8d7c0eb59f97e6e34408e53bc489f9f90e66e568f329bff3e443", + "sha256:a00514362df15af041cc06e97aebabf2895e0a7c42c83c21894be12b84402d79", + "sha256:a39efc3ade8c1fb27c097fd112baf09d7fd70b8cb10ef1de4da6efbe066d381d", + "sha256:a4ee8000454ad4486fb9f28b0cab7fa1cd796fc36d639882d0b34109b5b3aec9", + "sha256:a7928283143a401e72a4fad43ecc85b35c27ae699cf5d54d39e1e72d97460e1d", + "sha256:adf4dd19d8875ac147bf926c727215a0faf21490b22c053db464e0bf0deb0485", + "sha256:ae8427a5e9062ba66fc2c62fb19a72276cf12c780e8db2b0956ea909c48acff5", + "sha256:b4c8b0bc5841e578d5fb32a16e0c305359b987b850a06964bd5a62739d688048", + "sha256:b84f29971f0ad4adaee391c6364e6f780d5aae7e9226d41964b26b49376071d0", + "sha256:c39c46d9e44447181cd502a35aad2bb178dbf1b1f86cf4db639d7b9614f837c6", + "sha256:cb2126603091902767d96bcb74093bd8b14982f41809f85c9b96e519c7e1dc41", + "sha256:dcef843f8de4e2ff5e35e96ec2a4abbdf403bd0f732ead127bd27e51f38ac298", + "sha256:de22905c5920f6a2af2fccc54a7897cf52f1d0a4b9de87f546c6fb70d13e4e0b", + "sha256:e3447d9e074abf0e3cd85aef8131e01ab93f9f0e86654db7ac8a3f73c63706ce", + "sha256:f299820a91e907fcfb92ca45923bfc0653c7314c05f0dfef226c76c608a95f4a", + "sha256:f52010e0a44e3d8530437e7da38d11fb822acfb0d5b12e9cd5ba655509937ca0", + "sha256:f8196f739092a78e4f6b1b2172679ed3343c39c61a3e9d722ce6fcf1dac2824a" + ], + "markers": "python_version >= '3.6'", + "version": "==2.0.0" + }, + "httptools": { + "hashes": [ + "sha256:01b392a166adcc8bc2f526a939a8aabf89fe079243e1543fd0e7dc1b58d737cb", + "sha256:200fc1cdf733a9ff554c0bb97a4047785cfaad9875307d6087001db3eb2b417f", + "sha256:3ab1f390d8867f74b3b5ee2a7ecc9b8d7f53750bd45714bf1cb72a953d7dfa77", + "sha256:78d03dd39b09c99ec917d50189e6743adbfd18c15d5944392d2eabda688bf149", + "sha256:79dbc21f3612a78b28384e989b21872e2e3cf3968532601544696e4ed0007ce5", + "sha256:80ffa04fe8c8dfacf6e4cef8277347d35b0442c581f5814f3b0cf41b65c43c6e", + "sha256:813871f961edea6cb2fe312f2d9b27d12a51ba92545380126f80d0de1917ea15", + "sha256:94505026be56652d7a530ab03d89474dc6021019d6b8682281977163b3471ea0", + "sha256:a23166e5ae2775709cf4f7ad4c2048755ebfb272767d244e1a96d55ac775cca7", + "sha256:a289c27ccae399a70eacf32df9a44059ca2ba4ac444604b00a19a6c1f0809943", + "sha256:a7594f9a010cdf1e16a58b3bf26c9da39bbf663e3b8d46d39176999d71816658", + "sha256:b08d00d889a118f68f37f3c43e359aab24ee29eb2e3fe96d64c6a2ba8b9d6557", + "sha256:cc9be041e428c10f8b6ab358c6b393648f9457094e1dcc11b4906026d43cd380", + "sha256:d5682eeb10cca0606c4a8286a3391d4c3c5a36f0c448e71b8bd05be4e1694bfb", + "sha256:dd38cad4599d58c23c01bca2102dcce30d61f6466328169bb17efb8a528f4d97", + "sha256:f8bfe0f77e373280f4878c8ed861c481eeadf62eb6d693e7aba4b059c1fd3c4c", + "sha256:fd3b8905e21431ad306eeaf56644a68fdd621bf8f3097eff54d0f6bdf7262065" + ], + "version": "==0.2.0" }, "humanfriendly": { "hashes": [ - "sha256:175ffa628aa76da2c17369a5da5856084562cc66dfe7f82ae93ca3ef175277a6", - "sha256:3c9ab8d28e88e6cc998e41963357736dafd555ee5bb666b50e42f6ce28dd3e3d" + "sha256:332da98c24cc150efcc91b5508b19115209272bfdf4b0764a56795932f854271", + "sha256:f7dba53ac7935fd0b4a2fc9a29e316ddd9ea135fb3052d3d0279d10c18ff9c48" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==9.0" + "version": "==9.2" + }, + "hyperlink": { + "hashes": [ + "sha256:427af957daa58bc909471c6c40f74c5450fa123dd093fc53efd2e91d2705a56b", + "sha256:e6b14c37ecb73e89c77d78cdb4c2cc8f3fb59a885c5b3f819ff4ed80f25af1b4" + ], + "version": "==21.0.0" + }, + "idna": { + "hashes": [ + "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a", + "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3" + ], + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", + "version": "==3.2" }, "imap-tools": { "hashes": [ - "sha256:72bf46dc135b039a5d5b59f4e079242ac15eac02a30038e8cb2dec7b153cab65", - "sha256:75dc1c72dd76d9e577df26a1e0ec3a809b5eebce77678851458dcd2eae127ac9" + "sha256:a4fafd33ed04dfaef48e8c1bd40b82d0e821d912d43a7ce96502029e9fc1da9e", + "sha256:b5f0dc2b4e45cf4a3724d0d845c218142e0cc01dbab220480dbc3c1cbf0bed69", + "sha256:f32284c3d55f17112b0e8db9a148106b391d5945277d9b89bd6bfd9712bdd6bf" ], "index": "pypi", - "version": "==0.33.0" + "version": "==0.46.0" }, "img2pdf": { "hashes": [ - "sha256:57905015579b1026acf1605aa95859cd79b051fa1c35485573d165526fc9dbb5", - "sha256:eaee690ab8403dd1a9cb4db10afee41dd3e6c7ed63bdace02a0121f9feadb0c9" + "sha256:38a1229ca84b211d7782d1d65ad7251a9781bf24f6f6497c738c755fcfed5552", + "sha256:e7eb85dc87062e308066f2b0a9dbfb3c5c0bd306dec0cf9130e3836c0547253a" ], - "version": "==0.4.0" + "version": "==0.4.1" }, - "importlib-metadata": { + "incremental": { "hashes": [ - "sha256:6112e21359ef8f344e7178aa5b72dc6e62b38b0d008e6d3cb212c5b84df72013", - "sha256:b0c2d3b226157ae4517d9625decf63591461c66b3a808c2666d538946519d170" + "sha256:02f5de5aff48f6b9f665d99d48bfc7ec03b6e3943210de7cfc88856d755d6f57", + "sha256:92014aebc6a20b78a8084cdd5645eeaa7f74b8933f70fa3ada2cfbd1e3b54321" ], - "markers": "python_version < '3.8'", - "version": "==3.1.1" + "version": "==21.3.0" }, "inotify-simple": { "hashes": [ @@ -270,206 +497,250 @@ }, "joblib": { "hashes": [ - "sha256:698c311779f347cf6b7e6b8a39bb682277b8ee4aba8cf9507bc0cf4cd4737b72", - "sha256:9e284edd6be6b71883a63c9b7f124738a3c16195513ad940eae7e3438de885d5" + "sha256:9c17567692206d2f3fb9ecf5e991084254fe631665c450b443761c4186a613f7", + "sha256:feeb1ec69c4d45129954f1b7034954241eedfd6ba39b5e9e4b6883be3332d5e5" ], "markers": "python_version >= '3.6'", - "version": "==0.17.0" + "version": "==1.0.1" }, "langdetect": { "hashes": [ - "sha256:363795ea005f1243c958e953245dac5d814fabdc025c9afa91588c5fa6b2fa83", - "sha256:ae53a024643df713274c297c0795dbfb5a16b329902f8e543e7b2d7d45f699e4", - "sha256:f37495e63607865e47deed08d78f7f8e58172658216ff954b2f14671bcd87740" + "sha256:7cbc0746252f19e76f77c0b1690aadf01963be835ef0cd4b56dddf2a8f1dfc2a", + "sha256:87963db181db734abf00b1c463df5cadb3f9edad8efb334d3e52d5ad008d9cc7", + "sha256:cbc1fef89f8d062739774bd51eda3da3274006b3661d199c2655f6b3f6d605a0" ], "index": "pypi", - "version": "==1.0.8" + "version": "==1.0.9" }, "lxml": { "hashes": [ - "sha256:0448576c148c129594d890265b1a83b9cd76fd1f0a6a04620753d9a6bcfd0a4d", - "sha256:127f76864468d6630e1b453d3ffbbd04b024c674f55cf0a30dc2595137892d37", - "sha256:1471cee35eba321827d7d53d104e7b8c593ea3ad376aa2df89533ce8e1b24a01", - "sha256:2363c35637d2d9d6f26f60a208819e7eafc4305ce39dc1d5005eccc4593331c2", - "sha256:2e5cc908fe43fe1aa299e58046ad66981131a66aea3129aac7770c37f590a644", - "sha256:2e6fd1b8acd005bd71e6c94f30c055594bbd0aa02ef51a22bbfa961ab63b2d75", - "sha256:366cb750140f221523fa062d641393092813b81e15d0e25d9f7c6025f910ee80", - "sha256:42ebca24ba2a21065fb546f3e6bd0c58c3fe9ac298f3a320147029a4850f51a2", - "sha256:4e751e77006da34643ab782e4a5cc21ea7b755551db202bc4d3a423b307db780", - "sha256:4fb85c447e288df535b17ebdebf0ec1cf3a3f1a8eba7e79169f4f37af43c6b98", - "sha256:50c348995b47b5a4e330362cf39fc503b4a43b14a91c34c83b955e1805c8e308", - "sha256:535332fe9d00c3cd455bd3dd7d4bacab86e2d564bdf7606079160fa6251caacf", - "sha256:535f067002b0fd1a4e5296a8f1bf88193080ff992a195e66964ef2a6cfec5388", - "sha256:5be4a2e212bb6aa045e37f7d48e3e1e4b6fd259882ed5a00786f82e8c37ce77d", - "sha256:60a20bfc3bd234d54d49c388950195d23a5583d4108e1a1d47c9eef8d8c042b3", - "sha256:648914abafe67f11be7d93c1a546068f8eff3c5fa938e1f94509e4a5d682b2d8", - "sha256:681d75e1a38a69f1e64ab82fe4b1ed3fd758717bed735fb9aeaa124143f051af", - "sha256:68a5d77e440df94011214b7db907ec8f19e439507a70c958f750c18d88f995d2", - "sha256:69a63f83e88138ab7642d8f61418cf3180a4d8cd13995df87725cb8b893e950e", - "sha256:6e4183800f16f3679076dfa8abf2db3083919d7e30764a069fb66b2b9eff9939", - "sha256:6fd8d5903c2e53f49e99359b063df27fdf7acb89a52b6a12494208bf61345a03", - "sha256:791394449e98243839fa822a637177dd42a95f4883ad3dec2a0ce6ac99fb0a9d", - "sha256:7a7669ff50f41225ca5d6ee0a1ec8413f3a0d8aa2b109f86d540887b7ec0d72a", - "sha256:7e9eac1e526386df7c70ef253b792a0a12dd86d833b1d329e038c7a235dfceb5", - "sha256:7ee8af0b9f7de635c61cdd5b8534b76c52cd03536f29f51151b377f76e214a1a", - "sha256:8246f30ca34dc712ab07e51dc34fea883c00b7ccb0e614651e49da2c49a30711", - "sha256:8c88b599e226994ad4db29d93bc149aa1aff3dc3a4355dd5757569ba78632bdf", - "sha256:91d6dace31b07ab47eeadd3f4384ded2f77b94b30446410cb2c3e660e047f7a7", - "sha256:923963e989ffbceaa210ac37afc9b906acebe945d2723e9679b643513837b089", - "sha256:94d55bd03d8671686e3f012577d9caa5421a07286dd351dfef64791cf7c6c505", - "sha256:97db258793d193c7b62d4e2586c6ed98d51086e93f9a3af2b2034af01450a74b", - "sha256:a9d6bc8642e2c67db33f1247a77c53476f3a166e09067c0474facb045756087f", - "sha256:cd11c7e8d21af997ee8079037fff88f16fda188a9776eb4b81c7e4c9c0a7d7fc", - "sha256:d8d3d4713f0c28bdc6c806a278d998546e8efc3498949e3ace6e117462ac0a5e", - "sha256:e0bfe9bb028974a481410432dbe1b182e8191d5d40382e5b8ff39cdd2e5c5931", - "sha256:e1dbb88a937126ab14d219a000728224702e0ec0fc7ceb7131c53606b7a76772", - "sha256:f4822c0660c3754f1a41a655e37cb4dbbc9be3d35b125a37fab6f82d47674ebc", - "sha256:f83d281bb2a6217cd806f4cf0ddded436790e66f393e124dfe9731f6b3fb9afe", - "sha256:fc37870d6716b137e80d19241d0e2cff7a7643b925dfa49b4c8ebd1295eb506e" + "sha256:079f3ae844f38982d156efce585bc540c16a926d4436712cf4baee0cce487a3d", + "sha256:0fbcf5565ac01dff87cbfc0ff323515c823081c5777a9fc7703ff58388c258c3", + "sha256:122fba10466c7bd4178b07dba427aa516286b846b2cbd6f6169141917283aae2", + "sha256:1822d16e5ac9061749955d099ff50c6cbc068f8cbfde28dfbeea845f7fbe6e09", + "sha256:1b38116b6e628118dea5b2186ee6820ab138dbb1e24a13e478490c7db2f326ae", + "sha256:1b7584d421d254ab86d4f0b13ec662a9014397678a7c4265a02a6d7c2b18a75f", + "sha256:26e761ab5b07adf5f555ee82fb4bfc35bf93750499c6c7614bd64d12aaa67927", + "sha256:289e9ca1a9287f08daaf796d96e06cb2bc2958891d7911ac7cae1c5f9e1e0ee3", + "sha256:2a9d50e69aac3ebee695424f7dbd7b8c6d6eb7de2a2eb6b0f6c7db6aa41e02b7", + "sha256:3082c518be8e97324390614dacd041bb1358c882d77108ca1957ba47738d9d59", + "sha256:33bb934a044cf32157c12bfcfbb6649807da20aa92c062ef51903415c704704f", + "sha256:3439c71103ef0e904ea0a1901611863e51f50b5cd5e8654a151740fde5e1cade", + "sha256:36108c73739985979bf302006527cf8a20515ce444ba916281d1c43938b8bb96", + "sha256:39b78571b3b30645ac77b95f7c69d1bffc4cf8c3b157c435a34da72e78c82468", + "sha256:4289728b5e2000a4ad4ab8da6e1db2e093c63c08bdc0414799ee776a3f78da4b", + "sha256:4bff24dfeea62f2e56f5bab929b4428ae6caba2d1eea0c2d6eb618e30a71e6d4", + "sha256:4c61b3a0db43a1607d6264166b230438f85bfed02e8cff20c22e564d0faff354", + "sha256:542d454665a3e277f76954418124d67516c5f88e51a900365ed54a9806122b83", + "sha256:5a0a14e264069c03e46f926be0d8919f4105c1623d620e7ec0e612a2e9bf1c04", + "sha256:5c8c163396cc0df3fd151b927e74f6e4acd67160d6c33304e805b84293351d16", + "sha256:64812391546a18896adaa86c77c59a4998f33c24788cadc35789e55b727a37f4", + "sha256:66e575c62792c3f9ca47cb8b6fab9e35bab91360c783d1606f758761810c9791", + "sha256:6f12e1427285008fd32a6025e38e977d44d6382cf28e7201ed10d6c1698d2a9a", + "sha256:74f69060f98a70e3654de3021cd8162859e0f30b9e69aed76ef2b7a2e3b84212", + "sha256:74f7d8d439b18fa4c385f3f5dfd11144bb87c1da034a466c5b5577d23a1d9b51", + "sha256:7610b8c31688f0b1be0ef882889817939490a36d0ee880ea562a4e1399c447a1", + "sha256:76fa7b1362d19f8fbd3e75fe2fb7c79359b0af8747e6f7141c338f0bee2f871a", + "sha256:7728e05c35412ba36d3e9795ae8995e3c86958179c9770e65558ec3fdfd3724f", + "sha256:8157dadbb09a34a6bd95a50690595e1fa0af1a99445e2744110e3dca7831c4ee", + "sha256:820628b7b3135403540202e60551e741f9b6d3304371712521be939470b454ec", + "sha256:884ab9b29feaca361f7f88d811b1eea9bfca36cf3da27768d28ad45c3ee6f969", + "sha256:89b8b22a5ff72d89d48d0e62abb14340d9e99fd637d046c27b8b257a01ffbe28", + "sha256:92e821e43ad382332eade6812e298dc9701c75fe289f2a2d39c7960b43d1e92a", + "sha256:b007cbb845b28db4fb8b6a5cdcbf65bacb16a8bd328b53cbc0698688a68e1caa", + "sha256:bc4313cbeb0e7a416a488d72f9680fffffc645f8a838bd2193809881c67dd106", + "sha256:bccbfc27563652de7dc9bdc595cb25e90b59c5f8e23e806ed0fd623755b6565d", + "sha256:c1a40c06fd5ba37ad39caa0b3144eb3772e813b5fb5b084198a985431c2f1e8d", + "sha256:c47ff7e0a36d4efac9fd692cfa33fbd0636674c102e9e8d9b26e1b93a94e7617", + "sha256:c4f05c5a7c49d2fb70223d0d5bcfbe474cf928310ac9fa6a7c6dddc831d0b1d4", + "sha256:cdaf11d2bd275bf391b5308f86731e5194a21af45fbaaaf1d9e8147b9160ea92", + "sha256:ce256aaa50f6cc9a649c51be3cd4ff142d67295bfc4f490c9134d0f9f6d58ef0", + "sha256:d2e35d7bf1c1ac8c538f88d26b396e73dd81440d59c1ef8522e1ea77b345ede4", + "sha256:d412de95c77655e4ff8d2042eef3deaadee87f95d7d47fc3e6f4e2d107df87df", + "sha256:d916d31fd85b2f78c76400d625076d9124de3e4bda8b016d25a050cc7d603f24", + "sha256:df7c53783a46febb0e70f6b05df2ba104610f2fb0d27023409734a3ecbb78fb2", + "sha256:e1cbd3f19a61e27e011e02f9600837b921ac661f0c40560eefb366e4e4fb275e", + "sha256:efac139c3f0bf4f0939f9375af4b02c5ad83a622de52d6dfa8e438e8e01d0eb0", + "sha256:efd7a09678fd8b53117f6bae4fa3825e0a22b03ef0a932e070c0bdbb3a35e654", + "sha256:f2380a6376dfa090227b663f9678150ef27543483055cc327555fb592c5967e2", + "sha256:f8380c03e45cf09f8557bdaa41e1fa7c81f3ae22828e1db470ab2a6c96d8bc23", + "sha256:f90ba11136bfdd25cae3951af8da2e95121c9b9b93727b1b896e3fa105b2f586" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==4.6.2" + "version": "==4.6.3" + }, + "msgpack": { + "hashes": [ + "sha256:0cb94ee48675a45d3b86e61d13c1e6f1696f0183f0715544976356ff86f741d9", + "sha256:1026dcc10537d27dd2d26c327e552f05ce148977e9d7b9f1718748281b38c841", + "sha256:26a1759f1a88df5f1d0b393eb582ec022326994e311ba9c5818adc5374736439", + "sha256:2a5866bdc88d77f6e1370f82f2371c9bc6fc92fe898fa2dec0c5d4f5435a2694", + "sha256:31c17bbf2ae5e29e48d794c693b7ca7a0c73bd4280976d408c53df421e838d2a", + "sha256:497d2c12426adcd27ab83144057a705efb6acc7e85957a51d43cdcf7f258900f", + "sha256:5a9ee2540c78659a1dd0b110f73773533ee3108d4e1219b5a15a8d635b7aca0e", + "sha256:7307e86f7ce75b49e65b55660b10b258e9e7b5e0f80d31d7a86a278d8204d1b4", + "sha256:8521e5be9e3b93d4d5e07cb80b7e32353264d143c1f072309e1863174c6aadb1", + "sha256:87869ba567fe371c4555d2e11e4948778ab6b59d6cc9d8460d543e4cfbbddd1c", + "sha256:8ffb24a3b7518e843cd83538cf859e026d24ec41ac5721c18ed0c55101f9775b", + "sha256:92be4b12de4806d3c36810b0fe2aeedd8d493db39e2eb90742b9c09299eb5759", + "sha256:9ea52fff0473f9f3000987f313310208c879493491ef3ccf66268eff8d5a0326", + "sha256:a4355d2193106c7aa77c98fc955252a737d8550320ecdb2e9ac701e15e2943bc", + "sha256:a99b144475230982aee16b3d249170f1cccebf27fb0a08e9f603b69637a62192", + "sha256:ac25f3e0513f6673e8b405c3a80500eb7be1cf8f57584be524c4fa78fe8e0c83", + "sha256:b28c0876cce1466d7c2195d7658cf50e4730667196e2f1355c4209444717ee06", + "sha256:b55f7db883530b74c857e50e149126b91bb75d35c08b28db12dcb0346f15e46e", + "sha256:b6d9e2dae081aa35c44af9c4298de4ee72991305503442a5c74656d82b581fe9", + "sha256:c747c0cc08bd6d72a586310bda6ea72eeb28e7505990f342552315b229a19b33", + "sha256:c82dc0ba34d620fb94d12a7725e9362958bb1be3938688a061f53ed86bee005a", + "sha256:d6c64601af8f3893d17ec233237030e3110f11b8a962cb66720bf70c0141aa54", + "sha256:d8167b84af26654c1124857d71650404336f4eb5cc06900667a493fc619ddd9f", + "sha256:d9c1976457ee552468a89315275e917f2997f9bddf219451888ccb425cfbf96c", + "sha256:de6bd7990a2c2dabe926b7e62a92886ccbf809425c347ae7de277067f97c2887", + "sha256:e36a812ef4705a291cdb4a2fd352f013134f26c6ff63477f20235138d1d21009", + "sha256:e89ec55871ed5473a041c0495b7b4e6099f6263438e0bd04ccd8418f92d5d7f2", + "sha256:f3e6aaf217ac1c7ce1563cf52a2f4f5d5b1f64e8729d794165db71da57257f0c", + "sha256:f484cd2dca68502de3704f056fa9b318c94b1539ed17a4c784266df5d6978c87", + "sha256:fae04496f5bc150eefad4e9571d1a76c55d021325dcd484ce45065ebbdd00984", + "sha256:fe07bc6735d08e492a327f496b7850e98cb4d112c56df69b0c844dbebcbb47f6" + ], + "version": "==1.0.2" }, "numpy": { "hashes": [ - "sha256:08308c38e44cc926bdfce99498b21eec1f848d24c302519e64203a8da99a97db", - "sha256:09c12096d843b90eafd01ea1b3307e78ddd47a55855ad402b157b6c4862197ce", - "sha256:13d166f77d6dc02c0a73c1101dd87fdf01339febec1030bd810dcd53fff3b0f1", - "sha256:141ec3a3300ab89c7f2b0775289954d193cc8edb621ea05f99db9cb181530512", - "sha256:16c1b388cc31a9baa06d91a19366fb99ddbe1c7b205293ed072211ee5bac1ed2", - "sha256:18bed2bcb39e3f758296584337966e68d2d5ba6aab7e038688ad53c8f889f757", - "sha256:1aeef46a13e51931c0b1cf8ae1168b4a55ecd282e6688fdb0a948cc5a1d5afb9", - "sha256:27d3f3b9e3406579a8af3a9f262f5339005dd25e0ecf3cf1559ff8a49ed5cbf2", - "sha256:2a2740aa9733d2e5b2dfb33639d98a64c3b0f24765fed86b0fd2aec07f6a0a08", - "sha256:4377e10b874e653fe96985c05feed2225c912e328c8a26541f7fc600fb9c637b", - "sha256:448ebb1b3bf64c0267d6b09a7cba26b5ae61b6d2dbabff7c91b660c7eccf2bdb", - "sha256:50e86c076611212ca62e5a59f518edafe0c0730f7d9195fec718da1a5c2bb1fc", - "sha256:5734bdc0342aba9dfc6f04920988140fb41234db42381cf7ccba64169f9fe7ac", - "sha256:5ddd1dfa2be066595c1993165b4cae84b9866b12339d0c903db7f21a094324a3", - "sha256:64324f64f90a9e4ef732be0928be853eee378fd6a01be21a0a8469c4f2682c83", - "sha256:6ae6c680f3ebf1cf7ad1d7748868b39d9f900836df774c453c11c5440bc15b36", - "sha256:6d7593a705d662be5bfe24111af14763016765f43cb6923ed86223f965f52387", - "sha256:8cac8790a6b1ddf88640a9267ee67b1aee7a57dfa2d2dd33999d080bc8ee3a0f", - "sha256:8ece138c3a16db8c1ad38f52eb32be6086cc72f403150a79336eb2045723a1ad", - "sha256:9eeb7d1d04b117ac0d38719915ae169aa6b61fca227b0b7d198d43728f0c879c", - "sha256:a09f98011236a419ee3f49cedc9ef27d7a1651df07810ae430a6b06576e0b414", - "sha256:a5d897c14513590a85774180be713f692df6fa8ecf6483e561a6d47309566f37", - "sha256:ad6f2ff5b1989a4899bf89800a671d71b1612e5ff40866d1f4d8bcf48d4e5764", - "sha256:c42c4b73121caf0ed6cd795512c9c09c52a7287b04d105d112068c1736d7c753", - "sha256:cb1017eec5257e9ac6209ac172058c430e834d5d2bc21961dceeb79d111e5909", - "sha256:d6c7bb82883680e168b55b49c70af29b84b84abb161cbac2800e8fcb6f2109b6", - "sha256:e452dc66e08a4ce642a961f134814258a082832c78c90351b75c41ad16f79f63", - "sha256:e5b6ed0f0b42317050c88022349d994fe72bfe35f5908617512cd8c8ef9da2a9", - "sha256:e9b30d4bd69498fc0c3fe9db5f62fffbb06b8eb9321f92cc970f2969be5e3949", - "sha256:ec149b90019852266fec2341ce1db513b843e496d5a8e8cdb5ced1923a92faab", - "sha256:edb01671b3caae1ca00881686003d16c2209e07b7ef8b7639f1867852b948f7c", - "sha256:f0d3929fe88ee1c155129ecd82f981b8856c5d97bcb0d5f23e9b4242e79d1de3", - "sha256:f29454410db6ef8126c83bd3c968d143304633d45dc57b51252afbd79d700893", - "sha256:fe45becb4c2f72a0907c1d0246ea6449fe7a9e2293bb0e11c4e9a32bb0930a15", - "sha256:fedbd128668ead37f33917820b704784aff695e0019309ad446a6d0b065b57e4" + "sha256:1676b0a292dd3c99e49305a16d7a9f42a4ab60ec522eac0d3dd20cdf362ac010", + "sha256:16f221035e8bd19b9dc9a57159e38d2dd060b48e93e1d843c49cb370b0f415fd", + "sha256:43909c8bb289c382170e0282158a38cf306a8ad2ff6dfadc447e90f9961bef43", + "sha256:4c020adbaa398721cd5e30c5aab52b505efcd35c3b0156e800245c1274c96962", + "sha256:4e465afc3b96dbc80cf4a5273e5e2b1e3451286361b4af70ce1adb2984d392f9", + "sha256:55b745fca0a5ab738647d0e4db099bd0a23279c32b31a783ad2ccea729e632df", + "sha256:5d050e1e4bc9ddb8656d7b4f414557720ddcca23a5b88dd7cff65e847864c400", + "sha256:637d827248f447e63585ca3f4a7d2dfaa882e094df6cfa177cc9cf9cd6cdf6d2", + "sha256:6690080810f77485667bfbff4f69d717c3be25e5b11bb2073e76bb3f578d99b4", + "sha256:66fbc6fed94a13b9801fb70b96ff30605ab0a123e775a5e7a26938b717c5d71a", + "sha256:67d44acb72c31a97a3d5d33d103ab06d8ac20770e1c5ad81bdb3f0c086a56cf6", + "sha256:6ca2b85a5997dabc38301a22ee43c82adcb53ff660b89ee88dded6b33687e1d8", + "sha256:6e51534e78d14b4a009a062641f465cfaba4fdcb046c3ac0b1f61dd97c861b1b", + "sha256:70eb5808127284c4e5c9e836208e09d685a7978b6a216db85960b1a112eeace8", + "sha256:830b044f4e64a76ba71448fce6e604c0fc47a0e54d8f6467be23749ac2cbd2fb", + "sha256:8b7bb4b9280da3b2856cb1fc425932f46fba609819ee1c62256f61799e6a51d2", + "sha256:a9c65473ebc342715cb2d7926ff1e202c26376c0dcaaee85a1fd4b8d8c1d3b2f", + "sha256:c1c09247ccea742525bdb5f4b5ceeacb34f95731647fe55774aa36557dbb5fa4", + "sha256:c5bf0e132acf7557fc9bb8ded8b53bbbbea8892f3c9a1738205878ca9434206a", + "sha256:d840d80623c96696fab0d15489f49425bec1c0f47f3bf0b3362ce0c75dbad993", + "sha256:db250fd3e90117e0312b611574cd1b3f78bec046783195075cbd7ba9c3d73f16", + "sha256:e515c9a93aebe27166ec9593411c58494fa98e5fcc219e47260d9ab8a1cc7f9f", + "sha256:e55185e51b18d788e49fe8305fd73ef4470596b33fc2c1ceb304566b99c71a69", + "sha256:ea9cff01e75a956dbee133fa8e5b68f2f92175233de2f88de3a682dd94deda65", + "sha256:f1452578d0516283c87608a5a5548b0cdde15b99650efdfd85182102ef7a7c17", + "sha256:f39a995e47cb8649673cfa0579fbdd1cdd33ea497d1728a6cb194d6252268e48" ], - "markers": "python_version >= '3.6'", - "version": "==1.19.4" + "index": "pypi", + "version": "==1.20.3" }, "ocrmypdf": { "hashes": [ - "sha256:91e7394172cedb3be801a229dbd3d308fb5ae80cbc3a77879fa7954beea407b1", - "sha256:e550b8e884150accab7ea41f4a576b5844594cb5cbd6ed514fbf1206720343ad" + "sha256:2d80b0372f869a5f05a73db0cdf14299bf0314faa66f1519a4a7d9c596a720f9", + "sha256:bcf9bdc12707bb10ec4a1fae09461c3e7a4d00ab2696a7ab6c386d774c4e0665", + "sha256:e041aadce7f1b5fa67f2fdbd26b93ce41ddeb78695d5a9cec41bb2faaa920642" ], "index": "pypi", - "version": "==11.3.4" - }, - "pathtools": { - "hashes": [ - "sha256:7c35c5421a39bb82e58018febd90e3b6e5db34c5443aaaf742b3f33d4655f1c0", - "sha256:d77d982475e87f32b82157a43b09f0a5ef3e66c1d8f3c7eb8d2580e783cd8202" - ], - "version": "==0.1.2" + "version": "==12.3.2" }, "pathvalidate": { "hashes": [ - "sha256:1697c8ea71ff4c48e7aa0eda72fe4581404be8f41e51a17363ef682dd6824d35", - "sha256:32d30dbacb711c16bb188b12ce7e9a46b41785f50a12f64500f747480a4b6ee3" + "sha256:3c9bd94c7ec23e9cfb211ffbe356ae75f979d6c099a2c745ee9490f524f32468", + "sha256:f5dde7efeeb4262784c5e1331e02752d07c1ec3ee5ea42683fe211155652b808" ], "index": "pypi", - "version": "==2.3.0" + "version": "==2.4.1" }, "pdfminer.six": { "hashes": [ "sha256:b9aac0ebeafb21c08bf65f2039f4b2c5f78a3449d0a41df711d72445649e952a", "sha256:d78877ba8d8bf957f3bb636c4f73f4f6f30f56c461993877ac22c39c20837509" ], - "markers": "python_version >= '3.4'", - "version": "==20201018" - }, - "pdftotext": { - "hashes": [ - "sha256:98aeb8b07a4127e1a30223bd933ef080bbd29aa88f801717ca6c5618380b8aa6" - ], "index": "pypi", - "version": "==2.1.5" + "version": "==20201018" }, "pikepdf": { "hashes": [ - "sha256:0829bd5dacd73bb4a37e7575bae523f49603479755563c92ddb55c206700cab1", - "sha256:0d2b631077cd6af6e4d1b396208020705842610a6f13fab489d5f9c47916baa2", - "sha256:21c98af08fae4ac9fbcad02b613b6768a4ca300fda4cba867f4a4b6f73c2d04b", - "sha256:2240372fed30124ddc35b0c15a613f2b687a426ea2f150091e0a0c58cca7a495", - "sha256:2a97f5f1403e058d217d7f6861cf51fca200c5687bce0d052f5f2fa89b5bfa22", - "sha256:3faaefca0ae80d19891acec8b0dd5e6235f59f2206d82375eb80d090285e9557", - "sha256:48ef45b64882901c0d69af3b85d16a19bd0f3e95b43e614fefb53521d8caf36c", - "sha256:5212fe41f2323fc7356ba67caa39737fe13080562cff37bcbb74a8094076c8d0", - "sha256:56859c32170663c57bd0658189ce44e180533eebe813853446cd6413810be9eb", - "sha256:5f8fd1cb3478c5534222018aca24fbbd2bc74460c899bda988ec76722c13caa9", - "sha256:74300a32c41b3d578772f6933f23a88b19f74484185e71e5225ce2f7ea5aea78", - "sha256:8cbc946bdd217148f4a9c029fcea62f4ae0f67d5346de4c865f4718cd0ddc37f", - "sha256:9ceefd30076f732530cf84a1be2ecb2fa9931af932706ded760a6d37c73b96ad", - "sha256:ad69c170fda41b07a4c6b668a3128e7a759f50d9aebcfcde0ccff1358abe0423", - "sha256:b715fe182189fb6870fab5b0383bb2fb278c88c46eade346b0f4c1ed8818c09d", - "sha256:bb01ecf95083ffcb9ad542dc5342ccc1059e46f1395fd966629d36d9cc766b4a", - "sha256:bd6328547219cf48cefb4e0a1bc54442910594de1c5a5feae847d9ff3c629031", - "sha256:edb128379bb1dea76b5bdbdacf5657a6e4754bacc2049640762725590d8ed905", - "sha256:f8e687900557fcd4c51b4e72b9e337fdae9e2c81049d1d80b624bb2e88b5769d", - "sha256:fe0ca120e3347c851c34a91041d574f3c588d832023906d8ae18d66d042e8a52", - "sha256:fe8e0152672f24d8bfdecc725f97e9013f2de1b41849150959526ca3562bd3ef" + "sha256:0517ea762eb2e450557827915a47fb37e768ea81d22f66d50b48d6eab56fd15b", + "sha256:06a4cba6b630cbfa851a56b815ca957ef7263c0b31e2ffc62d492c834c9a7bca", + "sha256:084a6e83d3ac9a804e36812cba5c9b321f7f3cab56e22a89ea080e460a99feed", + "sha256:0b45b602e1735543ca65f93cda6714481862a57bcf629a3b404ac0efa280675a", + "sha256:13b14bf92f020356a676503a11c62db714e8dfd81eed4fd062dd19f9ff5752b4", + "sha256:13c30f2acf558b3c290e8ab908d5d3d3a788fef347b5a80ae6698baa6660ff5c", + "sha256:19a14e4d7b0a10c64143b5cddc0f2e464a614bbd4ba9f88f1081ed65d7c9e9ac", + "sha256:1f6655d5ced542b1132f76b1c8df27b659ed7ebe02da2c3758be61d6036d755e", + "sha256:3da160d68d3aaf9197306cc06c423261c7b150778b86ff143d67b4698cb6366e", + "sha256:41bbdd5db3eae8177752c89ec2dc7e5daae3a92b408e0c8593f945baf225f879", + "sha256:51e25e2d4bcf8e6b35a84edce3244f8a8b7b421f976074a1f39d9cafcf238e79", + "sha256:5ca94ccf5894fb435af8d97e1115a48ea24185519d16af8bfade8274d7420f43", + "sha256:6b5e9039bcf850927354a8c52e0eb2d89783072554767d8e00464684d81e928e", + "sha256:6f6f81045009935612c4422fbb0b65f63b11338d2c49db7ca6f03a7ed462c75a", + "sha256:7163bf60fe0253d82fb1b01d34ff3ec7e5da35d48b30694f5c65dd72c07b744a", + "sha256:7634f9b2c4948bff96af033d604d8f760ad7ac82d2ec942d24413496a7b71d0e", + "sha256:78c9e740f5fb0a4aae7846647b5039a01098e1f39e68384a71967b7480ab8ebc", + "sha256:94e97bdf90aff307b0d7dd839070a7d121d43da297887c3bd4580d34a7ae2166", + "sha256:a8f1ba7fe9113e5c4ab6e2775a8aceb33a8459ee95766085a47441b21d484b58", + "sha256:ae019cb866405e7ec8ae1a9cef1ecacb12c57d9fc0b6a9171d89046e6de8d8b0", + "sha256:c1cbbdb3d777813c24a166e192cb81d5a66e4e0e5737a04fea8fcccb8793a6aa", + "sha256:c1d47d8fc34c3f3adbcf89a4e636b99031cbcac8798ac9f0e84f028ae83da760", + "sha256:e24dff6af31f1eb732fcb5db4678835d1f312643996fdcd9dbeb8aca52bc0dde", + "sha256:e333282b99887fbdf29ea693c09c8123482c88d76918f94a8026d8f86ec5b282", + "sha256:ef43c1a1aeb4355c4c1759da7105b95aa80d9981d880ba90cba80c54d1e4b026", + "sha256:f6f0b7d2db3cfa772e490b02fe184855c150e22c8a5f773e9a4057e71ee4e8b6", + "sha256:ffd6efb681b681e021eeeb6463df8f993bd86120ef855e3c421e0a1505744826" ], "index": "pypi", - "version": "==2.2.0" + "version": "==2.16.1" }, "pillow": { "hashes": [ - "sha256:006de60d7580d81f4a1a7e9f0173dc90a932e3905cc4d47ea909bc946302311a", - "sha256:0a2e8d03787ec7ad71dc18aec9367c946ef8ef50e1e78c71f743bc3a770f9fae", - "sha256:0eeeae397e5a79dc088d8297a4c2c6f901f8fb30db47795113a4a605d0f1e5ce", - "sha256:11c5c6e9b02c9dac08af04f093eb5a2f84857df70a7d4a6a6ad461aca803fb9e", - "sha256:2fb113757a369a6cdb189f8df3226e995acfed0a8919a72416626af1a0a71140", - "sha256:4b0ef2470c4979e345e4e0cc1bbac65fda11d0d7b789dbac035e4c6ce3f98adb", - "sha256:59e903ca800c8cfd1ebe482349ec7c35687b95e98cefae213e271c8c7fffa021", - "sha256:5a3342d34289715928c914ee7f389351eb37fa4857caa9297fc7948f2ed3e53d", - "sha256:5abd653a23c35d980b332bc0431d39663b1709d64142e3652890df4c9b6970f6", - "sha256:5f9403af9c790cc18411ea398a6950ee2def2a830ad0cfe6dc9122e6d528b302", - "sha256:6b4a8fd632b4ebee28282a9fef4c341835a1aa8671e2770b6f89adc8e8c2703c", - "sha256:6c1aca8231625115104a06e4389fcd9ec88f0c9befbabd80dc206c35561be271", - "sha256:795e91a60f291e75de2e20e6bdd67770f793c8605b553cb6e4387ce0cb302e09", - "sha256:7ba0ba61252ab23052e642abdb17fd08fdcfdbbf3b74c969a30c58ac1ade7cd3", - "sha256:7c9401e68730d6c4245b8e361d3d13e1035cbc94db86b49dc7da8bec235d0015", - "sha256:81f812d8f5e8a09b246515fac141e9d10113229bc33ea073fec11403b016bcf3", - "sha256:895d54c0ddc78a478c80f9c438579ac15f3e27bf442c2a9aa74d41d0e4d12544", - "sha256:8de332053707c80963b589b22f8e0229f1be1f3ca862a932c1bcd48dafb18dd8", - "sha256:92c882b70a40c79de9f5294dc99390671e07fc0b0113d472cbea3fde15db1792", - "sha256:95edb1ed513e68bddc2aee3de66ceaf743590bf16c023fb9977adc4be15bd3f0", - "sha256:b63d4ff734263ae4ce6593798bcfee6dbfb00523c82753a3a03cbc05555a9cc3", - "sha256:bd7bf289e05470b1bc74889d1466d9ad4a56d201f24397557b6f65c24a6844b8", - "sha256:cc3ea6b23954da84dbee8025c616040d9aa5eaf34ea6895a0a762ee9d3e12e11", - "sha256:cc9ec588c6ef3a1325fa032ec14d97b7309db493782ea8c304666fb10c3bd9a7", - "sha256:d3d07c86d4efa1facdf32aa878bd508c0dc4f87c48125cc16b937baa4e5b5e11", - "sha256:d8a96747df78cda35980905bf26e72960cba6d355ace4780d4bdde3b217cdf1e", - "sha256:e38d58d9138ef972fceb7aeec4be02e3f01d383723965bfcef14d174c8ccd039", - "sha256:eb472586374dc66b31e36e14720747595c2b265ae962987261f044e5cce644b5", - "sha256:fbd922f702582cb0d71ef94442bfca57624352622d75e3be7a1e7e9360b07e72" + "sha256:0b2efa07f69dc395d95bb9ef3299f4ca29bcb2157dc615bae0b42c3c20668ffc", + "sha256:114f816e4f73f9ec06997b2fde81a92cbf0777c9e8f462005550eed6bae57e63", + "sha256:147bd9e71fb9dcf08357b4d530b5167941e222a6fd21f869c7911bac40b9994d", + "sha256:15a2808e269a1cf2131930183dcc0419bc77bb73eb54285dde2706ac9939fa8e", + "sha256:196560dba4da7a72c5e7085fccc5938ab4075fd37fe8b5468869724109812edd", + "sha256:1c03e24be975e2afe70dfc5da6f187eea0b49a68bb2b69db0f30a61b7031cee4", + "sha256:1fd5066cd343b5db88c048d971994e56b296868766e461b82fa4e22498f34d77", + "sha256:29c9569049d04aaacd690573a0398dbd8e0bf0255684fee512b413c2142ab723", + "sha256:2b6dfa068a8b6137da34a4936f5a816aba0ecc967af2feeb32c4393ddd671cba", + "sha256:2cac53839bfc5cece8fdbe7f084d5e3ee61e1303cccc86511d351adcb9e2c792", + "sha256:2ee77c14a0299d0541d26f3d8500bb57e081233e3fa915fa35abd02c51fa7fae", + "sha256:37730f6e68bdc6a3f02d2079c34c532330d206429f3cee651aab6b66839a9f0e", + "sha256:3f08bd8d785204149b5b33e3b5f0ebbfe2190ea58d1a051c578e29e39bfd2367", + "sha256:479ab11cbd69612acefa8286481f65c5dece2002ffaa4f9db62682379ca3bb77", + "sha256:4bc3c7ef940eeb200ca65bd83005eb3aae8083d47e8fcbf5f0943baa50726856", + "sha256:660a87085925c61a0dcc80efb967512ac34dbb256ff7dd2b9b4ee8dbdab58cf4", + "sha256:67b3666b544b953a2777cb3f5a922e991be73ab32635666ee72e05876b8a92de", + "sha256:70af7d222df0ff81a2da601fab42decb009dc721545ed78549cb96e3a1c5f0c8", + "sha256:75e09042a3b39e0ea61ce37e941221313d51a9c26b8e54e12b3ececccb71718a", + "sha256:8960a8a9f4598974e4c2aeb1bff9bdd5db03ee65fd1fce8adf3223721aa2a636", + "sha256:9364c81b252d8348e9cc0cb63e856b8f7c1b340caba6ee7a7a65c968312f7dab", + "sha256:969cc558cca859cadf24f890fc009e1bce7d7d0386ba7c0478641a60199adf79", + "sha256:9a211b663cf2314edbdb4cf897beeb5c9ee3810d1d53f0e423f06d6ebbf9cd5d", + "sha256:a17ca41f45cf78c2216ebfab03add7cc350c305c38ff34ef4eef66b7d76c5229", + "sha256:a2f381932dca2cf775811a008aa3027671ace723b7a38838045b1aee8669fdcf", + "sha256:a4eef1ff2d62676deabf076f963eda4da34b51bc0517c70239fafed1d5b51500", + "sha256:c088a000dfdd88c184cc7271bfac8c5b82d9efa8637cd2b68183771e3cf56f04", + "sha256:c0e0550a404c69aab1e04ae89cca3e2a042b56ab043f7f729d984bf73ed2a093", + "sha256:c11003197f908878164f0e6da15fce22373ac3fc320cda8c9d16e6bba105b844", + "sha256:c2a5ff58751670292b406b9f06e07ed1446a4b13ffced6b6cab75b857485cbc8", + "sha256:c35d09db702f4185ba22bb33ef1751ad49c266534339a5cebeb5159d364f6f82", + "sha256:c379425c2707078dfb6bfad2430728831d399dc95a7deeb92015eb4c92345eaf", + "sha256:cc866706d56bd3a7dbf8bac8660c6f6462f2f2b8a49add2ba617bc0c54473d83", + "sha256:d0da39795049a9afcaadec532e7b669b5ebbb2a9134576ebcc15dd5bdae33cc0", + "sha256:dce0a6e85ddc74cefec738d9939befb0ecd78560cca8d0dd1e95ae8533127c9d", + "sha256:f05762fbb40cf686b8c3f04cf99c9180151fad5c585c610b2f20dbd9c3663ec1", + "sha256:f156d6ecfc747ee111c167f8faf5f4953761b5e66e91a4e6767e548d0f80129c", + "sha256:f4ebde71785f8bceb39dcd1e7f06bcc5d5c3cf48b9f69ab52636309387b097c8", + "sha256:fc214a6b75d2e0ea7745488da7da3c381f41790812988c7a92345978414fad37", + "sha256:fd7eef578f5b2200d066db1b50c4aa66410786201669fb76d5238b007918fb24", + "sha256:ff04c373477723430dce2e9d024c708a047d44cf17166bf16e604b379bf0ca14" ], "index": "pypi", - "version": "==8.0.1" + "version": "==8.3.1" }, "pluggy": { "hashes": [ @@ -479,48 +750,86 @@ "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", "version": "==0.13.1" }, + "portalocker": { + "hashes": [ + "sha256:1d43fd8223e1743f5725e8910f69e7a45858ffd298e19252633ac903eafd83bc", + "sha256:4e913d807aa6598c320e8a50c50e2ee0602bc45240b485e3f8bc06f13060084c" + ], + "markers": "python_version >= '3'", + "version": "==2.3.0" + }, "psycopg2-binary": { "hashes": [ - "sha256:0deac2af1a587ae12836aa07970f5cb91964f05a7c6cdb69d8425ff4c15d4e2c", - "sha256:0e4dc3d5996760104746e6cfcdb519d9d2cd27c738296525d5867ea695774e67", - "sha256:11b9c0ebce097180129e422379b824ae21c8f2a6596b159c7659e2e5a00e1aa0", - "sha256:15978a1fbd225583dd8cdaf37e67ccc278b5abecb4caf6b2d6b8e2b948e953f6", - "sha256:1fabed9ea2acc4efe4671b92c669a213db744d2af8a9fc5d69a8e9bc14b7a9db", - "sha256:2dac98e85565d5688e8ab7bdea5446674a83a3945a8f416ad0110018d1501b94", - "sha256:42ec1035841b389e8cc3692277a0bd81cdfe0b65d575a2c8862cec7a80e62e52", - "sha256:6422f2ff0919fd720195f64ffd8f924c1395d30f9a495f31e2392c2efafb5056", - "sha256:6a32f3a4cb2f6e1a0b15215f448e8ce2da192fd4ff35084d80d5e39da683e79b", - "sha256:7312e931b90fe14f925729cde58022f5d034241918a5c4f9797cac62f6b3a9dd", - "sha256:7d92a09b788cbb1aec325af5fcba9fed7203897bbd9269d5691bb1e3bce29550", - "sha256:833709a5c66ca52f1d21d41865a637223b368c0ee76ea54ca5bad6f2526c7679", - "sha256:89705f45ce07b2dfa806ee84439ec67c5d9a0ef20154e0e475e2b2ed392a5b83", - "sha256:8cd0fb36c7412996859cb4606a35969dd01f4ea34d9812a141cd920c3b18be77", - "sha256:950bc22bb56ee6ff142a2cb9ee980b571dd0912b0334aa3fe0fe3788d860bea2", - "sha256:a0c50db33c32594305b0ef9abc0cb7db13de7621d2cadf8392a1d9b3c437ef77", - "sha256:a0eb43a07386c3f1f1ebb4dc7aafb13f67188eab896e7397aa1ee95a9c884eb2", - "sha256:aaa4213c862f0ef00022751161df35804127b78adf4a2755b9f991a507e425fd", - "sha256:ac0c682111fbf404525dfc0f18a8b5f11be52657d4f96e9fcb75daf4f3984859", - "sha256:ad20d2eb875aaa1ea6d0f2916949f5c08a19c74d05b16ce6ebf6d24f2c9f75d1", - "sha256:b4afc542c0ac0db720cf516dd20c0846f71c248d2b3d21013aa0d4ef9c71ca25", - "sha256:b8a3715b3c4e604bcc94c90a825cd7f5635417453b253499664f784fc4da0152", - "sha256:ba28584e6bca48c59eecbf7efb1576ca214b47f05194646b081717fa628dfddf", - "sha256:ba381aec3a5dc29634f20692349d73f2d21f17653bda1decf0b52b11d694541f", - "sha256:bd1be66dde2b82f80afb9459fc618216753f67109b859a361cf7def5c7968729", - "sha256:c2507d796fca339c8fb03216364cca68d87e037c1f774977c8fc377627d01c71", - "sha256:cec7e622ebc545dbb4564e483dd20e4e404da17ae07e06f3e780b2dacd5cee66", - "sha256:d14b140a4439d816e3b1229a4a525df917d6ea22a0771a2a78332273fd9528a4", - "sha256:d1b4ab59e02d9008efe10ceabd0b31e79519da6fb67f7d8e8977118832d0f449", - "sha256:d5227b229005a696cc67676e24c214740efd90b148de5733419ac9aaba3773da", - "sha256:d9f3a909b59ac4a3ca9beb77716f4bce627276edb039a71d4e9ec4b7548536a0", - "sha256:e1f57aa70d3f7cc6947fd88636a481638263ba04a742b4a37dd25c373e41491a", - "sha256:e74a55f6bad0e7d3968399deb50f61f4db1926acf4a6d83beaaa7df986f48b1c", - "sha256:e7f5a465c6431c0ad8d4e69603ee3306e521a09d3c6af76a16bdb62946bdddf0", - "sha256:e82aba2188b9ba309fd8e271702bd0d0fc9148ae3150532bbb474f4590039ffb", - "sha256:ee69dad2c7155756ad114c02db06002f4cded41132cc51378e57aad79cc8e4f4", - "sha256:f5ab93a2cb2d8338b1674be43b442a7f544a0971da062a5da774ed40587f18f5" + "sha256:0b7dae87f0b729922e06f85f667de7bf16455d411971b2043bbd9577af9d1975", + "sha256:0f2e04bd2a2ab54fa44ee67fe2d002bb90cee1c0f1cc0ebc3148af7b02034cbd", + "sha256:123c3fb684e9abfc47218d3784c7b4c47c8587951ea4dd5bc38b6636ac57f616", + "sha256:1473c0215b0613dd938db54a653f68251a45a78b05f6fc21af4326f40e8360a2", + "sha256:14db1752acdd2187d99cb2ca0a1a6dfe57fc65c3281e0f20e597aac8d2a5bd90", + "sha256:1e3a362790edc0a365385b1ac4cc0acc429a0c0d662d829a50b6ce743ae61b5a", + "sha256:1e85b74cbbb3056e3656f1cc4781294df03383127a8114cbc6531e8b8367bf1e", + "sha256:20f1ab44d8c352074e2d7ca67dc00843067788791be373e67a0911998787ce7d", + "sha256:2f62c207d1740b0bde5c4e949f857b044818f734a3d57f1d0d0edc65050532ed", + "sha256:3242b9619de955ab44581a03a64bdd7d5e470cc4183e8fcadd85ab9d3756ce7a", + "sha256:35c4310f8febe41f442d3c65066ca93cccefd75013df3d8c736c5b93ec288140", + "sha256:38fa2413b60eba2a0b30efda083d3efa52e22dde530679665985e2b8244cb553", + "sha256:4235f9d5ddcab0b8dbd723dca56ea2922b485ea00e1dafacf33b0c7e840b3d32", + "sha256:5ced67f1e34e1a450cdb48eb53ca73b60aa0af21c46b9b35ac3e581cf9f00e31", + "sha256:7360647ea04db2e7dff1648d1da825c8cf68dc5fbd80b8fb5b3ee9f068dcd21a", + "sha256:8c13d72ed6af7fd2c8acbd95661cf9477f94e381fce0792c04981a8283b52917", + "sha256:988b47ac70d204aed01589ed342303da7c4d84b56c2f4c4b8b00deda123372bf", + "sha256:995fc41ebda5a7a663a254a1dcac52638c3e847f48307b5416ee373da15075d7", + "sha256:a36c7eb6152ba5467fb264d73844877be8b0847874d4822b7cf2d3c0cb8cdcb0", + "sha256:a580f04ad7a67c082a2580a129cd011e94d379e36797941892ccff7e9b5bc2ee", + "sha256:aed4a9a7e3221b3e252c39d0bf794c438dc5453bc2963e8befe9d4cd324dff72", + "sha256:aef9aee84ec78af51107181d02fe8773b100b01c5dfde351184ad9223eab3698", + "sha256:b0221ca5a9837e040ebf61f48899926b5783668b7807419e4adae8175a31f773", + "sha256:b4d7679a08fea64573c969f6994a2631908bb2c0e69a7235648642f3d2e39a68", + "sha256:c250a7ec489b652c892e4f0a5d122cc14c3780f9f643e1a326754aedf82d9a76", + "sha256:ca86db5b561b894f9e5f115d6a159fff2a2570a652e07889d8a383b5fae66eb4", + "sha256:cfc523edecddaef56f6740d7de1ce24a2fdf94fd5e704091856a201872e37f9f", + "sha256:da113b70f6ec40e7d81b43d1b139b9db6a05727ab8be1ee559f3a69854a69d34", + "sha256:f6fac64a38f6768e7bc7b035b9e10d8a538a9fadce06b983fb3e6fa55ac5f5ce", + "sha256:f8559617b1fcf59a9aedba2c9838b5b6aa211ffedecabca412b92a1ff75aac1a", + "sha256:fbb42a541b1093385a2d8c7eec94d26d30437d0e77c1d25dae1dcc46741a385e" ], "index": "pypi", - "version": "==2.8.6" + "version": "==2.9.1" + }, + "pyasn1": { + "hashes": [ + "sha256:014c0e9976956a08139dc0712ae195324a75e142284d5f87f1a87ee1b068a359", + "sha256:03840c999ba71680a131cfaee6fab142e1ed9bbd9c693e285cc6aca0d555e576", + "sha256:0458773cfe65b153891ac249bcf1b5f8f320b7c2ce462151f8fa74de8934becf", + "sha256:08c3c53b75eaa48d71cf8c710312316392ed40899cb34710d092e96745a358b7", + "sha256:39c7e2ec30515947ff4e87fb6f456dfc6e84857d34be479c9d4a4ba4bf46aa5d", + "sha256:5c9414dcfede6e441f7e8f81b43b34e834731003427e5b09e4e00e3172a10f00", + "sha256:6e7545f1a61025a4e58bb336952c5061697da694db1cae97b116e9c46abcf7c8", + "sha256:78fa6da68ed2727915c4767bb386ab32cdba863caa7dbe473eaae45f9959da86", + "sha256:7ab8a544af125fb704feadb008c99a88805126fb525280b2270bb25cc1d78a12", + "sha256:99fcc3c8d804d1bc6d9a099921e39d827026409a58f2a720dcdb89374ea0c776", + "sha256:aef77c9fb94a3ac588e87841208bdec464471d9871bd5050a287cc9a475cd0ba", + "sha256:e89bf84b5437b532b0803ba5c9a5e054d21fec423a89952a74f87fa2c9b7bce2", + "sha256:fec3e9d8e36808a28efb59b489e4528c10ad0f480e57dcc32b4de5c9d8c9fdf3" + ], + "version": "==0.4.8" + }, + "pyasn1-modules": { + "hashes": [ + "sha256:0845a5582f6a02bb3e1bde9ecfc4bfcae6ec3210dd270522fee602365430c3f8", + "sha256:0fe1b68d1e486a1ed5473f1302bd991c1611d319bba158e98b106ff86e1d7199", + "sha256:15b7c67fabc7fc240d87fb9aabf999cf82311a6d6fb2c70d00d3d0604878c811", + "sha256:426edb7a5e8879f1ec54a1864f16b882c2837bfd06eee62f2c982315ee2473ed", + "sha256:65cebbaffc913f4fe9e4808735c95ea22d7a7775646ab690518c056784bc21b4", + "sha256:905f84c712230b2c592c19470d3ca8d552de726050d1d1716282a1f6146be65e", + "sha256:a50b808ffeb97cb3601dd25981f6b016cbb3d31fbf57a8b8a87428e6158d0c74", + "sha256:a99324196732f53093a84c4369c996713eb8c89d360a496b599fb1a9c47fc3eb", + "sha256:b80486a6c77252ea3a3e9b1e360bc9cf28eaac41263d173c032581ad2f20fe45", + "sha256:c29a5e5cc7a3f05926aff34e097e84f8589cd790ce0ed41b67aed6857b26aafd", + "sha256:cbac4bc38d117f2a49aeedec4407d23e8866ea4ac27ff2cf7fb3e5b570df19e0", + "sha256:f39edd8c4ecaa4556e989147ebf219227e2cd2e8a43c7e7fcb1f1c18c5fd6a3d", + "sha256:fe0644d9ab041506b62782e92b06b8c68cca799e1a9636ec398675459e031405" + ], + "version": "==0.2.8" }, "pycparser": { "hashes": [ @@ -530,54 +839,98 @@ "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", "version": "==2.20" }, + "pyopenssl": { + "hashes": [ + "sha256:4c231c759543ba02560fcd2480c48dcec4dae34c9da7d3747c508227e0624b51", + "sha256:818ae18e06922c066f777a33f1fca45786d85edfe71cd043de6379337a7f274b" + ], + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==20.0.1" + }, "python-dateutil": { "hashes": [ - "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c", - "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a" + "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86", + "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9" ], "index": "pypi", - "version": "==2.8.1" + "version": "==2.8.2" }, "python-dotenv": { "hashes": [ - "sha256:0c8d1b80d1a1e91717ea7d526178e3882732420b03f08afea0406db6402e220e", - "sha256:587825ed60b1711daea4832cf37524dfd404325b7db5e25ebe88c495c9f807a0" + "sha256:aae25dc1ebe97c420f50b81fb0e5c949659af713f31fdb63c749ca68748f34b1", + "sha256:b31f6f743c32826287e2faf09ef9b184d2caa628f45952611bf1c56ab88e1e4a", + "sha256:f521bc2ac9a8e03c736f62911605c5d83970021e3fa95b37d769e2bbbe9b6172" ], "index": "pypi", - "version": "==0.15.0" + "version": "==0.19.0" }, "python-gnupg": { "hashes": [ - "sha256:3aa0884b3bd414652c2385b9df39e7b87272c2eca1b8fcc3089bc9e58652019a", - "sha256:cba3566e8a8fb7bb417d6897a6e17bfc7f9371052e57eb0057783c07d762a679" + "sha256:2061f56b1942c29b92727bf9aecbd3cea3893acc9cccbdc7eb4604285efe4ac7", + "sha256:3ff5b1bf5e397de6e1fe41a7c0f403dad4e242ac92b345f440eaecfb72a7ebae" ], "index": "pypi", - "version": "==0.4.6" + "version": "==0.4.7" }, "python-levenshtein": { "hashes": [ - "sha256:033a11de5e3d19ea25c9302d11224e1a1898fe5abd23c61c7c360c25195e3eb1", - "sha256:15e26882728c29ccdf74cfc6ac4b49fc22c08b44d152348cb0eb1ec4f3dbf9df", - "sha256:3df5e5eb144570ecf5ad38864a2393068798328c7f05e7b167a49391d36a2db1", - "sha256:7f049b3ddc4b525bd469febafb98bf5202f789b722e0e4ccbec2ffbe8c07d7b4" + "sha256:212db61934fcb819f5cb2fcb5ad5c0e2e43f3161f524eef4a135f7285dfb308a", + "sha256:a31d3065bf4e8fc6721038c783df132650ad24711e986b25ae0c37e140a99da3", + "sha256:d92fe5c3b10c8ad8f2d880499f5e96ed24dbc7cd0414a665c2e2505feaf4ec58", + "sha256:dc2395fbd148a1ab31090dd113c366695934b9e85fe5a4b2a032745efd0346f6" ], - "index": "pypi", - "version": "==0.12.0" + "version": "==0.12.2" }, "python-magic": { "hashes": [ - "sha256:356efa93c8899047d1eb7d3eb91e871ba2f5b1376edbaf4cc305e3c872207355", - "sha256:b757db2a5289ea3f1ced9e60f072965243ea43a2221430048fd8cacab17be0ce" + "sha256:4fec8ee805fea30c07afccd1592c0f17977089895bdfaae5fec870a84e997626", + "sha256:de800df9fb50f8ec5974761054a708af6e4246b03b4bdaee993f948947b0ebcf" ], "index": "pypi", - "version": "==0.4.18" + "version": "==0.4.24" }, "pytz": { "hashes": [ - "sha256:3e6b7dd2d1e0a59084bcee14a17af60c5c562cdc16d828e8eba2e683d3a7e268", - "sha256:5c55e189b682d420be27c6995ba6edce0c0a77dd67bfbe2ae6607134d5851ffd" + "sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da", + "sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798" ], - "version": "==2020.4" + "version": "==2021.1" + }, + "pyyaml": { + "hashes": [ + "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf", + "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696", + "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393", + "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77", + "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922", + "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5", + "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8", + "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10", + "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc", + "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018", + "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e", + "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253", + "sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347", + "sha256:7318f36ecd539646767465d194cbbe9c248f243ba0117c3d043c660d69cfeceb", + "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183", + "sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541", + "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb", + "sha256:8fc87fd1a1d30991d9198a4317e90dfaf3d2963c013608a019e8311568fa08b2", + "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185", + "sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc", + "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db", + "sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa", + "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46", + "sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122", + "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b", + "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63", + "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df", + "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc", + "sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247", + "sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6", + "sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0" + ], + "version": "==5.4.1" }, "redis": { "hashes": [ @@ -590,168 +943,165 @@ }, "regex": { "hashes": [ - "sha256:02951b7dacb123d8ea6da44fe45ddd084aa6777d4b2454fa0da61d569c6fa538", - "sha256:0d08e71e70c0237883d0bef12cad5145b84c3705e9c6a588b2a9c7080e5af2a4", - "sha256:1862a9d9194fae76a7aaf0150d5f2a8ec1da89e8b55890b1786b8f88a0f619dc", - "sha256:1ab79fcb02b930de09c76d024d279686ec5d532eb814fd0ed1e0051eb8bd2daa", - "sha256:1fa7ee9c2a0e30405e21031d07d7ba8617bc590d391adfc2b7f1e8b99f46f444", - "sha256:262c6825b309e6485ec2493ffc7e62a13cf13fb2a8b6d212f72bd53ad34118f1", - "sha256:2a11a3e90bd9901d70a5b31d7dd85114755a581a5da3fc996abfefa48aee78af", - "sha256:2c99e97d388cd0a8d30f7c514d67887d8021541b875baf09791a3baad48bb4f8", - "sha256:3128e30d83f2e70b0bed9b2a34e92707d0877e460b402faca908c6667092ada9", - "sha256:38c8fd190db64f513fe4e1baa59fed086ae71fa45083b6936b52d34df8f86a88", - "sha256:3bddc701bdd1efa0d5264d2649588cbfda549b2899dc8d50417e47a82e1387ba", - "sha256:4902e6aa086cbb224241adbc2f06235927d5cdacffb2425c73e6570e8d862364", - "sha256:49cae022fa13f09be91b2c880e58e14b6da5d10639ed45ca69b85faf039f7a4e", - "sha256:56e01daca75eae420bce184edd8bb341c8eebb19dd3bce7266332258f9fb9dd7", - "sha256:5862975b45d451b6db51c2e654990c1820523a5b07100fc6903e9c86575202a0", - "sha256:6a8ce43923c518c24a2579fda49f093f1397dad5d18346211e46f134fc624e31", - "sha256:6c54ce4b5d61a7129bad5c5dc279e222afd00e721bf92f9ef09e4fae28755683", - "sha256:6e4b08c6f8daca7d8f07c8d24e4331ae7953333dbd09c648ed6ebd24db5a10ee", - "sha256:717881211f46de3ab130b58ec0908267961fadc06e44f974466d1887f865bd5b", - "sha256:749078d1eb89484db5f34b4012092ad14b327944ee7f1c4f74d6279a6e4d1884", - "sha256:7913bd25f4ab274ba37bc97ad0e21c31004224ccb02765ad984eef43e04acc6c", - "sha256:7a25fcbeae08f96a754b45bdc050e1fb94b95cab046bf56b016c25e9ab127b3e", - "sha256:80ef188c0e47a6c964eed71c55a73c245f8daf9f0a4a9d804e91275afb468ca4", - "sha256:83d6b356e116ca119db8e7c6fc2983289d87b27b3fac238cfe5dca529d884562", - "sha256:842fb985b2b99a82a2b145b6bbd588c5f5cfd83693402920fcb985d515794666", - "sha256:8b882a78c320478b12ff024e81dc7d43c1462aa4a3341c754ee65d857a521f85", - "sha256:8f6a2229e8ad946e36815f2a03386bb8353d4bde368fdf8ca5f0cb97264d3b5c", - "sha256:9801c4c1d9ae6a70aeb2128e5b4b68c45d4f0af0d1535500884d644fa9b768c6", - "sha256:a15f64ae3a027b64496a71ab1f722355e570c3fac5ba2801cafce846bf5af01d", - "sha256:a3d748383762e56337c39ab35c6ed4deb88df5326f97a38946ddd19028ecce6b", - "sha256:a63f1a07932c9686d2d416fb295ec2c01ab246e89b4d58e5fa468089cab44b70", - "sha256:b2b1a5ddae3677d89b686e5c625fc5547c6e492bd755b520de5332773a8af06b", - "sha256:b2f4007bff007c96a173e24dcda236e5e83bde4358a557f9ccf5e014439eae4b", - "sha256:baf378ba6151f6e272824b86a774326f692bc2ef4cc5ce8d5bc76e38c813a55f", - "sha256:bafb01b4688833e099d79e7efd23f99172f501a15c44f21ea2118681473fdba0", - "sha256:bba349276b126947b014e50ab3316c027cac1495992f10e5682dc677b3dfa0c5", - "sha256:c084582d4215593f2f1d28b65d2a2f3aceff8342aa85afd7be23a9cad74a0de5", - "sha256:d1ebb090a426db66dd80df8ca85adc4abfcbad8a7c2e9a5ec7513ede522e0a8f", - "sha256:d2d8ce12b7c12c87e41123997ebaf1a5767a5be3ec545f64675388970f415e2e", - "sha256:e32f5f3d1b1c663af7f9c4c1e72e6ffe9a78c03a31e149259f531e0fed826512", - "sha256:e3faaf10a0d1e8e23a9b51d1900b72e1635c2d5b0e1bea1c18022486a8e2e52d", - "sha256:f7d29a6fc4760300f86ae329e3b6ca28ea9c20823df123a2ea8693e967b29917", - "sha256:f8f295db00ef5f8bae530fc39af0b40486ca6068733fb860b42115052206466f" + "sha256:026beb631097a4a3def7299aa5825e05e057de3c6d72b139c37813bfa351274b", + "sha256:0dcb13eb9340a0c676a6260d67c0aa6e3eeef0305c6e766c9161fb9de44d85bf", + "sha256:14caacd1853e40103f59571f169704367e79fb78fac3d6d09ac84d9197cadd16", + "sha256:16d9eaa8c7e91537516c20da37db975f09ac2e7772a0694b245076c6d68f85da", + "sha256:18fdc51458abc0a974822333bd3a932d4e06ba2a3243e9a1da305668bd62ec6d", + "sha256:28e8af338240b6f39713a34e337c3813047896ace09d51593d6907c66c0708ba", + "sha256:316332a0af33f3c40dd5bbdd7b47abbebcf29ef81d83cb7f59af4008ca3611fd", + "sha256:3835de96524a7b6869a6c710b26c90e94558c31006e96ca3cf6af6751b27dca1", + "sha256:3905c86cc4ab6d71635d6419a6f8d972cab7c634539bba6053c47354fd04452c", + "sha256:3c09d88a07483231119f5017904db8f60ad67906efac3f1baa31b9b7f7cca281", + "sha256:4551728b767f35f86b8e5ec19a363df87450c7376d7419c3cac5b9ceb4bce576", + "sha256:459bbe342c5b2dec5c5223e7c363f291558bc27982ef39ffd6569e8c082bdc83", + "sha256:45af20983c01dab82b56b05e3950ba68722720744da4bffde70f14a57723d213", + "sha256:4f421e3cdd3a273bace013751c345f4ebeef08f05e8c10757533ada360b51a39", + "sha256:577737ec3d4c195c4aef01b757905779a9e9aee608fa1cf0aec16b5576c893d3", + "sha256:57fece29f7cc55d882fe282d9de52f2f522bb85290555b49394102f3621751ee", + "sha256:7976d410e42be9ae7458c1816a416218364e06e162b82e42f7060737e711d9ce", + "sha256:85f568892422a0e96235eb8ea6c5a41c8ccbf55576a2260c0160800dbd7c4f20", + "sha256:8764a78c5464ac6bde91a8c87dd718c27c1cabb7ed2b4beaf36d3e8e390567f9", + "sha256:8935937dad2c9b369c3d932b0edbc52a62647c2afb2fafc0c280f14a8bf56a6a", + "sha256:8fe58d9f6e3d1abf690174fd75800fda9bdc23d2a287e77758dc0e8567e38ce6", + "sha256:937b20955806381e08e54bd9d71f83276d1f883264808521b70b33d98e4dec5d", + "sha256:9569da9e78f0947b249370cb8fadf1015a193c359e7e442ac9ecc585d937f08d", + "sha256:a3b73390511edd2db2d34ff09aa0b2c08be974c71b4c0505b4a048d5dc128c2b", + "sha256:a4eddbe2a715b2dd3849afbdeacf1cc283160b24e09baf64fa5675f51940419d", + "sha256:a5c6dbe09aff091adfa8c7cfc1a0e83fdb8021ddb2c183512775a14f1435fe16", + "sha256:b63e3571b24a7959017573b6455e05b675050bbbea69408f35f3cb984ec54363", + "sha256:bb350eb1060591d8e89d6bac4713d41006cd4d479f5e11db334a48ff8999512f", + "sha256:bf6d987edd4a44dd2fa2723fca2790f9442ae4de2c8438e53fcb1befdf5d823a", + "sha256:bfa6a679410b394600eafd16336b2ce8de43e9b13f7fb9247d84ef5ad2b45e91", + "sha256:c856ec9b42e5af4fe2d8e75970fcc3a2c15925cbcc6e7a9bcb44583b10b95e80", + "sha256:cea56288eeda8b7511d507bbe7790d89ae7049daa5f51ae31a35ae3c05408531", + "sha256:ea212df6e5d3f60341aef46401d32fcfded85593af1d82b8b4a7a68cd67fdd6b", + "sha256:f35567470ee6dbfb946f069ed5f5615b40edcbb5f1e6e1d3d2b114468d505fc6", + "sha256:fbc20975eee093efa2071de80df7f972b7b35e560b213aafabcec7c0bd00bd8c", + "sha256:ff4a8ad9638b7ca52313d8732f37ecd5fd3c8e3aff10a8ccb93176fd5b3812f6" ], - "version": "==2020.11.13" + "version": "==2021.8.3" }, "reportlab": { "hashes": [ - "sha256:0008b5baa39d7e3a8132c4b47ecae88d6858ad386518e754e5e7b8025ee4722b", - "sha256:0ad5a540c336941272fe161ef3a9830da3d4b3a65a195531cebd3cad5db58b2a", - "sha256:0c965a5691686d746f558ee1c52aa9c63a01a0e13cba61ffc661573948e32f61", - "sha256:0fd568fa5615ae99f76289c52ff230207852ee942d4934f6c893c93d2a79544e", - "sha256:1117d905a3404c696869c7aabec9454b43ed6acbbc73f9256c6fcea23e7ae93e", - "sha256:1ea7c388e91ad9d823655ad6a13751ff67e8a0e7cf4065cf051b4c931cdd9450", - "sha256:26c0ee8f62652cc7fcdc47a1cb3b34775a4d625738025c1a7edb8718bda5a315", - "sha256:368c5b3fc3d5a541cb9dcacefa563fdb445365f517e3cbf64b4326631d1cf13c", - "sha256:451d42fdcdd7d84587d6d9c8f5d9a7d0e997305efb606705063ca1fe8bcca551", - "sha256:47394acba4da8e56ef8e55d8eb483b868521696ba49ab0f0fcf8a1a4a5ac6e49", - "sha256:51b16e297f7b937fc530dd151e4b38f1d305b01c9aa10657bc32a5d2901b8ad7", - "sha256:51c0cdcf606ded0a7b4b50050400f25125ea797fbfc3c817135993b38f8b764e", - "sha256:55c672c579618843e0fd00140fb71f1ffebc4f1c542ac385c4f4999f2f5398d9", - "sha256:5c34a96ecfbf595caf16178a06abcd26a5f8720e01fe1285d4c97333382cfaeb", - "sha256:61aa89a00754b18c4f2956b8bff831f1fd3affef6476dc63462d92211941605e", - "sha256:62234d29c97279917903e4587faf240a5dea4617be250db55386ff268eb5a7c5", - "sha256:670f2a8dcc23bf798c39b95c64bf76ee387549b962f76783670821978a226663", - "sha256:69387f171f6c7b55109caa6d061b17a18f2f9e724a0212c07cd692aeb369dd19", - "sha256:6c5c8871b659f7c2975382d7b61f3c182701fa9eb62cf649c3c73ba8fc5e2595", - "sha256:80139ceb3a568f5be908094f1701fd05391b71425e8b69aaed0d30db647ca2aa", - "sha256:80661a76d0019b5e2c315ccd3bc7093d754067d6142b36a3a0ec4f416073d23b", - "sha256:85a2236f324ae336da7f4b183fa99bed261bcc00ac1255ee91a504e68b086d00", - "sha256:89a3acd98bd4478d6bbc5cb32e0665ea546c98bff8b58d5e1014659daa6ef75a", - "sha256:8a39119fcab146bde41fd1c6d148f9ee1e2cca10c6f9c2b7eb4dd710a3a2c6ac", - "sha256:9c31c2526401da6cc92018f68483f2aac0a731cb98435445ea4b72d46b438c84", - "sha256:9e8ae1c3b8a1697147c5c97f00d66ab1c54d88c4615b0cdd9b1a667d7baf3eb7", - "sha256:a479c38ab2b997ce05d3bef906783ac20cf4cb224a154e80c9018c5e4d943a35", - "sha256:a79aab8d069543d5085d58260f18705a08acd92a4501a41261913fddc2137d46", - "sha256:b0a8314383de853599ca531dfe55eaa49bb8d6b0bb663b2f8479b7a0f3385ea2", - "sha256:b3d9926e64bd8008007b2d9819d7b30179b069ce95431d5060f71afc36885389", - "sha256:c2a9a77ce4f25ffb52d705be82a9f41b47f6b0da23870ebc3587709e7242da30", - "sha256:c578dd0799f70fb577474cd383f035c6e1057e4fe837278113f9cfa6eee4b076", - "sha256:c5abd9d0023ad20030524ab0d5fa39d77aed025519b1fa426304ab2dd0328b89", - "sha256:ced96125525ba21311e9512adf391170b9e149f89e27e45b06ff07b70f97a0b2", - "sha256:d692fb88d6ef5e75242b00009b54953a0425eaa8bd3a36db9db8b396785e1f57", - "sha256:d70c2104286459658e61388af9eee838b612986bd8a36e1d21ba36152983ac15", - "sha256:de47c65c10ac6f0d2addb28f1b1657b1c707aca014d09d01b3b728cf19e8f791", - "sha256:e6e7592527791841db0820a72c6afae52655a05b0b6d4df184fd2bafe82ee1ee", - "sha256:e8a7e95ee6ea5566291b59ede5b9fadce809dca43ebfbfe11e3ff3d6492c6f0e", - "sha256:f041759138b3a95508c4281b3db3bf9bb28636d84c554272a58a5ca7c9f9bbf4", - "sha256:f39c7fc1fa2e4a1d9747a3effd70731a9d0e9eb5738247fa089c059eff19d43e", - "sha256:f65ac89ee0ba569f5279360eae08783f7f2e95c9810a9846c957fbd5950f4896" + "sha256:00e9ffb955972a8f6a3a0d61a12231fcaf5e23ee238c98421d65fecc29bd88a1", + "sha256:115177b3fc51209b5f50371735311c9a6cd9d260ffedbdce5fbc965645b7567c", + "sha256:17130f034dae50aaf22fce2292e0077a0c2093ba4363211bcafb54418fb8dc09", + "sha256:200bdfc327d5b06cb400ae86c972b579efe03a1fd8a2e8cb7a5d9aaa744e5adb", + "sha256:496b28ef414d9a7734e07221c4386bb00f416a3aa276b9f349ed9a328c73ec23", + "sha256:4bc378039f70141176f3d511d84bc1a172820d4d2edee4f9fcff52cde753dc08", + "sha256:4f357b4c39b0fa0071de47e8be7af44e07f375d2e59e395daccb7fd13b275668", + "sha256:57b39303e6dbe3de91e60a14269543ac058ac98a0ea6cf900f5403d9c226022f", + "sha256:6472478e597ef4a8f5c621d811d08b7ef09fc5af5bc85c2cf4a4505a7164f8b8", + "sha256:68f9324000cfc5570b5a59a92306691b5d655078a399f20bc72c2581fe903261", + "sha256:69870e2bbf39b60ebe9a31b31324e249bf314bdc2798e46efc58c67db74b56cb", + "sha256:6adb17ba89829d5e77fd81baac396f1af99241d7dfc121a065217334131662e7", + "sha256:7c360aee2bdaa05c24cadddc2f10924961dc7cad125d8876b4d307c879b3b4e8", + "sha256:7c4c8e87ef29714ccc7fa9764efe30d849cd38f8a9a1742ab7aedf8b5e23494d", + "sha256:8a07672e86bf288ea3e55959d2e06d6c01320318662241f9b7a71c583e15e5b5", + "sha256:9f583295f7dd523bf6e5619720677279dc7b9db22671573888f0591fc46b90b2", + "sha256:ae27853be6cf0c346469f313abc2938aed5fe0474d6ec0c8bc5ce45e89cb9bad", + "sha256:b668433f32ac955a94633e58ed7800c06c00f9c46d3b99e2189b3d88dc3184c8", + "sha256:b7a92564198c5a5ff4efdb83ace215c73343afb80d9379183bc736fea76edd6d", + "sha256:bd52e1715c70a96a116a61c8477e586b3a46047c85581195bc74162b19b46286", + "sha256:c7ddc9a6234267bbb52059b017ca22f59ffd7d41d545524cb85f68086a2cbb43", + "sha256:c8586d72932b8e3bd50a5230d6f1cfbb85c2605bad34253c6d6fe757211b2bf7", + "sha256:ce3d8e782e3776f19d3accc706aab85ff06caedb70a52016532bebacf5537567", + "sha256:e3ac401b266bb2430b674ac9317065b03ef2325315c611792761ad636326958c", + "sha256:f3fd26f63c4a9033115707a8718154538a1cebfd6ec992f214e6423524450e3e" ], - "version": "==3.5.56" + "markers": "python_version >= '2.7' and python_version < '4'", + "version": "==3.6.1" + }, + "requests": { + "hashes": [ + "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24", + "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7" + ], + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'", + "version": "==2.26.0" }, "scikit-learn": { "hashes": [ - "sha256:090bbf144fd5823c1f2efa3e1a9bf180295b24294ca8f478e75b40ed54f8036e", - "sha256:0a127cc70990d4c15b1019680bfedc7fec6c23d14d3719fdf9b64b22d37cdeca", - "sha256:0d39748e7c9669ba648acf40fb3ce96b8a07b240db6888563a7cb76e05e0d9cc", - "sha256:1b8a391de95f6285a2f9adffb7db0892718950954b7149a70c783dc848f104ea", - "sha256:20766f515e6cd6f954554387dfae705d93c7b544ec0e6c6a5d8e006f6f7ef480", - "sha256:2aa95c2f17d2f80534156215c87bee72b6aa314a7f8b8fe92a2d71f47280570d", - "sha256:5ce7a8021c9defc2b75620571b350acc4a7d9763c25b7593621ef50f3bd019a2", - "sha256:6c28a1d00aae7c3c9568f61aafeaad813f0f01c729bee4fd9479e2132b215c1d", - "sha256:7671bbeddd7f4f9a6968f3b5442dac5f22bf1ba06709ef888cc9132ad354a9ab", - "sha256:914ac2b45a058d3f1338d7736200f7f3b094857758895f8667be8a81ff443b5b", - "sha256:98508723f44c61896a4e15894b2016762a55555fbf09365a0bb1870ecbd442de", - "sha256:a64817b050efd50f9abcfd311870073e500ae11b299683a519fbb52d85e08d25", - "sha256:cb3e76380312e1f86abd20340ab1d5b3cc46a26f6593d3c33c9ea3e4c7134028", - "sha256:d0dcaa54263307075cb93d0bee3ceb02821093b1b3d25f66021987d305d01dce", - "sha256:d9a1ce5f099f29c7c33181cc4386660e0ba891b21a60dc036bf369e3a3ee3aec", - "sha256:da8e7c302003dd765d92a5616678e591f347460ac7b53e53d667be7dfe6d1b10", - "sha256:daf276c465c38ef736a79bd79fc80a249f746bcbcae50c40945428f7ece074f8" + "sha256:076369634ee72b5a5941440661e2f306ff4ac30903802dc52031c7e9199ac640", + "sha256:18f7131e62265bf2691ed1d0303c640313894ccfe4278427478c6b2f45094b53", + "sha256:26f66b3726b54dfb76ea51c5d9c2431ed17ebc066cb4527662b9e851a3e7ba61", + "sha256:2951f87d35e72f007701c6e028aa230f6df6212a3194677c0c950486066a454d", + "sha256:2a5348585aa793bc8cc5a72f8e9067c9380834b0aadbd55f924843b071f13282", + "sha256:3eeff086f7329521d27249a082ea3c48c085cedb110db5f65968ab55c3ba2e09", + "sha256:4395e91b3548005f4a645018435b5a94f8cce232b5b70753020e606c6a750656", + "sha256:44e452ea8491225c5783d49577aad0f36202dfd52aec7f82c0fdfe5fbd5f7400", + "sha256:490436b44b3a1957cb625e871764b0aa330b34cc416aea4abc6c38ca63d0d682", + "sha256:5e6e3c042cea83f2e20a45e563b8eabc1f8f72446251fe23ebefdf111a173a33", + "sha256:66f27bf21202a850bcd7b6303916e4907f6e22ec59a14974ede4955aed5c7ed0", + "sha256:743b6edd98c98991be46c08e6b21df3861d5ae915f91d59f988384d93f7263e7", + "sha256:758619e49cd7c17282e6cc60d5cc73c02c072b47c9a10010bb3bb47e0d976e50", + "sha256:7f654befc5ad413690cc58f3f34a3e906caf825195ce0fda00a8e9565e1403e6", + "sha256:800aaf63f8838c00e85db2267dd226f89858594843fd03932a9eda95746d2c40", + "sha256:80ca024154b84b6ac4cfc86930ba13fdc348a209753bf2c16129db6f9eb8a80b", + "sha256:890d7d588f65acb0c4f6c083347c9076916bda5e6bd8400f06244b1afc1009af", + "sha256:905d8934d1e27a686698864a5863ff2c0e13a2ae1adb78a8a848aacc8a49927d", + "sha256:a83fcd9d59c42a2f66b307e3b0b0f08aa8e6e45be33da055697ea499f0e4f7c2", + "sha256:afeb06dc69847927634e58579b9cdc72e1390b79497336b2324b1b173f33bd47", + "sha256:b0d13fd56d26cf3de0314a4fd48037108c638fe126d813f5c1222bb0f08b6a76", + "sha256:c08b27cb78ee8d2dc781a7affed09859441f5b624f9f92da59ac0791c8774dfc", + "sha256:c912247e42114f389858ae05d63f4359d4e667ea72aaabee191aee9ad3f9774a", + "sha256:c9ee65e3e0179951f8c196db0de7f243564c631ec671d1e5ca87f6846029bbab", + "sha256:d7fe05fcb44eadd6d6c874c768f085f5de1239db3a3b7be4d3d23d12e4120589", + "sha256:d819d625832fb2969911a243e009cfa135cb8ef1e150866e417d6e9d75290087", + "sha256:e534f5f3796db6781c87e9835dcd51b7854c8c5a379c9210b93605965c1941fd" ], "index": "pypi", - "version": "==0.23.2" + "version": "==0.24.0" }, "scipy": { "hashes": [ - "sha256:168c45c0c32e23f613db7c9e4e780bc61982d71dcd406ead746c7c7c2f2004ce", - "sha256:213bc59191da2f479984ad4ec39406bf949a99aba70e9237b916ce7547b6ef42", - "sha256:25b241034215247481f53355e05f9e25462682b13bd9191359075682adcd9554", - "sha256:2c872de0c69ed20fb1a9b9cf6f77298b04a26f0b8720a5457be08be254366c6e", - "sha256:3397c129b479846d7eaa18f999369a24322d008fac0782e7828fa567358c36ce", - "sha256:368c0f69f93186309e1b4beb8e26d51dd6f5010b79264c0f1e9ca00cd92ea8c9", - "sha256:3d5db5d815370c28d938cf9b0809dade4acf7aba57eaf7ef733bfedc9b2474c4", - "sha256:4598cf03136067000855d6b44d7a1f4f46994164bcd450fb2c3d481afc25dd06", - "sha256:4a453d5e5689de62e5d38edf40af3f17560bfd63c9c5bd228c18c1f99afa155b", - "sha256:4f12d13ffbc16e988fa40809cbbd7a8b45bc05ff6ea0ba8e3e41f6f4db3a9e47", - "sha256:634568a3018bc16a83cda28d4f7aed0d803dd5618facb36e977e53b2df868443", - "sha256:65923bc3809524e46fb7eb4d6346552cbb6a1ffc41be748535aa502a2e3d3389", - "sha256:6b0ceb23560f46dd236a8ad4378fc40bad1783e997604ba845e131d6c680963e", - "sha256:8c8d6ca19c8497344b810b0b0344f8375af5f6bb9c98bd42e33f747417ab3f57", - "sha256:9ad4fcddcbf5dc67619379782e6aeef41218a79e17979aaed01ed099876c0e62", - "sha256:a254b98dbcc744c723a838c03b74a8a34c0558c9ac5c86d5561703362231107d", - "sha256:b03c4338d6d3d299e8ca494194c0ae4f611548da59e3c038813f1a43976cb437", - "sha256:b5e9d3e4474644915809d6aa1416ff20430a3ed9ae723a5d295da5ddb24985e2", - "sha256:cc1f78ebc982cd0602c9a7615d878396bec94908db67d4ecddca864d049112f2", - "sha256:d6d25c41a009e3c6b7e757338948d0076ee1dd1770d1c09ec131f11946883c54", - "sha256:d84cadd7d7998433334c99fa55bcba0d8b4aeff0edb123b2a1dfcface538e474", - "sha256:e360cb2299028d0b0d0f65a5c5e51fc16a335f1603aa2357c25766c8dab56938", - "sha256:e98d49a5717369d8241d6cf33ecb0ca72deee392414118198a8e5b4c35c56340", - "sha256:ed572470af2438b526ea574ff8f05e7f39b44ac37f712105e57fc4d53a6fb660", - "sha256:f87b39f4d69cf7d7529d7b1098cb712033b17ea7714aed831b95628f483fd012", - "sha256:fa789583fc94a7689b45834453fec095245c7e69c58561dc159b5d5277057e4c" + "sha256:2a0eeaab01258e0870c4022a6cd329aef3b7c6c2b606bd7cf7bb2ba9820ae561", + "sha256:3304bd5bc32e00954ac4b3f4cc382ca8824719bf348aacbec6347337d6b125fe", + "sha256:3f52470e0548cdb74fb8ddf06773ffdcca7c97550f903b1c51312ec19243a7f7", + "sha256:4729b41a4cdaf4cd011aeac816b532f990bdf97710cef59149d3e293115cf467", + "sha256:4ee952f39a4a4c7ba775a32b664b1f4b74818548b65f765987adc14bb78f5802", + "sha256:611f9cb459d0707dd8e4de0c96f86e93f61aac7475fcb225e9ec71fecdc5cebf", + "sha256:6b47d5fa7ea651054362561a28b1ccc8da9368a39514c1bbf6c0977a1c376764", + "sha256:71cfc96297617eab911e22216e8a8597703202e95636d9406df9af5c2ac99a2b", + "sha256:787749110a23502031fb1643c55a2236c99c6b989cca703ea2114d65e21728ef", + "sha256:90c07ba5f34f33299a428b0d4fa24c30d2ceba44d63f8385b2b05be460819fcb", + "sha256:a496b42dbcd04ea9924f5e92be63af3d8e0f43a274b769bfaca0a297327d54ee", + "sha256:bc61e3e5ff92d2f32bb263621d54a9cff5e3f7c420af3d1fa122ce2529de2bd9", + "sha256:c9951e3746b68974125e5e3445008a4163dd6d20ae0bbdae22b38cb8951dc11b", + "sha256:d1388fbac9dd591ea630da75c455f4cc637a7ca5ecb31a6b6cef430914749cde", + "sha256:d13f31457f2216e5705304d9f28e2826edf75487410a57aa99263fa4ffd792c2", + "sha256:d648aa85dd5074b1ed83008ae987c3fbb53d68af619fce1dee231f4d8bd40e2f", + "sha256:da9c6b336e540def0b7fd65603da8abeb306c5fc9a5f4238665cbbb5ff95cf58", + "sha256:de777332715005438a1c4469451941ec9193f0ac2b5497e625e6b6d30e22b2ae", + "sha256:e101bceeb9e65a90dadbc5ca31283403a2d4667b9c178db29109750568e8d112", + "sha256:efdd3825d54c58df2cc394366ca4b9166cf940a0ebddeb87b6c10053deb625ea", + "sha256:f749b4f6bb334d6166aa8452ffa40f231519f257e37a1deddbdd1ba5e3e9419e" ], - "markers": "python_version >= '3.6'", - "version": "==1.5.4" + "markers": "python_version < '3.10' and python_version >= '3.7'", + "version": "==1.7.1" + }, + "service-identity": { + "hashes": [ + "sha256:6e6c6086ca271dc11b033d17c3a8bea9f24ebff920c587da090afc9519419d34", + "sha256:f0b0caac3d40627c3c04d7a51b6e06721857a0e10a8775f2d1d7e72901b3a7db" + ], + "version": "==21.1.0" }, "six": { "hashes": [ - "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259", - "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced" + "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926", + "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.15.0" + "version": "==1.16.0" }, "sortedcontainers": { "hashes": [ - "sha256:37257a32add0a3ee490bb170b599e93095eed89a55da91fa9f48753ea12fd73f", - "sha256:59cc937650cf60d677c16775597c89a960658a09cf7c1a668f86e1e4464b10a1" + "sha256:25caa5a06cc30b6b83d11423433f65d1f9d76c4c6a0c90e3379eaa43b9bfdb88", + "sha256:a163dcaede0f1c021485e957a39245190e74249897e2ae4b2aa38595db237ee0" ], - "version": "==2.3.0" + "version": "==2.4.0" }, "sqlparse": { "hashes": [ @@ -763,34 +1113,143 @@ }, "threadpoolctl": { "hashes": [ - "sha256:38b74ca20ff3bb42caca8b00055111d74159ee95c4370882bbff2b93d24da725", - "sha256:ddc57c96a38beb63db45d6c159b5ab07b6bced12c45a1f07b2b92f272aebfa6b" + "sha256:86d4b6801456d780e94681d155779058759eaef3c3564758b17b6c99db5f81cb", + "sha256:e5a995e3ffae202758fa8a90082e35783b9370699627ae2733cd1c3a73553616" ], - "markers": "python_version >= '3.5'", - "version": "==2.1.0" + "markers": "python_version >= '3.6'", + "version": "==2.2.0" + }, + "tika": { + "hashes": [ + "sha256:c2c50f405622f74531841104f9e85c17511aede11de8e5385eab1a29a31f191b", + "sha256:d1f2eddb93caa9a2857569486aa2bc0320d0bf1796cdbe03066954cbc4b4bf62" + ], + "index": "pypi", + "version": "==1.24" }, "tqdm": { "hashes": [ - "sha256:38b658a3e4ecf9b4f6f8ff75ca16221ae3378b2e175d846b6b33ea3a20852cf5", - "sha256:d4f413aecb61c9779888c64ddf0c62910ad56dcbe857d8922bb505d4dbff0df1" + "sha256:07856e19a1fe4d2d9621b539d3f072fa88c9c1ef1f3b7dd4d4953383134c3164", + "sha256:35540feeaca9ac40c304e916729e6b78045cbbeccd3e941b2868f09306798ac9", + "sha256:f959986660e27ecdb4ddef8fe7fa11e6e060e1eecff480d8095f38a68c9dde5d" ], "index": "pypi", - "version": "==4.54.1" + "version": "==4.62.1" + }, + "twisted": { + "extras": [ + "tls" + ], + "hashes": [ + "sha256:13c1d1d2421ae556d91e81e66cf0d4f4e4e1e4a36a0486933bee4305c6a4fb9b", + "sha256:2cd652542463277378b0d349f47c62f20d9306e57d1247baabd6d1d38a109006", + "sha256:4ceba6ae6672f78209741951add007ffeae1524a85999853400918c30ecbda2d" + ], + "markers": "python_full_version >= '3.6.7'", + "version": "==21.7.0" + }, + "txaio": { + "hashes": [ + "sha256:7d6f89745680233f1c4db9ddb748df5e88d2a7a37962be174c0fd04c8dba1dc8", + "sha256:c16b55f9a67b2419cfdf8846576e2ec9ba94fe6978a83080c352a80db31c93fb" + ], + "markers": "python_version >= '3.6'", + "version": "==21.2.1" + }, + "typing-extensions": { + "hashes": [ + "sha256:0ac0f89795dd19de6b97debb0c6af1c70987fd80a2d62d1958f7e56fcc31b497", + "sha256:50b6f157849174217d0656f99dc82fe932884fb250826c18350e159ec6cdf342", + "sha256:779383f6086d90c99ae41cf0ff39aac8a7937a9283ce0a414e5dd782f4c94a84" + ], + "version": "==3.10.0.0" }, "tzlocal": { "hashes": [ - "sha256:643c97c5294aedc737780a49d9df30889321cbe1204eac2c2ec6134035a92e44", - "sha256:e2cb6c6b5b604af38597403e9852872d7f534962ae2954c7f35efcb1ccacf4a4" + "sha256:705fc271c3fdf1b759c9d6eb52931050b6504022efb44061631bb0b0edac2a8c", + "sha256:c736f2540713deb5938d789ca7c3fc25391e9a20803f05b60ec64987cf086559", + "sha256:f4e6e36db50499e0d92f79b67361041f048e2609d166e93456b50746dc4aef12" ], - "version": "==2.1" + "markers": "python_version >= '3.6'", + "version": "==3.0" + }, + "urllib3": { + "hashes": [ + "sha256:39fb8672126159acb139a7718dd10806104dec1e2f0f6c88aab05d17df10c8d4", + "sha256:f57b4c16c62fa2760b7e3d97c35b255512fb6b59a259730f36ba32ce9f8e342f" + ], + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", + "version": "==1.26.6" + }, + "uvicorn": { + "extras": [ + "standard" + ], + "hashes": [ + "sha256:17f898c64c71a2640514d4089da2689e5db1ce5d4086c2d53699bf99513421c1", + "sha256:8b686a3de71cf25fefa434a291aa48f8b427f6a814e090bd842771732f32192a", + "sha256:d9a3c0dd1ca86728d3e235182683b4cf94cd53a867c288eaeca80ee781b2caff" + ], + "index": "pypi", + "version": "==0.15.0" + }, + "uvloop": { + "hashes": [ + "sha256:04ff57aa137230d8cc968f03481176041ae789308b4d5079118331ab01112450", + "sha256:089b4834fd299d82d83a25e3335372f12117a7d38525217c2258e9b9f4578897", + "sha256:1e5f2e2ff51aefe6c19ee98af12b4ae61f5be456cd24396953244a30880ad861", + "sha256:30ba9dcbd0965f5c812b7c2112a1ddf60cf904c1c160f398e7eed3a6b82dcd9c", + "sha256:3a19828c4f15687675ea912cc28bbcb48e9bb907c801873bd1519b96b04fb805", + "sha256:54dbcb085c076f8749954d1497ecabdb227a94114dac2e7296bc0df26b240c3e", + "sha256:6224f1401025b748ffecb7a6e2652b17768f30b1a6a3f7b44660e5b5b690b12d", + "sha256:647e481940379eebd314c00440314c81ea547aa636056f554d491e40503c8464", + "sha256:6ccd57ae8db17d677e9e06192e9c9ec4bd2066b77790f9aa7dede2cc4008ee8f", + "sha256:772206116b9b57cd625c8a88f2413df2fcfd0b496eb188b82a43bed7af2c2ec9", + "sha256:8e0d26fa5875d43ddbb0d9d79a447d2ace4180d9e3239788208527c4784f7cab", + "sha256:98d117332cc9e5ea8dfdc2b28b0a23f60370d02e1395f88f40d1effd2cb86c4f", + "sha256:b3b768c390952983694e5465382e3769f21a88003f608a4f9ca4a44700d95eaf", + "sha256:b572256409f194521a9895aef274cea88731d14732343da3ecdb175228881638", + "sha256:bd53f7f5db562f37cd64a3af5012df8cac2c464c97e732ed556800129505bd64", + "sha256:bd8f42ea1ea8f4e84d265769089964ddda95eb2bb38b5cbe26712b0616c3edee", + "sha256:e814ac2c6f9daf4c36eb8e85266859f42174a4ff0d71b99405ed559257750382", + "sha256:f74bc20c7b67d1c27c72601c78cf95be99d5c2cdd4514502b4f3eb0933ff1228" + ], + "index": "pypi", + "version": "==0.16.0" }, "watchdog": { "hashes": [ - "sha256:3caefdcc8f06a57fdc5ef2d22aa7c0bfda4f55e71a0bee74cbf3176d97536ef3", - "sha256:e38bffc89b15bafe2a131f0e1c74924cf07dcec020c2e0a26cccd208831fcd43" + "sha256:0bcdf7b99b56a3ae069866c33d247c9994ffde91b620eaf0306b27e099bd1ae0", + "sha256:0bcfe904c7d404eb6905f7106c54873503b442e8e918cc226e1828f498bdc0ca", + "sha256:201cadf0b8c11922f54ec97482f95b2aafca429c4c3a4bb869a14f3c20c32686", + "sha256:3a7d242a7963174684206093846537220ee37ba9986b824a326a8bb4ef329a33", + "sha256:3e305ea2757f81d8ebd8559d1a944ed83e3ab1bdf68bcf16ec851b97c08dc035", + "sha256:431a3ea70b20962e6dee65f0eeecd768cd3085ea613ccb9b53c8969de9f6ebd2", + "sha256:44acad6f642996a2b50bb9ce4fb3730dde08f23e79e20cd3d8e2a2076b730381", + "sha256:54e057727dd18bd01a3060dbf5104eb5a495ca26316487e0f32a394fd5fe725a", + "sha256:6fe9c8533e955c6589cfea6f3f0a1a95fb16867a211125236c82e1815932b5d7", + "sha256:85b851237cf3533fabbc034ffcd84d0fa52014b3121454e5f8b86974b531560c", + "sha256:8805a5f468862daf1e4f4447b0ccf3acaff626eaa57fbb46d7960d1cf09f2e6d", + "sha256:9628f3f85375a17614a2ab5eac7665f7f7be8b6b0a2a228e6f6a2e91dd4bfe26", + "sha256:a12539ecf2478a94e4ba4d13476bb2c7a2e0a2080af2bb37df84d88b1b01358a", + "sha256:acc4e2d5be6f140f02ee8590e51c002829e2c33ee199036fcd61311d558d89f4", + "sha256:b5fc5c127bad6983eecf1ad117ab3418949f18af9c8758bd10158be3647298a9", + "sha256:b8ddb2c9f92e0c686ea77341dcb58216fa5ff7d5f992c7278ee8a392a06e86bb", + "sha256:bf84bd94cbaad8f6b9cbaeef43080920f4cb0e61ad90af7106b3de402f5fe127", + "sha256:d9456f0433845e7153b102fffeb767bde2406b76042f2216838af3b21707894e", + "sha256:e4929ac2aaa2e4f1a30a36751160be391911da463a8799460340901517298b13", + "sha256:e5236a8e8602ab6db4b873664c2d356c365ab3cac96fbdec4970ad616415dd45", + "sha256:fd8c595d5a93abd441ee7c5bb3ff0d7170e79031520d113d6f401d0cf49d7c8f" ], "index": "pypi", - "version": "==0.10.4" + "version": "==2.1.3" + }, + "watchgod": { + "hashes": [ + "sha256:48140d62b0ebe9dd9cf8381337f06351e1f2e70b2203fa9c6eff4e572ca84f29", + "sha256:d6c1ea21df37847ac0537ca0d6c2f4cdf513562e95f77bb93abbcf05573407b7" + ], + "version": "==0.7" }, "wcwidth": { "hashes": [ @@ -799,13 +1258,53 @@ ], "version": "==0.2.5" }, + "websockets": { + "hashes": [ + "sha256:0dd4eb8e0bbf365d6f652711ce21b8fd2b596f873d32aabb0fbb53ec604418cc", + "sha256:1d0971cc7251aeff955aa742ec541ee8aaea4bb2ebf0245748fbec62f744a37e", + "sha256:1d6b4fddb12ab9adf87b843cd4316c4bd602db8d5efd2fb83147f0458fe85135", + "sha256:230a3506df6b5f446fed2398e58dcaafdff12d67fe1397dff196411a9e820d02", + "sha256:276d2339ebf0df4f45df453923ebd2270b87900eda5dfd4a6b0cfa15f82111c3", + "sha256:2cf04601633a4ec176b9cc3d3e73789c037641001dbfaf7c411f89cd3e04fcaf", + "sha256:3ddff38894c7857c476feb3538dd847514379d6dc844961dc99f04b0384b1b1b", + "sha256:48c222feb3ced18f3dc61168ca18952a22fb88e5eb8902d2bf1b50faefdc34a2", + "sha256:51d04df04ed9d08077d10ccbe21e6805791b78eac49d16d30a1f1fe2e44ba0af", + "sha256:597c28f3aa7a09e8c070a86b03107094ee5cdafcc0d55f2f2eac92faac8dc67d", + "sha256:5b42372c9ee5530947c891e993b48755c1afc075ca689f5e13d20e61522b21f6", + "sha256:5c8f0d82ea2468282e08b0cf5307f3ad022290ed50c45d5cb7767957ca782880", + "sha256:7189e51955f9268b2bdd6cc537e0faa06f8fffda7fb386e5922c6391de51b077", + "sha256:7df3596838b2a0c07c6f6d67752c53859a54993d4f062689fdf547cb56d0f84f", + "sha256:826ccf85d4514609219725ba4a7abd569228c2c9f1968e8be05be366f68291ec", + "sha256:836d14eb53b500fd92bd5db2fc5894f7c72b634f9c2a28f546f75967503d8e25", + "sha256:85db8090ba94e22d964498a47fdd933b8875a1add6ebc514c7ac8703eb97bbf0", + "sha256:85e701a6c316b7067f1e8675c638036a796fe5116783a4c932e7eb8e305a3ffe", + "sha256:8c941af1b20971d9e98c2369bdf118ef546b2569957469a9ab1e6d76b97d675f", + "sha256:900589e19200be76dd7cbaa95e9771605b5ce3f62512d039fb3bc5da9014912a", + "sha256:9147868bb0cc01e6846606cd65cbf9c58598f187b96d14dd1ca17338b08793bb", + "sha256:9e7fdc775fe7403dbd8bc883ba59576a6232eac96dacb56512daacf7af5d618d", + "sha256:ab5ee15d3462198c794c49ccd31773d8a2b8c17d622aa184f669d2b98c2f0857", + "sha256:ad893d889bc700a5835e0a95a3e4f2c39e91577ab232a3dc03c262a0f8fc4b5c", + "sha256:b2e71c4670ebe1067fa8632f0d081e47254ee2d3d409de54168b43b0ba9147e0", + "sha256:b43b13e5622c5a53ab12f3272e6f42f1ce37cd5b6684b2676cb365403295cd40", + "sha256:b4ad84b156cf50529b8ac5cc1638c2cf8680490e3fccb6121316c8c02620a2e4", + "sha256:be5fd35e99970518547edc906efab29afd392319f020c3c58b0e1a158e16ed20", + "sha256:caa68c95bc1776d3521f81eeb4d5b9438be92514ec2a79fececda814099c8314", + "sha256:d144b350045c53c8ff09aa1cfa955012dd32f00c7e0862c199edcabb1a8b32da", + "sha256:d2c2d9b24d3c65b5a02cac12cbb4e4194e590314519ed49db2f67ef561c3cf58", + "sha256:e9e5fd6dbdf95d99bc03732ded1fc8ef22ebbc05999ac7e0c7bf57fe6e4e5ae2", + "sha256:ebf459a1c069f9866d8569439c06193c586e72c9330db1390af7c6a0a32c4afd", + "sha256:f31722f1c033c198aa4a39a01905951c00bd1c74f922e8afc1b1c62adbcdd56a", + "sha256:f68c352a68e5fdf1e97288d5cec9296664c590c25932a8476224124aaf90dbcd" + ], + "version": "==9.1" + }, "whitenoise": { "hashes": [ - "sha256:05ce0be39ad85740a78750c86a93485c40f08ad8c62a6006de0233765996e5c7", - "sha256:05d00198c777028d72d8b0bbd234db605ef6d60e9410125124002518a48e515d" + "sha256:d234b871b52271ae7ed6d9da47ffe857c76568f11dd30e28e18c5869dbd11e12", + "sha256:d963ef25639d1417e8a247be36e6aedd8c7c6f0a08adcb5a89146980a96b577c" ], "index": "pypi", - "version": "==5.2.0" + "version": "==5.3.0" }, "whoosh": { "hashes": [ @@ -816,13 +1315,64 @@ "index": "pypi", "version": "==2.7.4" }, - "zipp": { + "zope.interface": { "hashes": [ - "sha256:102c24ef8f171fd729d46599845e95c7ab894a4cf45f5de11a44cc7444fb1108", - "sha256:ed5eee1974372595f9e416cc7bbeeb12335201d8081ca8a0743c954d4446e5cb" + "sha256:08f9636e99a9d5410181ba0729e0408d3d8748026ea938f3b970a0249daa8192", + "sha256:0b465ae0962d49c68aa9733ba92a001b2a0933c317780435f00be7ecb959c702", + "sha256:0cba8477e300d64a11a9789ed40ee8932b59f9ee05f85276dbb4b59acee5dd09", + "sha256:0cee5187b60ed26d56eb2960136288ce91bcf61e2a9405660d271d1f122a69a4", + "sha256:0ea1d73b7c9dcbc5080bb8aaffb776f1c68e807767069b9ccdd06f27a161914a", + "sha256:0f91b5b948686659a8e28b728ff5e74b1be6bf40cb04704453617e5f1e945ef3", + "sha256:15e7d1f7a6ee16572e21e3576d2012b2778cbacf75eb4b7400be37455f5ca8bf", + "sha256:17776ecd3a1fdd2b2cd5373e5ef8b307162f581c693575ec62e7c5399d80794c", + "sha256:194d0bcb1374ac3e1e023961610dc8f2c78a0f5f634d0c737691e215569e640d", + "sha256:1c0e316c9add0db48a5b703833881351444398b04111188069a26a61cfb4df78", + "sha256:205e40ccde0f37496904572035deea747390a8b7dc65146d30b96e2dd1359a83", + "sha256:273f158fabc5ea33cbc936da0ab3d4ba80ede5351babc4f577d768e057651531", + "sha256:2876246527c91e101184f63ccd1d716ec9c46519cc5f3d5375a3351c46467c46", + "sha256:2c98384b254b37ce50eddd55db8d381a5c53b4c10ee66e1e7fe749824f894021", + "sha256:2e5a26f16503be6c826abca904e45f1a44ff275fdb7e9d1b75c10671c26f8b94", + "sha256:334701327f37c47fa628fc8b8d28c7d7730ce7daaf4bda1efb741679c2b087fc", + "sha256:3748fac0d0f6a304e674955ab1365d515993b3a0a865e16a11ec9d86fb307f63", + "sha256:3c02411a3b62668200910090a0dff17c0b25aaa36145082a5a6adf08fa281e54", + "sha256:3dd4952748521205697bc2802e4afac5ed4b02909bb799ba1fe239f77fd4e117", + "sha256:3f24df7124c323fceb53ff6168da70dbfbae1442b4f3da439cd441681f54fe25", + "sha256:469e2407e0fe9880ac690a3666f03eb4c3c444411a5a5fddfdabc5d184a79f05", + "sha256:4de4bc9b6d35c5af65b454d3e9bc98c50eb3960d5a3762c9438df57427134b8e", + "sha256:5208ebd5152e040640518a77827bdfcc73773a15a33d6644015b763b9c9febc1", + "sha256:52de7fc6c21b419078008f697fd4103dbc763288b1406b4562554bd47514c004", + "sha256:5bb3489b4558e49ad2c5118137cfeaf59434f9737fa9c5deefc72d22c23822e2", + "sha256:5dba5f530fec3f0988d83b78cc591b58c0b6eb8431a85edd1569a0539a8a5a0e", + "sha256:5dd9ca406499444f4c8299f803d4a14edf7890ecc595c8b1c7115c2342cadc5f", + "sha256:5f931a1c21dfa7a9c573ec1f50a31135ccce84e32507c54e1ea404894c5eb96f", + "sha256:63b82bb63de7c821428d513607e84c6d97d58afd1fe2eb645030bdc185440120", + "sha256:6508154df547b9d8651fa21e944c5946436614093d5a58d662a28448ed80df78", + "sha256:66c0061c91b3b9cf542131148ef7ecbecb2690d48d1612ec386de9d36766058f", + "sha256:6f0c02cbb9691b7c91d5009108f975f8ffeab5dff8f26d62e21c493060eff2a1", + "sha256:71aace0c42d53abe6fc7f726c5d3b60d90f3c5c055a447950ad6ea9cec2e37d9", + "sha256:7d97a4306898b05404a0dcdc32d9709b7d8832c0c542b861d9a826301719794e", + "sha256:7df1e1c05304f26faa49fa752a8c690126cf98b40b91d54e6e9cc3b7d6ffe8b7", + "sha256:8270252effc60b9642b423189a2fe90eb6b59e87cbee54549db3f5562ff8d1b8", + "sha256:867a5ad16892bf20e6c4ea2aab1971f45645ff3102ad29bd84c86027fa99997b", + "sha256:877473e675fdcc113c138813a5dd440da0769a2d81f4d86614e5d62b69497155", + "sha256:8892f89999ffd992208754851e5a052f6b5db70a1e3f7d54b17c5211e37a98c7", + "sha256:9a9845c4c6bb56e508651f005c4aeb0404e518c6f000d5a1123ab077ab769f5c", + "sha256:a1e6e96217a0f72e2b8629e271e1b280c6fa3fe6e59fa8f6701bec14e3354325", + "sha256:a8156e6a7f5e2a0ff0c5b21d6bcb45145efece1909efcbbbf48c56f8da68221d", + "sha256:a9506a7e80bcf6eacfff7f804c0ad5350c8c95b9010e4356a4b36f5322f09abb", + "sha256:af310ec8335016b5e52cae60cda4a4f2a60a788cbb949a4fbea13d441aa5a09e", + "sha256:b0297b1e05fd128d26cc2460c810d42e205d16d76799526dfa8c8ccd50e74959", + "sha256:bf68f4b2b6683e52bec69273562df15af352e5ed25d1b6641e7efddc5951d1a7", + "sha256:d0c1bc2fa9a7285719e5678584f6b92572a5b639d0e471bb8d4b650a1a910920", + "sha256:d4d9d6c1a455d4babd320203b918ccc7fcbefe308615c521062bc2ba1aa4d26e", + "sha256:db1fa631737dab9fa0b37f3979d8d2631e348c3b4e8325d6873c2541d0ae5a48", + "sha256:dd93ea5c0c7f3e25335ab7d22a507b1dc43976e1345508f845efc573d3d779d8", + "sha256:e0653479008a6118d2deec9ea3461d79331c65973ab4b25434ddc1018a982188", + "sha256:f44e517131a98f7a76696a7b21b164bcb85291cee106a23beccce454e1f433a4", + "sha256:f7ee479e96f7ee350db1cf24afa5685a5899e2b34992fb99e1f7c1b0b758d263" ], - "markers": "python_version >= '3.6'", - "version": "==3.4.0" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==5.4.0" } }, "develop": { @@ -833,108 +1383,120 @@ ], "version": "==0.7.12" }, - "apipkg": { - "hashes": [ - "sha256:37228cda29411948b422fae072f57e31d3396d2ee1c9783775980ee9c9990af6", - "sha256:58587dd4dc3daefad0487f6d9ae32b4542b185e1c36db6993290e7c41ca2b47c" - ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.5" - }, - "appdirs": { - "hashes": [ - "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41", - "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128" - ], - "version": "==1.4.4" - }, "attrs": { "hashes": [ - "sha256:31b2eced602aa8423c2aea9c76a724617ed67cf9513173fd3a4f03e3a929c7e6", - "sha256:832aa3cde19744e49938b91fea06d69ecb9e649c93ba974535d08ad92164f700" + "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1", + "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==20.3.0" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==21.2.0" }, "babel": { "hashes": [ - "sha256:9d35c22fcc79893c3ecc85ac4a56cde1ecf3f19c540bba0922308a6c06ca6fa5", - "sha256:da031ab54472314f210b0adcff1588ee5d1d1d0ba4dbd07b94dba82bde791e05" + "sha256:ab49e12b91d937cd11f0b67cb259a57ab4ad2b59ac7a3b41d6c06c0ac5b0def9", + "sha256:bc0c176f9f6a994582230df350aa6e05ba2ebe4b3ac317eab29d9be5d2768da0" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==2.9.0" + "version": "==2.9.1" + }, + "backports.entry-points-selectable": { + "hashes": [ + "sha256:988468260ec1c196dab6ae1149260e2f5472c9110334e5d51adcb77867361f6a", + "sha256:a6d9a871cde5e15b4c4a53e3d43ba890cc6861ec1332c9c2428c92f977192acc" + ], + "markers": "python_version >= '2.7'", + "version": "==1.1.0" }, "certifi": { "hashes": [ - "sha256:1a4995114262bffbc2413b159f2a1a480c969de6e6eb13ee966d470af86af59c", - "sha256:719a74fb9e33b9bd44cc7f3a8d94bc35e4049deebe19ba7d8e108280cfd59830" + "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee", + "sha256:50b1e4f8446b06f41be7dd6338db18e0990601dce795c2b1686458aa7e8fa7d8" ], - "version": "==2020.12.5" + "version": "==2021.5.30" }, - "chardet": { + "charset-normalizer": { "hashes": [ - "sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae", - "sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691" + "sha256:0c8911edd15d19223366a194a513099a302055a962bca2cec0f54b8b63175d8b", + "sha256:762c76ab955f265f68d4683641bb10d7f2723dbc10091505886e357e0b42760f", + "sha256:f23667ebe1084be45f6ae0538e4a5a865206544097e4e8bbcacf42cd02a348f3" ], - "markers": "python_version >= '3.1'", - "version": "==3.0.4" + "markers": "python_version >= '3'", + "version": "==2.0.4" }, "coverage": { "hashes": [ - "sha256:0203acd33d2298e19b57451ebb0bed0ab0c602e5cf5a818591b4918b1f97d516", - "sha256:0f313707cdecd5cd3e217fc68c78a960b616604b559e9ea60cc16795c4304259", - "sha256:1c6703094c81fa55b816f5ae542c6ffc625fec769f22b053adb42ad712d086c9", - "sha256:1d44bb3a652fed01f1f2c10d5477956116e9b391320c94d36c6bf13b088a1097", - "sha256:280baa8ec489c4f542f8940f9c4c2181f0306a8ee1a54eceba071a449fb870a0", - "sha256:29a6272fec10623fcbe158fdf9abc7a5fa032048ac1d8631f14b50fbfc10d17f", - "sha256:2b31f46bf7b31e6aa690d4c7a3d51bb262438c6dcb0d528adde446531d0d3bb7", - "sha256:2d43af2be93ffbad25dd959899b5b809618a496926146ce98ee0b23683f8c51c", - "sha256:3188a7dfd96f734a7498f37cde6598b1e9c084f1ca68bc1aa04e88db31168ab6", - "sha256:381ead10b9b9af5f64646cd27107fb27b614ee7040bb1226f9c07ba96625cbb5", - "sha256:47a11bdbd8ada9b7ee628596f9d97fbd3851bd9999d398e9436bd67376dbece7", - "sha256:4d6a42744139a7fa5b46a264874a781e8694bb32f1d76d8137b68138686f1729", - "sha256:50691e744714856f03a86df3e2bff847c2acede4c191f9a1da38f088df342978", - "sha256:530cc8aaf11cc2ac7430f3614b04645662ef20c348dce4167c22d99bec3480e9", - "sha256:582ddfbe712025448206a5bc45855d16c2e491c2dd102ee9a2841418ac1c629f", - "sha256:63808c30b41f3bbf65e29f7280bf793c79f54fb807057de7e5238ffc7cc4d7b9", - "sha256:71b69bd716698fa62cd97137d6f2fdf49f534decb23a2c6fc80813e8b7be6822", - "sha256:7858847f2d84bf6e64c7f66498e851c54de8ea06a6f96a32a1d192d846734418", - "sha256:78e93cc3571fd928a39c0b26767c986188a4118edc67bc0695bc7a284da22e82", - "sha256:7f43286f13d91a34fadf61ae252a51a130223c52bfefb50310d5b2deb062cf0f", - "sha256:86e9f8cd4b0cdd57b4ae71a9c186717daa4c5a99f3238a8723f416256e0b064d", - "sha256:8f264ba2701b8c9f815b272ad568d555ef98dfe1576802ab3149c3629a9f2221", - "sha256:9342dd70a1e151684727c9c91ea003b2fb33523bf19385d4554f7897ca0141d4", - "sha256:9361de40701666b034c59ad9e317bae95c973b9ff92513dd0eced11c6adf2e21", - "sha256:9669179786254a2e7e57f0ecf224e978471491d660aaca833f845b72a2df3709", - "sha256:aac1ba0a253e17889550ddb1b60a2063f7474155465577caa2a3b131224cfd54", - "sha256:aef72eae10b5e3116bac6957de1df4d75909fc76d1499a53fb6387434b6bcd8d", - "sha256:bd3166bb3b111e76a4f8e2980fa1addf2920a4ca9b2b8ca36a3bc3dedc618270", - "sha256:c1b78fb9700fc961f53386ad2fd86d87091e06ede5d118b8a50dea285a071c24", - "sha256:c3888a051226e676e383de03bf49eb633cd39fc829516e5334e69b8d81aae751", - "sha256:c5f17ad25d2c1286436761b462e22b5020d83316f8e8fcb5deb2b3151f8f1d3a", - "sha256:c851b35fc078389bc16b915a0a7c1d5923e12e2c5aeec58c52f4aa8085ac8237", - "sha256:cb7df71de0af56000115eafd000b867d1261f786b5eebd88a0ca6360cccfaca7", - "sha256:cedb2f9e1f990918ea061f28a0f0077a07702e3819602d3507e2ff98c8d20636", - "sha256:e8caf961e1b1a945db76f1b5fa9c91498d15f545ac0ababbe575cfab185d3bd8", - "sha256:ef221855191457fffeb909d5787d1807800ab4d0111f089e6c93ee68f577634d" + "sha256:004d1880bed2d97151facef49f08e255a20ceb6f9432df75f4eef018fdd5a78c", + "sha256:01d84219b5cdbfc8122223b39a954820929497a1cb1422824bb86b07b74594b6", + "sha256:040af6c32813fa3eae5305d53f18875bedd079960822ef8ec067a66dd8afcd45", + "sha256:06191eb60f8d8a5bc046f3799f8a07a2d7aefb9504b0209aff0b47298333302a", + "sha256:13034c4409db851670bc9acd836243aeee299949bd5673e11844befcb0149f03", + "sha256:13c4ee887eca0f4c5a247b75398d4114c37882658300e153113dafb1d76de529", + "sha256:184a47bbe0aa6400ed2d41d8e9ed868b8205046518c52464fde713ea06e3a74a", + "sha256:18ba8bbede96a2c3dde7b868de9dcbd55670690af0988713f0603f037848418a", + "sha256:1aa846f56c3d49205c952d8318e76ccc2ae23303351d9270ab220004c580cfe2", + "sha256:1d33b64c9acc1d9d90a67422ff50d79d0131552947a546570fa02328a9ea69b7", + "sha256:217658ec7187497e3f3ebd901afdca1af062b42cfe3e0dafea4cced3983739f6", + "sha256:24d4a7de75446be83244eabbff746d66b9240ae020ced65d060815fac3423759", + "sha256:2910f4d36a6a9b4214bb7038d537f015346f413a975d57ca6b43bf23d6563b53", + "sha256:2949cad1c5208b8298d5686d5a85b66aae46d73eec2c3e08c817dd3513e5848a", + "sha256:2a3859cb82dcbda1cfd3e6f71c27081d18aa251d20a17d87d26d4cd216fb0af4", + "sha256:2cafbbb3af0733db200c9b5f798d18953b1a304d3f86a938367de1567f4b5bff", + "sha256:2e0d881ad471768bf6e6c2bf905d183543f10098e3b3640fc029509530091502", + "sha256:30c77c1dc9f253283e34c27935fded5015f7d1abe83bc7821680ac444eaf7793", + "sha256:3487286bc29a5aa4b93a072e9592f22254291ce96a9fbc5251f566b6b7343cdb", + "sha256:372da284cfd642d8e08ef606917846fa2ee350f64994bebfbd3afb0040436905", + "sha256:41179b8a845742d1eb60449bdb2992196e211341818565abded11cfa90efb821", + "sha256:44d654437b8ddd9eee7d1eaee28b7219bec228520ff809af170488fd2fed3e2b", + "sha256:4a7697d8cb0f27399b0e393c0b90f0f1e40c82023ea4d45d22bce7032a5d7b81", + "sha256:51cb9476a3987c8967ebab3f0fe144819781fca264f57f89760037a2ea191cb0", + "sha256:52596d3d0e8bdf3af43db3e9ba8dcdaac724ba7b5ca3f6358529d56f7a166f8b", + "sha256:53194af30d5bad77fcba80e23a1441c71abfb3e01192034f8246e0d8f99528f3", + "sha256:5fec2d43a2cc6965edc0bb9e83e1e4b557f76f843a77a2496cbe719583ce8184", + "sha256:6c90e11318f0d3c436a42409f2749ee1a115cd8b067d7f14c148f1ce5574d701", + "sha256:74d881fc777ebb11c63736622b60cb9e4aee5cace591ce274fb69e582a12a61a", + "sha256:7501140f755b725495941b43347ba8a2777407fc7f250d4f5a7d2a1050ba8e82", + "sha256:796c9c3c79747146ebd278dbe1e5c5c05dd6b10cc3bcb8389dfdf844f3ead638", + "sha256:869a64f53488f40fa5b5b9dcb9e9b2962a66a87dab37790f3fcfb5144b996ef5", + "sha256:8963a499849a1fc54b35b1c9f162f4108017b2e6db2c46c1bed93a72262ed083", + "sha256:8d0a0725ad7c1a0bcd8d1b437e191107d457e2ec1084b9f190630a4fb1af78e6", + "sha256:900fbf7759501bc7807fd6638c947d7a831fc9fdf742dc10f02956ff7220fa90", + "sha256:92b017ce34b68a7d67bd6d117e6d443a9bf63a2ecf8567bb3d8c6c7bc5014465", + "sha256:970284a88b99673ccb2e4e334cfb38a10aab7cd44f7457564d11898a74b62d0a", + "sha256:972c85d205b51e30e59525694670de6a8a89691186012535f9d7dbaa230e42c3", + "sha256:9a1ef3b66e38ef8618ce5fdc7bea3d9f45f3624e2a66295eea5e57966c85909e", + "sha256:af0e781009aaf59e25c5a678122391cb0f345ac0ec272c7961dc5455e1c40066", + "sha256:b6d534e4b2ab35c9f93f46229363e17f63c53ad01330df9f2d6bd1187e5eaacf", + "sha256:b7895207b4c843c76a25ab8c1e866261bcfe27bfaa20c192de5190121770672b", + "sha256:c0891a6a97b09c1f3e073a890514d5012eb256845c451bd48f7968ef939bf4ae", + "sha256:c2723d347ab06e7ddad1a58b2a821218239249a9e4365eaff6649d31180c1669", + "sha256:d1f8bf7b90ba55699b3a5e44930e93ff0189aa27186e96071fac7dd0d06a1873", + "sha256:d1f9ce122f83b2305592c11d64f181b87153fc2c2bbd3bb4a3dde8303cfb1a6b", + "sha256:d314ed732c25d29775e84a960c3c60808b682c08d86602ec2c3008e1202e3bb6", + "sha256:d636598c8305e1f90b439dbf4f66437de4a5e3c31fdf47ad29542478c8508bbb", + "sha256:deee1077aae10d8fa88cb02c845cfba9b62c55e1183f52f6ae6a2df6a2187160", + "sha256:ebe78fe9a0e874362175b02371bdfbee64d8edc42a044253ddf4ee7d3c15212c", + "sha256:f030f8873312a16414c0d8e1a1ddff2d3235655a2174e3648b4fa66b3f2f1079", + "sha256:f0b278ce10936db1a37e6954e15a3730bea96a0997c26d7fee88e6c396c2086d", + "sha256:f11642dddbb0253cc8853254301b51390ba0081750a8ac03f20ea8103f0c56b6" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", - "version": "==5.3" + "version": "==5.5" }, "coveralls": { "hashes": [ - "sha256:2301a19500b06649d2ec4f2858f9c69638d7699a4c63027c5d53daba666147cc", - "sha256:b990ba1f7bc4288e63340be0433698c1efe8217f78c689d254c2540af3d38617" + "sha256:15a987d9df877fff44cd81948c5806ffb6eafb757b3443f737888358e96156ee", + "sha256:16c9e36985d1e1f420ab25259da7e0455090ddb49183aa707aed2d996506af12", + "sha256:aedfcc5296b788ebaf8ace8029376e5f102f67c53d1373f2e821515c15b36527" ], "index": "pypi", - "version": "==2.2.0" + "version": "==3.2.0" }, "distlib": { "hashes": [ - "sha256:8c09de2c67b3e7deef7184574fc060ab8a793e7adbb183d942c389c8b13c52fb", - "sha256:edf6116872c863e1aa9d5bb7cb5e05a022c519a4594dc703843343a9ddd9bff1" + "sha256:106fef6dc37dd8c0e2c0a60d3fca3e77460a48907f335fa28420463a6f799736", + "sha256:23e223426b28491b1ced97dc3bbe183027419dfc7982b4fa2f05d5f3ff10711c" ], - "version": "==0.3.1" + "version": "==0.3.2" }, "docopt": { "hashes": [ @@ -953,27 +1515,28 @@ }, "execnet": { "hashes": [ - "sha256:cacb9df31c9680ec5f95553976c4da484d407e85e41c83cb812aa014f0eddc50", - "sha256:d4efd397930c46415f62f8a31388d6be4f27a91d7550eb79bc64a756e0056547" + "sha256:8f694f3ba9cc92cab508b152dcfe322153975c29bda272e2fd7f3f00f36e47c5", + "sha256:a295f7cc774947aac58dde7fdc85f4aa00c42adf5d8f5468fc630c1acf30a142" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.7.1" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==1.9.0" }, "factory-boy": { "hashes": [ - "sha256:d8626622550c8ba31392f9e19fdbcef9f139cf1ad643c5923f20490a7b3e2e3d", - "sha256:ded73e49135c24bd4d3f45bf1eb168f8d290090f5cf4566b8df3698317dc9c08" + "sha256:1d3db4b44b8c8c54cdd8b83ae4bdb9aeb121e464400035f1f03ae0e1eade56a4", + "sha256:401cc00ff339a022f84d64a4339503d1689e8263a4478d876e58a3295b155c5b" ], "index": "pypi", - "version": "==3.1.0" + "version": "==3.2.0" }, "faker": { "hashes": [ - "sha256:1fcb415562ee6e2395b041e85fa6901d4708d30b84d54015226fa754ed0822c3", - "sha256:e8beccb398ee9b8cc1a91d9295121d66512b6753b4846eb1e7370545d46b3311" + "sha256:3e737576ff50cd98dfed643d6b3fd63194eca9df00e7f595960fe7da5220723d", + "sha256:b9e81e9da3dda3ac54189e034cfb943de576a259caeb226ccab43fcbcf6a7891", + "sha256:dc9713dcf35fa3e9b4c08caac97136d1c7daa9bc2fa7f703a8354f4b0f04053d" ], "markers": "python_version >= '3.6'", - "version": "==5.0.1" + "version": "==8.11.0" }, "filelock": { "hashes": [ @@ -985,12 +1548,11 @@ }, "idna": { "hashes": [ - "sha256:4a57a6379512ade94fa99e2fa46d3cd0f2f553040548d0e2958c6ed90ee48226", - "sha256:b307872f855b18632ce0c21c5e45be78c0ea7ae4c15c828c20788b26921eb3f6", - "sha256:b97d804b1e9b523befed77c48dacec60e6dcb0b5391d57af6a65a312a90648c0" + "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a", + "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==2.10" + "version": "==3.2" }, "imagesize": { "hashes": [ @@ -1000,22 +1562,6 @@ "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", "version": "==1.2.0" }, - "importlib-metadata": { - "hashes": [ - "sha256:6112e21359ef8f344e7178aa5b72dc6e62b38b0d008e6d3cb212c5b84df72013", - "sha256:b0c2d3b226157ae4517d9625decf63591461c66b3a808c2666d538946519d170" - ], - "markers": "python_version < '3.8'", - "version": "==3.1.1" - }, - "importlib-resources": { - "hashes": [ - "sha256:7b51f0106c8ec564b1bef3d9c588bc694ce2b92125bbb6278f4f2f5b54ec3592", - "sha256:a3d34a8464ce1d5d7c92b0ea4e921e696d86f2aa212e684451cb1482c8d84ed5" - ], - "markers": "python_version < '3.7'", - "version": "==3.3.0" - }, "iniconfig": { "hashes": [ "sha256:011e24c64b7f47f6ebd835bb12a743f2fbe9a26d4cecaa7f53bc4f35ee9da8b3", @@ -1026,62 +1572,90 @@ }, "jinja2": { "hashes": [ - "sha256:3f172970d5670703bd3812e8ca6459a9a7e069fa8e51b40195f83c81db191ec4", - "sha256:89aab215427ef59c34ad58735269eb58b1a5808103067f7bb9d5836c651b3bb0", - "sha256:f0a4641d3cf955324a89c04f3d94663aa4d638abe8f733ecd3582848e1c37035" + "sha256:1f06f2da51e7b56b8f238affdd6b4e2c61e39598a378cc49345bc1bd42a978a4", + "sha256:703f484b47a6af502e743c9122595cc812b0271f661722403114f71a79d0f5a4" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==2.11.2" + "markers": "python_version >= '3.6'", + "version": "==3.0.1" }, "markupsafe": { "hashes": [ - "sha256:00bc623926325b26bb9605ae9eae8a215691f33cae5df11ca5424f06f2d1f473", - "sha256:09027a7803a62ca78792ad89403b1b7a73a01c8cb65909cd876f7fcebd79b161", - "sha256:09c4b7f37d6c648cb13f9230d847adf22f8171b1ccc4d5682398e77f40309235", - "sha256:1027c282dad077d0bae18be6794e6b6b8c91d58ed8a8d89a89d59693b9131db5", - "sha256:13d3144e1e340870b25e7b10b98d779608c02016d5184cfb9927a9f10c689f42", - "sha256:19536834abffb3fa155017053c607cb835b2ecc6a3a2554a88043d991dffb736", - "sha256:24982cc2533820871eba85ba648cd53d8623687ff11cbb805be4ff7b4c971aff", - "sha256:29872e92839765e546828bb7754a68c418d927cd064fd4708fab9fe9c8bb116b", - "sha256:3d61f15e39611aacd91b7e71d903787da86d9e80896e683c0103fced9add7834", - "sha256:43a55c2930bbc139570ac2452adf3d70cdbb3cfe5912c71cdce1c2c6bbd9c5d1", - "sha256:46c99d2de99945ec5cb54f23c8cd5689f6d7177305ebff350a58ce5f8de1669e", - "sha256:500d4957e52ddc3351cabf489e79c91c17f6e0899158447047588650b5e69183", - "sha256:535f6fc4d397c1563d08b88e485c3496cf5784e927af890fb3c3aac7f933ec66", - "sha256:596510de112c685489095da617b5bcbbac7dd6384aeebeda4df6025d0256a81b", - "sha256:62fe6c95e3ec8a7fad637b7f3d372c15ec1caa01ab47926cfdf7a75b40e0eac1", - "sha256:6788b695d50a51edb699cb55e35487e430fa21f1ed838122d722e0ff0ac5ba15", - "sha256:6dd73240d2af64df90aa7c4e7481e23825ea70af4b4922f8ede5b9e35f78a3b1", - "sha256:717ba8fe3ae9cc0006d7c451f0bb265ee07739daf76355d06366154ee68d221e", - "sha256:7952deddf24b85c88dab48f6ec366ac6e39d2761b5280f2f9594911e03fcd064", - "sha256:79855e1c5b8da654cf486b830bd42c06e8780cea587384cf6545b7d9ac013a0b", - "sha256:7c1699dfe0cf8ff607dbdcc1e9b9af1755371f92a68f706051cc8c37d447c905", - "sha256:88e5fcfb52ee7b911e8bb6d6aa2fd21fbecc674eadd44118a9cc3863f938e735", - "sha256:8defac2f2ccd6805ebf65f5eeb132adcf2ab57aa11fdf4c0dd5169a004710e7d", - "sha256:98c7086708b163d425c67c7a91bad6e466bb99d797aa64f965e9d25c12111a5e", - "sha256:9add70b36c5666a2ed02b43b335fe19002ee5235efd4b8a89bfcf9005bebac0d", - "sha256:9bf40443012702a1d2070043cb6291650a0841ece432556f784f004937f0f32c", - "sha256:ade5e387d2ad0d7ebf59146cc00c8044acbd863725f887353a10df825fc8ae21", - "sha256:b00c1de48212e4cc9603895652c5c410df699856a2853135b3967591e4beebc2", - "sha256:b1282f8c00509d99fef04d8ba936b156d419be841854fe901d8ae224c59f0be5", - "sha256:b2051432115498d3562c084a49bba65d97cf251f5a331c64a12ee7e04dacc51b", - "sha256:ba59edeaa2fc6114428f1637ffff42da1e311e29382d81b339c1817d37ec93c6", - "sha256:c8716a48d94b06bb3b2524c2b77e055fb313aeb4ea620c8dd03a105574ba704f", - "sha256:cd5df75523866410809ca100dc9681e301e3c27567cf498077e8551b6d20e42f", - "sha256:cdb132fc825c38e1aeec2c8aa9338310d29d337bebbd7baa06889d09a60a1fa2", - "sha256:e249096428b3ae81b08327a63a485ad0878de3fb939049038579ac0ef61e17e7", - "sha256:e8313f01ba26fbbe36c7be1966a7b7424942f670f38e666995b88d012765b9be" + "sha256:01a9b8ea66f1658938f65b93a85ebe8bc016e6769611be228d797c9d998dd298", + "sha256:023cb26ec21ece8dc3907c0e8320058b2e0cb3c55cf9564da612bc325bed5e64", + "sha256:0446679737af14f45767963a1a9ef7620189912317d095f2d9ffa183a4d25d2b", + "sha256:060b790af48b3d1cb6776a1e8956fb41106c6befaca717181c41a1728147dc73", + "sha256:0717a7390a68be14b8c793ba258e075c6f4ca819f15edfc2a3a027c823718567", + "sha256:0955295dd5eec6cb6cc2fe1698f4c6d84af2e92de33fbcac4111913cd100a6ff", + "sha256:0d4b31cc67ab36e3392bbf3862cfbadac3db12bdd8b02a2731f509ed5b829724", + "sha256:10f82115e21dc0dfec9ab5c0223652f7197feb168c940f3ef61563fc2d6beb74", + "sha256:168cd0a3642de83558a5153c8bd34f175a9a6e7f6dc6384b9655d2697312a646", + "sha256:1d609f577dc6e1aa17d746f8bd3c31aa4d258f4070d61b2aa5c4166c1539de35", + "sha256:1f2ade76b9903f39aa442b4aadd2177decb66525062db244b35d71d0ee8599b6", + "sha256:2a7d351cbd8cfeb19ca00de495e224dea7e7d919659c2841bbb7f420ad03e2d6", + "sha256:2d7d807855b419fc2ed3e631034685db6079889a1f01d5d9dac950f764da3dad", + "sha256:2ef54abee730b502252bcdf31b10dacb0a416229b72c18b19e24a4509f273d26", + "sha256:36bc903cbb393720fad60fc28c10de6acf10dc6cc883f3e24ee4012371399a38", + "sha256:37205cac2a79194e3750b0af2a5720d95f786a55ce7df90c3af697bfa100eaac", + "sha256:3c112550557578c26af18a1ccc9e090bfe03832ae994343cfdacd287db6a6ae7", + "sha256:3dd007d54ee88b46be476e293f48c85048603f5f516008bee124ddd891398ed6", + "sha256:47ab1e7b91c098ab893b828deafa1203de86d0bc6ab587b160f78fe6c4011f75", + "sha256:49e3ceeabbfb9d66c3aef5af3a60cc43b85c33df25ce03d0031a608b0a8b2e3f", + "sha256:4efca8f86c54b22348a5467704e3fec767b2db12fc39c6d963168ab1d3fc9135", + "sha256:53edb4da6925ad13c07b6d26c2a852bd81e364f95301c66e930ab2aef5b5ddd8", + "sha256:5855f8438a7d1d458206a2466bf82b0f104a3724bf96a1c781ab731e4201731a", + "sha256:594c67807fb16238b30c44bdf74f36c02cdf22d1c8cda91ef8a0ed8dabf5620a", + "sha256:5bb28c636d87e840583ee3adeb78172efc47c8b26127267f54a9c0ec251d41a9", + "sha256:60bf42e36abfaf9aff1f50f52644b336d4f0a3fd6d8a60ca0d054ac9f713a864", + "sha256:611d1ad9a4288cf3e3c16014564df047fe08410e628f89805e475368bd304914", + "sha256:6557b31b5e2c9ddf0de32a691f2312a32f77cd7681d8af66c2692efdbef84c18", + "sha256:693ce3f9e70a6cf7d2fb9e6c9d8b204b6b39897a2c4a1aa65728d5ac97dcc1d8", + "sha256:6a7fae0dd14cf60ad5ff42baa2e95727c3d81ded453457771d02b7d2b3f9c0c2", + "sha256:6c4ca60fa24e85fe25b912b01e62cb969d69a23a5d5867682dd3e80b5b02581d", + "sha256:6fcf051089389abe060c9cd7caa212c707e58153afa2c649f00346ce6d260f1b", + "sha256:7d91275b0245b1da4d4cfa07e0faedd5b0812efc15b702576d103293e252af1b", + "sha256:905fec760bd2fa1388bb5b489ee8ee5f7291d692638ea5f67982d968366bef9f", + "sha256:97383d78eb34da7e1fa37dd273c20ad4320929af65d156e35a5e2d89566d9dfb", + "sha256:984d76483eb32f1bcb536dc27e4ad56bba4baa70be32fa87152832cdd9db0833", + "sha256:99df47edb6bda1249d3e80fdabb1dab8c08ef3975f69aed437cb69d0a5de1e28", + "sha256:a30e67a65b53ea0a5e62fe23682cfe22712e01f453b95233b25502f7c61cb415", + "sha256:ab3ef638ace319fa26553db0624c4699e31a28bb2a835c5faca8f8acf6a5a902", + "sha256:add36cb2dbb8b736611303cd3bfcee00afd96471b09cda130da3581cbdc56a6d", + "sha256:b2b16984860d962a9f7f42c8f84ecdb74993b803a705309bb0acea7f032e37e7", + "sha256:b2f4bf27480f5e5e8ce285a8c8fd176c0b03e93dcc6646477d4630e83440c6a9", + "sha256:b7f2d075102dc8c794cbde1947378051c4e5180d52d276987b8d28a3bd58c17d", + "sha256:baa1a4e8f868845af802979fcdbf0bb11f94f1cb7ced4c4b8a351bb60d108145", + "sha256:be98f628055368795d818ebf93da628541e10b75b41c559fdf36d104c5787066", + "sha256:bf5d821ffabf0ef3533c39c518f3357b171a1651c1ff6827325e4489b0e46c3c", + "sha256:c47adbc92fc1bb2b3274c4b3a43ae0e4573d9fbff4f54cd484555edbf030baf1", + "sha256:d7f9850398e85aba693bb640262d3611788b1f29a79f0c93c565694658f4071f", + "sha256:d8446c54dc28c01e5a2dbac5a25f071f6653e6e40f3a8818e8b45d790fe6ef53", + "sha256:e0f138900af21926a02425cf736db95be9f4af72ba1bb21453432a07f6082134", + "sha256:e9936f0b261d4df76ad22f8fee3ae83b60d7c3e871292cd42f40b81b70afae85", + "sha256:f5653a225f31e113b152e56f154ccbe59eeb1c7487b39b9d9f9cdb58e6c79dc5", + "sha256:f826e31d18b516f653fe296d967d700fddad5901ae07c622bb3705955e1faa94", + "sha256:f8ba0e8349a38d3001fae7eadded3f6606f0da5d748ee53cc1dab1d6527b9509", + "sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51", + "sha256:fa130dd50c57d53368c9d59395cb5526eda596d3ffe36666cd81a44d56e48872" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.1.1" + "markers": "python_version >= '3.6'", + "version": "==2.0.1" }, "packaging": { "hashes": [ - "sha256:05af3bb85d320377db281cf254ab050e1a7ebcbf5410685a9a407e18a1f81236", - "sha256:eb41423378682dadb7166144a4926e443093863024de508ca5c9737d6bc08376" + "sha256:7dc96269f53a4ccec5c0670940a4281106dd0bb343f47b7471f779df49c2fbe7", + "sha256:c86254f9220d55e31cc94d69bade760f0847da8000def4dfe1c6b872fd14ff14" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==20.7" + "markers": "python_version >= '3.6'", + "version": "==21.0" + }, + "platformdirs": { + "hashes": [ + "sha256:4666d822218db6a262bdfdc9c39d21f23b4cfdb08af331a81e92751daf6c866c", + "sha256:632daad3ab546bd8e6af0537d09805cec458dce201bccfe23012df73332e181e", + "sha256:ded78f63b43fee3256f4b4eb42cc297954d4ae6d198d04418107c25e8a8682ed" + ], + "markers": "python_version >= '3.6'", + "version": "==2.2.0" }, "pluggy": { "hashes": [ @@ -1093,27 +1667,28 @@ }, "py": { "hashes": [ - "sha256:366389d1db726cd2fcfc79732e75410e5fe4d31db13692115529d34069a043c2", - "sha256:9ca6883ce56b4e8da7e79ac18787889fa5206c79dcc67fb065376cd2fe03f342" + "sha256:21b81bda15b66ef5e1a777a21c4dcd9c20ad3efd0b3f817e7a809035269e1bd3", + "sha256:3b80836aa6d1feeaa108e046da6423ab8f6ceda6468545ae8d02d9d58d18818a" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.9.0" + "version": "==1.10.0" }, "pycodestyle": { "hashes": [ - "sha256:2295e7b2f6b5bd100585ebcb1f616591b652db8a741695b3d8f5d28bdc934367", - "sha256:c58a7d2815e0e8d7972bf1803331fb0152f867bd89adf8a01dfd55085434192e" + "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068", + "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef" ], "index": "pypi", - "version": "==2.6.0" + "version": "==2.7.0" }, "pygments": { "hashes": [ - "sha256:ccf3acacf3782cbed4a989426012f1c535c9a90d3a7fc3f16d231b9372d2b716", - "sha256:f275b6c0909e5dafd2d6269a656aa90fa58ebf4a74f8fcf9053195d226b24a08" + "sha256:9cf781c67f72dac7c566b333a2daeb4ef04a0d445205eaa03abe97959d0beb0e", + "sha256:b8e67fe6af78f492b3c4b3e2970c0624cbf08beb1e493b2c99b9fa1b67a20380", + "sha256:f398865f7eb6874156579fdf36bc840a03cab64d1cde9e93d68f46a425ec52c6" ], "markers": "python_version >= '3.5'", - "version": "==2.7.3" + "version": "==2.10.0" }, "pyparsing": { "hashes": [ @@ -1125,27 +1700,27 @@ }, "pytest": { "hashes": [ - "sha256:4288fed0d9153d9646bfcdf0c0428197dba1ecb27a33bb6e031d002fa88653fe", - "sha256:c0a7e94a8cdbc5422a51ccdad8e6f1024795939cc89159a0ae7f0b316ad3823e" + "sha256:50bcad0a0b9c5a72c8e4e7c9855a3ad496ca6a881a3641b4260605450772c54b", + "sha256:91ef2131a9bd6be8f76f1f08eac5c5317221d6ad1e143ae03894b862e8976890" ], "index": "pypi", - "version": "==6.1.2" + "version": "==6.2.4" }, "pytest-cov": { "hashes": [ - "sha256:45ec2d5182f89a81fc3eb29e3d1ed3113b9e9a873bcddb2a71faaab066110191", - "sha256:47bd0ce14056fdd79f93e1713f88fad7bdcc583dcd7783da86ef2f085a0bb88e" + "sha256:261bb9e47e65bd099c89c3edf92972865210c36813f80ede5277dceb77a4a62a", + "sha256:261ceeb8c227b726249b376b8526b600f38667ee314f910353fa318caa01f4d7" ], "index": "pypi", - "version": "==2.10.1" + "version": "==2.12.1" }, "pytest-django": { "hashes": [ - "sha256:10e384e6b8912ded92db64c58be8139d9ae23fb8361e5fc139d8e4f8fc601bc2", - "sha256:26f02c16d36fd4c8672390deebe3413678d89f30720c16efb8b2a6bf63b9041f" + "sha256:65783e78382456528bd9d79a35843adde9e6a47347b20464eb2c885cb0f1f606", + "sha256:b5171e3798bf7e3fc5ea7072fe87324db67a4dd9f1192b037fed4cc3c1b7f455" ], "index": "pypi", - "version": "==4.1.0" + "version": "==4.4.0" }, "pytest-env": { "hashes": [ @@ -1174,65 +1749,65 @@ }, "pytest-xdist": { "hashes": [ - "sha256:7c629016b3bb006b88ac68e2b31551e7becf173c76b977768848e2bbed594d90", - "sha256:82d938f1a24186520e2d9d3a64ef7d9ac7ecdf1a0659e095d18e596b8cbd0672" + "sha256:e8ecde2f85d88fbcadb7d28cb33da0fa29bca5cf7d5967fa89fc0e97e5299ea5", + "sha256:ed3d7da961070fce2a01818b51f6888327fb88df4379edeb6b9d990e789d9c8d" ], "index": "pypi", - "version": "==2.1.0" + "version": "==2.3.0" }, "python-dateutil": { "hashes": [ - "sha256:73ebfe9dbf22e832286dafa60473e4cd239f8592f699aa5adaf10050e6e1823c", - "sha256:75bb3f31ea686f1197762692a9ee6a7550b59fc6ca3a1f4b5d7e32fb98e2da2a" + "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86", + "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9" ], "index": "pypi", - "version": "==2.8.1" + "version": "==2.8.2" }, "pytz": { "hashes": [ - "sha256:3e6b7dd2d1e0a59084bcee14a17af60c5c562cdc16d828e8eba2e683d3a7e268", - "sha256:5c55e189b682d420be27c6995ba6edce0c0a77dd67bfbe2ae6607134d5851ffd" + "sha256:83a4a90894bf38e243cf052c8b58f381bfe9a7a483f6a9cab140bc7f702ac4da", + "sha256:eb10ce3e7736052ed3623d49975ce333bcd712c7bb19a58b9e2089d4057d0798" ], - "version": "==2020.4" + "version": "==2021.1" }, "requests": { "hashes": [ - "sha256:7f1a0b932f4a60a1a65caa4263921bb7d9ee911957e0ae4a23a6dd08185ad5f8", - "sha256:e786fa28d8c9154e6a4de5d46a1d921b8749f8b74e28bde23768e5e16eece998" + "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24", + "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", - "version": "==2.25.0" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5'", + "version": "==2.26.0" }, "six": { "hashes": [ - "sha256:30639c035cdb23534cd4aa2dd52c3bf48f06e5f4a941509c8bafd8ce11080259", - "sha256:8b74bedcbbbaca38ff6d7491d76f2b06b3592611af620f8426e82dddb04a5ced" + "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926", + "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==1.15.0" + "version": "==1.16.0" }, "snowballstemmer": { "hashes": [ - "sha256:209f257d7533fdb3cb73bdbd24f436239ca3b2fa67d56f6ff88e86be08cc5ef0", - "sha256:df3bac3df4c2c01363f3dd2cfa78cce2840a79b9f1c2d2de9ce8d31683992f52" + "sha256:b51b447bea85f9968c13b650126a888aabd4cb4463fca868ec596826325dedc2", + "sha256:e997baa4f2e9139951b6f4c631bad912dfd3c792467e2f03d7239464af90e914" ], - "version": "==2.0.0" + "version": "==2.1.0" }, "sphinx": { "hashes": [ - "sha256:1e8d592225447104d1172be415bc2972bd1357e3e12fdc76edf2261105db4300", - "sha256:d4e59ad4ea55efbb3c05cde3bfc83bfc14f0c95aa95c3d75346fcce186a47960" + "sha256:41cad293f954f7d37f803d97eb184158cfd90f51195131e94875bc07cd08b93c", + "sha256:c314c857e7cd47c856d2c5adff514ac2e6495f8b8e0f886a8a37e9305dfea0d8" ], "index": "pypi", - "version": "==3.3.1" + "version": "==3.4.3" }, "sphinx-rtd-theme": { "hashes": [ - "sha256:22c795ba2832a169ca301cd0a083f7a434e09c538c70beb42782c073651b707d", - "sha256:373413d0f82425aaa28fb288009bf0d0964711d347763af2f1b65cafcb028c82" + "sha256:32bd3b5d13dc8186d7a42fc816a23d32e83a4827d7d9882948e7b837c232da5a", + "sha256:4a05bdbe8b1446d77a01e20a23ebc6777c74f43237035e76be89699308987d6f" ], "index": "pypi", - "version": "==0.5.0" + "version": "==0.5.2" }, "sphinxcontrib-applehelp": { "hashes": [ @@ -1252,11 +1827,11 @@ }, "sphinxcontrib-htmlhelp": { "hashes": [ - "sha256:3c0bc24a2c41e340ac37c85ced6dafc879ab485c095b1d65d2461ac2f7cca86f", - "sha256:e8f5bb7e31b2dbb25b9cc435c8ab7a79787ebf7f906155729338f3156d93659b" + "sha256:d412243dfb797ae3ec2b59eca0e52dac12e75a241bf0e4eb861e450d06c6ed07", + "sha256:f5f8bb2d0d629f398bf47d0d69c07bc13b65f75a81ad9e2f71a63d4b7a2f6db2" ], - "markers": "python_version >= '3.5'", - "version": "==1.0.3" + "markers": "python_version >= '3.6'", + "version": "==2.0.0" }, "sphinxcontrib-jsmath": { "hashes": [ @@ -1276,11 +1851,11 @@ }, "sphinxcontrib-serializinghtml": { "hashes": [ - "sha256:eaa0eccc86e982a9b939b2b82d12cc5d013385ba5eadcc7e4fed23f4405f77bc", - "sha256:f242a81d423f59617a8e5cf16f5d4d74e28ee9a66f9e5b637a18082991db5a9a" + "sha256:352a9a00ae864471d3a7ead8d7d79f5fc0b57e8b3f95e9867eb9eb28999b92fd", + "sha256:aa5f6de5dfdf809ef505c4895e51ef5c9eac17d0f287933eb49ec495280b6952" ], "markers": "python_version >= '3.5'", - "version": "==1.1.4" + "version": "==1.1.5" }, "termcolor": { "hashes": [ @@ -1306,35 +1881,29 @@ }, "tox": { "hashes": [ - "sha256:42ce19ce5dc2f6d6b1fdc5666c476e1f1e2897359b47e0aa3a5b774f335d57c2", - "sha256:4321052bfe28f9d85082341ca8e233e3ea901fdd14dab8a5d3fbd810269fbaf6" + "sha256:ae442d4d51d5a3afb3711e4c7d94f5ca8461afd27c53f5dd994aba34896cf02d", + "sha256:d45d39203b10fdb2f6887c6779865e31de82cea07419a739844cc4bd4b3493e2", + "sha256:f5ad550fb63b72cdac1e5ef24565d7cc0766de585411ce4da945f0c150807dfc" ], "index": "pypi", - "version": "==3.20.1" + "version": "==3.24.2" }, "urllib3": { "hashes": [ - "sha256:19188f96923873c92ccb987120ec4acaa12f0461fa9ce5d3d0772bc965a39e08", - "sha256:d8ff90d979214d7b4f8ce956e80f4028fc6860e4431f731ea4a8c08f23f99473" + "sha256:39fb8672126159acb139a7718dd10806104dec1e2f0f6c88aab05d17df10c8d4", + "sha256:f57b4c16c62fa2760b7e3d97c35b255512fb6b59a259730f36ba32ce9f8e342f" ], "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4'", - "version": "==1.26.2" + "version": "==1.26.6" }, "virtualenv": { "hashes": [ - "sha256:54b05fc737ea9c9ee9f8340f579e5da5b09fb64fd010ab5757eb90268616907c", - "sha256:b7a8ec323ee02fb2312f098b6b4c9de99559b462775bc8fe3627a73706603c1b" + "sha256:4df1e1476915c986e2fece1e64c78d34bf49551fec73e303b534d37f785f0929", + "sha256:9ef4e8ee4710826e98ff3075c9a4739e2cb1040de6a2a8d35db0055840dc96a0", + "sha256:e4670891b3a03eb071748c569a87cceaefbf643c5bac46d996c5a45c34aa0f06" ], - "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3'", - "version": "==20.2.2" - }, - "zipp": { - "hashes": [ - "sha256:102c24ef8f171fd729d46599845e95c7ab894a4cf45f5de11a44cc7444fb1108", - "sha256:ed5eee1974372595f9e416cc7bbeeb12335201d8081ca8a0743c954d4446e5cb" - ], - "markers": "python_version >= '3.6'", - "version": "==3.4.0" + "markers": "python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4'", + "version": "==20.7.2" } } } diff --git a/README.md b/README.md index e754669a8..5c51c6f24 100644 --- a/README.md +++ b/README.md @@ -1,113 +1,116 @@ -[![Build Status](https://travis-ci.org/jonaswinkler/paperless-ng.svg?branch=master)](https://travis-ci.org/jonaswinkler/paperless-ng) +[![ci](https://github.com/jonaswinkler/paperless-ng/workflows/ci/badge.svg)](https://github.com/jonaswinkler/paperless-ng/actions) +![Ansible Role](https://github.com/jonaswinkler/paperless-ng/workflows/Ansible%20Role/badge.svg) +[![Crowdin](https://badges.crowdin.net/paperless-ng/localized.svg)](https://crowdin.com/project/paperless-ng) [![Documentation Status](https://readthedocs.org/projects/paperless-ng/badge/?version=latest)](https://paperless-ng.readthedocs.io/en/latest/?badge=latest) +[![Gitter](https://badges.gitter.im/paperless-ng/community.svg)](https://gitter.im/paperless-ng/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) [![Docker Hub Pulls](https://img.shields.io/docker/pulls/jonaswinkler/paperless-ng.svg)](https://hub.docker.com/r/jonaswinkler/paperless-ng) [![Coverage Status](https://coveralls.io/repos/github/jonaswinkler/paperless-ng/badge.svg?branch=master)](https://coveralls.io/github/jonaswinkler/paperless-ng?branch=master) # Paperless-ng -[Paperless](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and others that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents. +[Paperless (click me)](https://github.com/the-paperless-project/paperless) is an application by Daniel Quinn and contributors that indexes your scanned documents and allows you to easily search for documents and store metadata alongside your documents. -Paperless-ng is a fork of the original project, adding a new interface and many other changes under the hood. For a detailed list of changes, see below. +Paperless-ng is a fork of the original project, adding a new interface and many other changes under the hood. These key points should help you decide whether Paperless-ng is something you would prefer over Paperless: -This project is still in development and some things may not work as expected. +* Interface: The new front end is the main interface for Paperless-ng, the old interface still exists but most customizations (such as thumbnails for the document list) have been removed.0 +* Encryption: Paperless-ng does not support GnuPG anymore, since storing your data on encrypted file systems (that you optionally mount on demand) achieves about the same result. +* Resource usage: Paperless-ng does use a bit more resources than Paperless. Running the web server requires about 300MB of RAM or more, depending on the configuration. While adding documents, it requires about 300MB additional RAM, depending on the document. It still runs on Raspberry Pi (many users do that), but it has been generally geared to better use the resources of more powerful systems. +* API changes: If you rely on the REST API of paperless, some of its functionality has been changed. + +For a detailed list of changes, have a look at the [change log](https://paperless-ng.readthedocs.io/en/latest/changelog.html) in the documentation, especially the section about the [0.9.0 release](https://paperless-ng.readthedocs.io/en/latest/changelog.html#paperless-ng-0-9-0). # How it Works Paperless does not control your scanner, it only helps you deal with what your scanner produces. -1. Buy a document scanner that can write to a place on your network. If you need some inspiration, have a look at the [scanner recommendations](https://paperless-ng.readthedocs.io/en/latest/scanners.html) page. -2. Set it up to "scan to FTP" or something similar. It should be able to push scanned images to a server without you having to do anything. Of course if your scanner doesn't know how to automatically upload the file somewhere, you can always do that manually. Paperless doesn't care how the documents get into its local consumption directory. -3. Have the target server run the Paperless consumption script to OCR the file and index it into a local database. -4. Use the web frontend to sift through the database and find what you want. -5. Download the PDF you need/want via the web interface and do whatever you like with it. You can even print it and send it as if it's the original. In most cases, no one will care or notice. +1. Buy a document scanner that can write to a place on your network. If you need some inspiration, have a look at the [scanner recommendations](https://paperless-ng.readthedocs.io/en/latest/scanners.html) page. Set it up to "scan to FTP" or something similar. It should be able to push scanned images to a server without you having to do anything. Of course if your scanner doesn't know how to automatically upload the file somewhere, you can always do that manually. Paperless doesn't care how the documents get into its local consumption directory. + + - Alternatively, you can use any of the mobile scanning apps out there. We have an app that allows you to share documents with paperless, if you're on Android. See the section on affiliated projects below. + +2. Wait for paperless to process your files. OCR is expensive, and depending on the power of your machine, this might take a bit of time. +3. Use the web frontend to sift through the database and find what you want. +4. Download the PDF you need/want via the web interface and do whatever you like with it. You can even print it and send it as if it's the original. In most cases, no one will care or notice. Here's what you get: ![Dashboard](https://github.com/jonaswinkler/paperless-ng/raw/master/docs/_static/screenshots/dashboard.png) +If you want to see paperless-ng in action, [more screenshots are available in the documentation](https://paperless-ng.readthedocs.io/en/latest/screenshots.html). + # Features * Performs OCR on your documents, adds selectable text to image only documents and adds tags, correspondents and document types to your documents. -* Single page application front end. Should be pretty snappy. Will be mobile friendly in the future. +* Supports PDF documents, images, plain text files, and Office documents (Word, Excel, Powerpoint, and LibreOffice equivalents). + * Office document support is optional and provided by Apache Tika (see [configuration](https://paperless-ng.readthedocs.io/en/latest/configuration.html#tika-settings)) +* Paperless stores your documents plain on disk. Filenames and folders are managed by paperless and their format can be configured freely. +* Single page application front end. * Includes a dashboard that shows basic statistics and has document upload. * Filtering by tags, correspondents, types, and more. * Customizable views can be saved and displayed on the dashboard. - * Full text search with auto completion, scored results and query highlighting allows you to quickly find what you need. +* Full text search helps you find what you need. + * Auto completion suggests relevant words from your documents. + * Results are sorted by relevance to your search query. + * Highlighting shows you which parts of the document matched the query. + * Searching for similar documents ("More like this") * Email processing: Paperless adds documents from your email accounts. * Configure multiple accounts and filters for each account. - * When adding documents from mails, paperless can move these mails to a new folder, mark them as read, flag them or delete them. + * When adding documents from mail, paperless can move these mail to a new folder, mark them as read, flag them as important or delete them. * Machine learning powered document matching. * Paperless learns from your documents and will be able to automatically assign tags, correspondents and types to documents once you've stored a few documents in paperless. -* We have a mobile app that offers a 'Share with paperless' option over at https://github.com/qcasey/paperless_share. You can use that in combination with any of the mobile scanning apps out there. It's still a little rough around the edges, but it works! -* A task processor that processes documents in parallel and also tells you when something goes wrong. On modern multi core systems, consumption is blazing fast. -* Code cleanup in many, MANY areas. Some of the code from OG paperless was just overly complicated. -* More tests, more stability. - -If you want to see some screenshots of paperless-ng in action, [some are available in the documentation](https://paperless-ng.readthedocs.io/en/latest/screenshots.html). - -For a complete list of changes from paperless, check out the [changelog](https://paperless-ng.readthedocs.io/en/latest/changelog.html) - -# Roadmap for 1.0 - -- Make the front end nice (except mobile). -- Test coverage at 90%. -- Fix whatever bugs I and you find. - -## Roadmap for versions beyond 1.0 - -These are things that I want to add to paperless eventually. They are sorted by priority. - -- **Bulk editing**. Add/remove metadata from multiple documents at once. -- **More search.** The search backend is incredibly versatile and customizable. Searching is the most important feature of this project and thus, I want to implement things like: - - Group and limit search results by correspondent, show “more from this” links in the results. - - Ability to search for “Similar documents” in the search results -- **Nested tags**. Organize tags in a hierarchical structure. This will combine the benefits of folders and tags in one coherent system. -- **An interactive consumer** that shows its progress for documents it processes on the web page. - - With live updates ans websockets. This already works on a dev branch, but requires a lot of new dependencies, which I'm not particular happy about. - - Notifications when a document was added with buttons to open the new document right away. -- **Arbitrary tag colors**. Allow the selection of any color with a color picker. - -## On the chopping block. - -- **GnuPG encrypion.** [Here's a note about encryption in paperless](https://paperless-ng.readthedocs.io/en/latest/administration.html#managing-encryption). The gist of it is that I don't see which attacks this implementation protects against. It gives a false sense of security to users who don't care about how it works. +* Optimized for multi core systems: Paperless-ng consumes multiple documents in parallel. +* The integrated sanity checker makes sure that your document archive is in good health. # Getting started -The recommended way to deploy paperless is docker-compose. Don't clone the repository, grab the latest release to get started instead. The dockerfiles archive contains just the docker files which will pull the image from docker hub. The source archive contains everything you need to build the docker image yourself (i.e. if you want to run on Raspberry Pi). +The recommended way to deploy paperless is docker-compose. The files in the /docker/compose directory are configured to pull the image from Docker Hub. Read the [documentation](https://paperless-ng.readthedocs.io/en/latest/setup.html#installation) on how to get started. -Alternatively, you can install the dependencies and setup apache and a database server yourself. The documenation has information about the individual components of paperless that you need to take care of. +Alternatively, you can install the dependencies and setup apache and a database server yourself. The documenation has a step by step guide on how to do it. Consider giving the Ansible role a shot, this essentially automates the entire bare metal installation process. -# Migrating to paperless-ng +# Migrating from Paperless to Paperless-ng -Read the section about [migration](https://paperless-ng.readthedocs.io/en/latest/setup.html#migration-to-paperless-ng) in the documentation. Its also entirely possible to go back to paperless by reverting the database migrations. +Read the section about [migration](https://paperless-ng.readthedocs.io/en/latest/setup.html#migration-to-paperless-ng) in the documentation. Its also entirely possible to go back to Paperless by reverting the database migrations. # Documentation The documentation for Paperless-ng is available on [ReadTheDocs](https://paperless-ng.readthedocs.io/). -# Suggestions? Questions? Something not working? +# Translation -Please open an issue and start a discussion about it! +Paperless is available in many different languages. Translation is coordinated at crowdin. If you want to help out by translating paperless into your language, please head over to https://github.com/jonaswinkler/paperless-ng/issues/212 for details! + +# Feature Requests + +Feature requests can be submitted via [GitHub Discussions](https://github.com/jonaswinkler/paperless-ng/discussions/categories/feature-requests), you can search for existing ideas, add your own and vote for the ones you care about! Note that some older feature requests can also be found under [issues](https://github.com/jonaswinkler/paperless-ng/issues). + +# Questions? Something not working? + +For bugs please [open an issue](https://github.com/jonaswinkler/paperless-ng/issues) or [start a discussion](https://github.com/jonaswinkler/paperless-ng/discussions) if you have questions. ## Feel like helping out? -There's still lots of things to be done, just have a look at that issue log. If you feel like contributing to the project, please do! Bug fixes and improvements to the front end (I just can't seem to get some of these CSS things right) are always welcome. The documentation has some basic information on how to get started. +There's still lots of things to be done, just have a look at open issues & discussions. If you feel like contributing to the project, please do! Bug fixes and improvements to the front end (I just can't seem to get some of these CSS things right) are always welcome. The documentation has some basic information on how to get started. -If you want to implement something big: Please start a discussion about that in the issues! Maybe I've already had something similar in mind and we can make it happen together. However, keep in mind that the general roadmap is to make the existing features stable and get them tested. See the roadmap above. +If you want to implement something big: Please start a discussion about that! Maybe I've already had something similar in mind and we can make it happen together. However, keep in mind that the general roadmap is to make the existing features stable and get them tested. # Affiliated Projects Paperless has been around a while now, and people are starting to build stuff on top of it. If you're one of those people, we can add your project to this list: -* [Paperless App](https://github.com/bauerj/paperless_app): An Android/iOS app for Paperless. We're working on making this compatible. -* [Paperless Desktop](https://github.com/thomasbrueggemann/paperless-desktop): A desktop UI for your Paperless installation. Runs on Mac, Linux, and Windows. -* [ansible-role-paperless](https://github.com/ovv/ansible-role-paperless): An easy way to get Paperless running via Ansible. +* [Paperless App](https://github.com/bauerj/paperless_app): An Android/iOS app for Paperless. Updated to work with paperless-ng. +* [Paperless Share](https://github.com/qcasey/paperless_share). Share any files from your Android application with paperless. Very simple, but works with all of the mobile scanning apps out there that allow you to share scanned documents. +* [Scan to Paperless](https://github.com/sbrunner/scan-to-paperless): Scan and prepare (crop, deskew, OCR, ...) your documents for Paperless. + +These projects also exist, but their status and compatibility with paperless-ng is unknown. + * [paperless-cli](https://github.com/stgarf/paperless-cli): A golang command line binary to interact with a Paperless instance. -Compatibility with Paperless-ng is unknown. +This project also exists, but needs updates to be compatile with paperless-ng. + +* [Paperless Desktop](https://github.com/thomasbrueggemann/paperless-desktop): A desktop UI for your Paperless installation. Runs on Mac, Linux, and Windows. + Known issues on Mac: (Could not load reminders and documents) # Important Note -Document scanners are typically used to scan sensitive documents. Things like your social insurance number, tax records, invoices, etc. Everything is stored in the clear without encryption by default (it needs to be searchable, so if someone has ideas on how to do that on encrypted data, I'm all ears). This means that Paperless should never be run on an untrusted host. Instead, I recommend that if you do want to use it, run it locally on a server in your own home. +Document scanners are typically used to scan sensitive documents. Things like your social insurance number, tax records, invoices, etc. Everything is stored in the clear without encryption. This means that Paperless should never be run on an untrusted host. Instead, I recommend that if you do want to use it, run it locally on a server in your own home. diff --git a/ansible/README.md b/ansible/README.md new file mode 100644 index 000000000..8771cd08c --- /dev/null +++ b/ansible/README.md @@ -0,0 +1,116 @@ +Ansible Role: paperless-ng +========================== + +Installs and configures paperless-ng EDMS on Debian/Ubuntu systems. + +Requirements +------------ + +No special system requirements. Ansible 2.7 or newer is required. + +Note that this role requires root access, so either run it in a playbook with a global `become: yes`, or invoke the role in your playbook like: + + - hosts: all + roles: + - role: paperless-ng + become: yes + +Role Variables +-------------- + +Most configuration variables from paperless-ng itself are available and accept their respective arguments. +Every `PAPERLESS_*` configuration variable is lowercased and instead prefixed with `paperlessng_*` in `defaults/main.yml`. + +For a full listing including explanations and allowed values, see the current [documentation](https://paperless-ng.readthedocs.io/en/latest/configuration.html). + +Additional variables available in this role are listed below, along with default values: + + paperlessng_version: latest + +The [release](https://github.com/jonaswinkler/paperless-ng/releases) archive version of paperless-ng to install. +`latest` stands for the latest release of paperless-ng. +To install a specific version of paperless-ng, use the tag name of the release, e. g. `ng-1.4.4`, or specify a branch or commit id. + + paperlessng_redis_host: localhost + paperlessng_redis_port: 6379 + +Separate configuration values that combine into `PAPERLESS_REDIS`. + + paperlessng_db_type: sqlite + +Database to use. Default is file-based SQLite. + + paperlessng_db_host: localhost + paperlessng_db_port: 5432 + paperlessng_db_name: paperlessng + paperlessng_db_user: paperlessng + paperlessng_db_pass: paperlessng + paperlessng_db_sslmode: prefer + +Database configuration (only applicable if `paperlessng_db_type == 'postgresql'`). + + paperlessng_directory: /opt/paperless-ng + +Root directory paperless-ng is installed into. + + paperlessng_virtualenv: "{{ paperlessng_directory }}/.venv" + +Directory used for the virtual environment for paperless-ng. + + paperlessng_ocr_languages: + - eng + +List of OCR languages to install and configure (`apt search tesseract-ocr-*`). + + paperlessng_use_jbig2enc: True + +Whether to install and use [jbig2enc](https://github.com/agl/jbig2enc) for OCRmyPDF. + + paperlessng_big2enc_lossy: False + +Whether to use jbig2enc's lossy compression mode. + + paperlessng_superuser_name: paperlessng + paperlessng_superuser_email: paperlessng@example.com + paperlessng_superuser_password: paperlessng + +Credentials of the initial superuser in paperless-ng. + + paperlessng_system_user: paperlessng + paperlessng_system_group: paperlessng + +System user and group to run the paperless-ng services as (will be created if required). + + paperlessng_listen_address: 127.0.0.1 + paperlessng_listen_port: 8000 + +Address and port for the paperless-ng service to listen on. + +Dependencies +------------ + +No ansible dependencies. + +Example Playbook +---------------- +`playbook.yml`: + + - hosts: all + become: yes + vars_files: + - vars/paperless-ng.yml + roles: + - paperless-ng + +`vars/paperless-ng.yml`: + + paperlessng_media_root: /mnt/media/smbshare + + paperlessng_db_type: postgresql + paperlessng_db_pass: PLEASEPROVIDEASTRONGPASSWORDHERE + + paperlessng_secret_key: AGAINPLEASECHANGETHISNOW + + paperlessng_ocr_languages: + - eng + - deu diff --git a/ansible/defaults/main.yml b/ansible/defaults/main.yml new file mode 100644 index 000000000..0a5824608 --- /dev/null +++ b/ansible/defaults/main.yml @@ -0,0 +1,83 @@ +--- +paperlessng_version: latest # 'latest', release number, or github branch/tag/commit/ref + +# Required services +paperlessng_redis_host: localhost +paperlessng_redis_port: 6379 +paperlessng_db_type: sqlite # or postgresql +# Below entries only apply for paperlessng_db_type=='postgresql' +paperlessng_db_host: localhost +paperlessng_db_port: 5432 +paperlessng_db_name: paperlessng +paperlessng_db_user: paperlessng +paperlessng_db_pass: paperlessng +paperlessng_db_sslmode: prefer + +# Paths and folders +paperlessng_directory: /opt/paperless-ng +paperlessng_consumption_dir: "{{ paperlessng_directory }}/consumption" +paperlessng_data_dir: "{{ paperlessng_directory }}/data" +paperlessng_media_root: "{{ paperlessng_directory }}/media" +paperlessng_staticdir: "{{ paperlessng_directory }}/static" +paperlessng_filename_format: +paperlessng_logging_dir: "{{ paperlessng_data_dir }}/log" +paperlessng_virtualenv: "{{ paperlessng_directory }}/.venv" + +# Hosting & Security +paperlessng_secret_key: PLEASECHANGETHISFORTHELOVEOFGOD +paperlessng_allowed_hosts: "*" +paperlessng_cors_allowed_hosts: http://localhost:8000 +paperlessng_force_script_name: +paperlessng_static_url: /static/ +paperlessng_auto_login_username: +paperlessng_cookie_prefix: "" +paperlessng_enable_http_remote_user: False + +# OCR settings +paperlessng_ocr_languages: + - eng +paperlessng_ocr_mode: skip +paperlessng_ocr_clean: clean +paperlessng_ocr_deskew: True +paperlessng_ocr_rotate_pages: True +paperlessng_ocr_rotate_pages_threshold: 12 +paperlessng_ocr_output_type: pdfa +paperlessng_ocr_pages: 0 +paperlessng_ocr_image_dpi: +# see https://ocrmypdf.readthedocs.io/en/latest/api.html#ocrmypdf.ocr +paperlessng_ocr_user_args: + - "optimize": 1 +paperlessng_use_jbig2enc: True +paperlessng_big2enc_lossy: False + +# Tika settings +paperlessng_tika_enabled: False +paperlessng_tika_endpoint: http://localhost:9998 +paperlessng_tika_gotenberg_endpoint: http://localhost:3000 + +# Software tweaks +paperlessng_time_zone: Europe/Berlin +paperlessng_consumer_polling: 0 +paperlessng_consumer_delete_duplicates: False +paperlessng_consumer_recursive: False +paperlessng_consumer_subdirs_as_tags: False +paperlessng_convert_memory_limit: 0 +paperlessng_convert_tmpdir: +paperlessng_optimize_thumbnails: True +paperlessng_post_consume_script: +paperlessng_filename_date_order: +paperlessng_thumbnail_font_name: /usr/share/fonts/liberation/LiberationSerif-Regular.ttf +paperlessng_ignore_dates: "" + +# Superuser settings +paperlessng_superuser_name: paperlessng +paperlessng_superuser_email: paperlessng@example.com +paperlessng_superuser_password: paperlessng + +# System user settings +paperlessng_system_user: paperlessng +paperlessng_system_group: paperlessng + +# Webserver settings +paperlessng_listen_address: 127.0.0.1 +paperlessng_listen_port: 8000 diff --git a/ansible/meta/main.yml b/ansible/meta/main.yml new file mode 100644 index 000000000..78a4c1a41 --- /dev/null +++ b/ansible/meta/main.yml @@ -0,0 +1,17 @@ +dependencies: [] + +galaxy_info: + author: C0nsultant + description: Bare-metal deployment of paperless-ng DMS + license: license (GPLv3) + min_ansible_version: 2.7 + + platforms: + - name: Debian + versions: + - buster + - name: Ubuntu + versions: + - focal + + galaxy_tags: [EDMS, django, python, web] diff --git a/ansible/molecule/default/converge.yml b/ansible/molecule/default/converge.yml new file mode 100644 index 000000000..c7f43c877 --- /dev/null +++ b/ansible/molecule/default/converge.yml @@ -0,0 +1,10 @@ +--- +- name: update previous release to newest release + hosts: all + tasks: + - name: set current github commit as version when available + set_fact: + paperlessng_version: "{{ lookup('env', 'TARGET_GITHUB_SHA') | default('master', True) }}" + - name: update to newest paperless-ng release + include_role: + name: ansible diff --git a/ansible/molecule/default/molecule.yml b/ansible/molecule/default/molecule.yml new file mode 100644 index 000000000..27f37ba63 --- /dev/null +++ b/ansible/molecule/default/molecule.yml @@ -0,0 +1,35 @@ +--- +dependency: + name: galaxy +driver: + name: docker +platforms: + - name: ubuntu_focal + image: jrei/systemd-ubuntu:20.04 + privileged: true + volumes: + - /sys/fs/cgroup:/sys/fs/cgroup:ro + tmpfs: + - /tmp + - /run + - /run/lock + override_command: False + # ubuntu 18.04 bionic works except that + # the default redis configuration expects IPv6 which is not enabled in docker by default + # the default Python environment is configured for ASCII instead of UTF-8 + # ubuntu 16.04 xenial only has Python 3.5 which is EOL and breaks multiple dependencies + - name: debian_buster + image: jrei/systemd-debian:10 + privileged: true + volumes: + - /sys/fs/cgroup:/sys/fs/cgroup:ro + tmpfs: + - /tmp + - /run + - /run/lock + override_command: False + # debian 9 stretch only has Python 3.5 which is EOL and breaks multiple dependencies +provisioner: + name: ansible +verifier: + name: ansible diff --git a/ansible/molecule/default/prepare.yml b/ansible/molecule/default/prepare.yml new file mode 100644 index 000000000..e175eff5b --- /dev/null +++ b/ansible/molecule/default/prepare.yml @@ -0,0 +1,10 @@ +- name: install previous release + hosts: all + tasks: + - name: set previous version as installation target + set_fact: + paperlessng_version: latest + + - name: install previous paperless-ng release + include_role: + name: ansible diff --git a/ansible/molecule/default/verify.yml b/ansible/molecule/default/verify.yml new file mode 100644 index 000000000..7646b223e --- /dev/null +++ b/ansible/molecule/default/verify.yml @@ -0,0 +1,94 @@ +--- +- name: Verify + hosts: all + gather_facts: false + + vars_files: + - ../../defaults/main.yml + + tasks: + - name: check if webserver is up + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}" + status_code: [200, 302] + return_content: yes + register: landingpage + failed_when: "'Sign in' not in landingpage.content" + + - name: generate random name and content + set_fact: + content: "{{ lookup('password', '/dev/null length=65536 chars=ascii_letters') }}" + filename: "{{ lookup('password', '/dev/null length=8 chars=ascii_letters') }}" + + - name: check if document posting works + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/post_document/" + method: POST + body_format: form-multipart + body: + document: + content: "{{ content }}" + filename: "{{ filename }}.txt" + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + Content-Type: text/plain + return_content: yes + register: post_document + failed_when: "'OK' not in post_document.content" + + - name: verify uploaded document has been accepted + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/logs/paperless/" + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + return_content: yes + register: logs + failed_when: "('Consuming ' + filename + '.txt') not in logs.content" + + - name: sleep till consumption finished + pause: + seconds: 10 + + - name: verify uploaded document has been consumed + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/logs/paperless/" + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + return_content: yes + register: logs + failed_when: "filename + ' consumption finished' not in logs.content" + + - name: get documents + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/" + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + return_content: yes + register: documents + + - name: set document index + set_fact: + index: "{{ documents.json['results'][0]['id'] }}" + + - name: verify uploaded document is avaiable + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/{{ index }}/" + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + return_content: yes + register: document + failed_when: "'Not found.' in document.content or content not in document.json['content']" + + - name: check if deleting uploaded document works + uri: + url: "http://{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}/api/documents/bulk_edit/" + method: POST + body_format: json + body: + documents: ["{{ index }}"] + method: delete + parameters: {} + headers: + Authorization: 'Basic {{ (paperlessng_superuser_name + ":" + paperlessng_superuser_password) | b64encode }}' + register: delete_document + failed_when: "'OK' not in delete_document.json['result']" diff --git a/ansible/tasks/install-release.yml b/ansible/tasks/install-release.yml new file mode 100644 index 000000000..c2dfb0b9f --- /dev/null +++ b/ansible/tasks/install-release.yml @@ -0,0 +1,6 @@ +--- +- name: extract paperless-ng + unarchive: + src: "https://github.com/jonaswinkler/paperless-ng/releases/download/ng-{{ paperlessng_version }}/paperless-ng-{{ paperlessng_version }}.tar.xz" + remote_src: yes + dest: "{{ tempdir.path }}" diff --git a/ansible/tasks/install-source.yml b/ansible/tasks/install-source.yml new file mode 100644 index 000000000..4fc8b1fd8 --- /dev/null +++ b/ansible/tasks/install-source.yml @@ -0,0 +1,112 @@ +--- +- name: install dev dependencies + apt: + pkg: + - git + - npm + - gettext + +- name: create output directories + file: + path: "{{ item }}" + state: directory + owner: "{{ paperlessng_system_user }}" + group: "{{ paperlessng_system_group }}" + mode: "750" + with_items: + - "{{ tempdir.path }}/paperless-ng" + - "{{ tempdir.path }}/paperless-ng/scripts" + +- block: + - name: create temporary git directory + tempfile: + state: directory + path: "{{ paperlessng_directory }}" + register: gitdir + + - name: pull paperless-ng + git: + repo: https://github.com/jonaswinkler/paperless-ng.git + dest: "{{ gitdir.path }}" + version: "{{ paperlessng_version }}" + refspec: "+refs/pull/*:refs/pull/*" + + - name: compile frontend + command: + cmd: "{{ item }}" + args: + chdir: "{{ gitdir.path }}/src-ui" + failed_when: false + with_items: + - npm install -g @angular/cli + - npm install + - ./node_modules/.bin/ng build --prod + + - name: copy application into place + copy: + src: "{{ gitdir.path }}/{{ item.src }}" + remote_src: yes + dest: "{{ tempdir.path }}/paperless-ng/{{ item.dest | default('') }}" + with_items: + - src: CONTRIBUTING.md + - src: LICENSE + - src: Pipfile + - src: Pipfile.lock + - src: README.md + - src: requirements.txt + - src: gunicorn.conf.py + - src: paperless.conf.example + dest: "paperless.conf" + + - name: glob all scripts + find: + paths: ["{{ gitdir.path }}/scripts/"] + patterns: + - "*.service" + - "*.sh" + register: glob + + - name: copy scripts + copy: + src: "{{ item.path }}" + remote_src: yes + dest: "{{ tempdir.path }}/paperless-ng/scripts/" + with_items: + - "{{ glob.files }}" + + - name: copy sources + command: + cmd: "cp -r src/ {{ tempdir.path }}/paperless-ng/src" + args: + chdir: "{{ gitdir.path }}" + + - name: create paperlessng venv + command: + cmd: "python3 -m virtualenv {{ gitdir.path }}/.venv/ -p /usr/bin/python3" + + - name: install paperlessng requirements + command: + cmd: "{{ gitdir.path }}/.venv/bin/python3 -m pip install -r {{ gitdir.path }}/requirements.txt" + + - name: compile messages + command: "{{ gitdir.path }}/.venv/bin/python3 manage.py compilemessages" + args: + chdir: "{{ tempdir.path }}/paperless-ng/src/" + + - name: collect static files + command: "{{ gitdir.path }}/.venv/bin/python3 manage.py collectstatic --no-input" + args: + chdir: "{{ tempdir.path }}/paperless-ng/src/" + + - name: remove pycache directories + shell: find . -name __pycache__ | xargs rm -r + args: + chdir: "{{ tempdir.path }}" + + - name: remove temporary git directory + file: + path: "{{ gitdir.path }}" + state: absent + + become: yes + become_user: "{{ paperlessng_system_user }}" diff --git a/ansible/tasks/main.yml b/ansible/tasks/main.yml new file mode 100644 index 000000000..9fc476597 --- /dev/null +++ b/ansible/tasks/main.yml @@ -0,0 +1,553 @@ +--- +- name: verify operating system + fail: + msg: Sorry, only Debian and Ubuntu supported at the moment. + when: not(ansible_distribution == 'Debian' or ansible_distribution == 'Ubuntu') + +- name: install base dependencies + apt: + update_cache: yes + pkg: + # paperless-ng + - python3-pip + - python3-dev + - fonts-liberation + - imagemagick + - optipng + - gnupg + - libpq-dev + - libmagic-dev + - mime-support + # OCRmyPDF + - unpaper + - ghostscript + - icc-profiles-free + - qpdf + - liblept5 + - libxml2 + - pngquant + - zlib1g + - tesseract-ocr + # dev + - sudo + - build-essential + - python3-setuptools + - python3-wheel + +# upstream virtualenv in Ubuntu 20.04 is broken +# https://github.com/pypa/virtualenv/issues/1873 +- name: install python virtualenv + pip: + name: virtualenv + extra_args: --upgrade + +- name: install ocr languages + apt: + pkg: "{{ paperlessng_ocr_languages | map('regex_replace', '^(.*)$', 'tesseract-ocr-\\1') | map('replace', '_', '-') | list }}" + +- name: set up notesalexp repository key (for jbig2enc) + apt_key: + url: https://notesalexp.org/debian/alexp_key.asc + state: present + when: paperlessng_use_jbig2enc + +- name: set up notesalexp repository (for jbig2enc) + apt_repository: + repo: "deb https://notesalexp.org/debian/{{ ansible_distribution_release }}/ {{ ansible_distribution_release }} main" + state: present + when: paperlessng_use_jbig2enc + +- name: set up notesalexp repository pinning + copy: + content: | + Package: * + Pin: release o=notesalexp.org + Pin-Priority: 1 + + Package: jbig2enc + Pin: release o=notesalexp.org + Pin-Priority: 500 + dest: /etc/apt/preferences.d/notesalexp + when: paperlessng_use_jbig2enc + +- name: install jbig2enc + apt: + pkg: jbig2enc + update_cache: yes + when: paperlessng_use_jbig2enc + +- name: install redis + apt: + pkg: redis-server + when: paperlessng_redis_host == 'localhost' or paperlessng_redis_host == '127.0.0.1' + +- name: enable redis + systemd: + name: redis-server + enabled: yes + masked: no + state: started + when: paperlessng_redis_host == 'localhost' or paperlessng_redis_host == '127.0.0.1' + +- name: create paperless system group + group: + name: "{{ paperlessng_system_group }}" + +- name: create paperless system user + user: + name: "{{ paperlessng_system_user }}" + groups: + - "{{ paperlessng_system_group }}" + shell: /usr/sbin/nologin + # GNUPG_HOME required due to paperless db.py + create_home: yes + +- block: + - name: get latest release version + uri: + url: https://api.github.com/repos/jonaswinkler/paperless-ng/releases/latest + method: GET + register: latest_release + - name: parse latest release version + set_fact: + paperlessng_version: "{{ latest_release.json['tag_name'] }}" + when: paperlessng_version == "latest" + +- name: check if version is ref + fail: + msg: "Specifying `paperlessng_version` as git ref may not work as expected!" + ignore_errors: True # Output failure (as warning), but don't consider play failed + when: paperlessng_version.startswith('refs/') + +- block: + - name: sanitize version string + set_fact: + paperlessng_version: "{{ paperlessng_version | regex_replace('^ng-(\\d+\\.\\d+\\.\\d+)$', '\\1') }}" + - name: get tag data + uri: + url: https://api.github.com/repos/jonaswinkler/paperless-ng/tags + method: GET + register: tags + - name: get commit for target tag + set_fact: + paperlessng_commit: "{{ tags.json | json_query('[?name==`ng-' + paperlessng_version +'`] | [0].commit.sha') }}" + when: paperlessng_version | regex_search("^(ng-)?(\d+\.\d+\.\d+)$") + +- block: + - name: check if version is branch + uri: + url: "https://api.github.com/repos/jonaswinkler/paperless-ng/branches/{{ paperlessng_version }}" + method: GET + status_code: [200, 404] + register: branch + - name: get commit for target branch + set_fact: + paperlessng_commit: "{{ branch.json | json_query('commit.sha') }}" + when: branch.status == 200 + - block: + - name: check if version is commit-or-ref + uri: + url: "https://api.github.com/repos/jonaswinkler/paperless-ng/commits/{{ paperlessng_version }}" + method: GET + status_code: [200, 404, 422] + register: commit + - name: get commit for target commit-or-ref + set_fact: + paperlessng_commit: "{{ commit.json | json_query('sha') }}" + when: commit.status == 200 + - name: fail + fail: + msg: "Can not determine commit from `paperlessng_version=={{ paperlessng_version }}`!" + when: commit.status != 200 + when: branch.status == 404 + when: not(paperlessng_version | regex_search("^(ng-)?(\d+\.\d+\.\d+)$")) + +- name: check for paperless-ng installation + command: + cmd: "cat {{ paperlessng_directory }}/.installed_version" + changed_when: '"No such file or directory" in paperlessng_current_commit.stderr or paperlessng_current_commit.stdout != paperlessng_commit | string' + failed_when: false + ignore_errors: yes + register: paperlessng_current_commit + +- name: register current state + set_fact: + fresh_installation: '{{ "No such file or directory" in paperlessng_current_commit.stderr }}' + update_installation: '{{ "No such file or directory" not in paperlessng_current_commit.stderr and paperlessng_current_commit.stdout != paperlessng_commit | string }}' + reconfigure_only: "{{ paperlessng_current_commit.stdout == paperlessng_commit | string }}" + +- block: + - name: backup current paperless-ng installation + copy: + src: "{{ paperlessng_directory }}" + remote_src: yes + dest: "{{ paperlessng_directory }}-{{ ansible_date_time.iso8601 }}/" + - name: remove current paperless sources + file: + path: "{{ paperlessng_directory }}/{{ item }}" + state: absent + with_items: + - docker + - docs + - scripts + - src + - static + when: update_installation + +- block: + - name: create paperless-ng directory and set permissions + file: + path: "{{ paperlessng_directory }}" + state: directory + owner: "{{ paperlessng_system_user }}" + group: "{{ paperlessng_system_group }}" + mode: "750" + - name: create temporary directory + become: yes + become_user: "{{ paperlessng_system_user }}" + tempfile: + state: directory + path: "{{ paperlessng_directory }}" + register: tempdir + - name: check if version is available as release archive + uri: + url: "https://github.com/jonaswinkler/paperless-ng/releases/download/ng-{{ paperlessng_version }}/paperless-ng-{{ paperlessng_version }}.tar.xz" + method: GET + status_code: [200, 404] + register: release_archive + - name: install paperless-ng from source + include_tasks: install-source.yml + when: release_archive.status == 404 + - name: install paperless-ng from release archive + include_tasks: install-release.yml + when: release_archive.status == 200 + - name: change owner and permissions of paperless-ng + command: + cmd: "{{ item }}" + warn: false + with_items: + - "chown -R {{ paperlessng_system_user }}:{{ paperlessng_system_group }} {{ tempdir.path }}" + - "find {{ tempdir.path }} -type d -exec chmod 0750 {} ;" + - "find {{ tempdir.path }} -type f -exec chmod 0640 {} ;" + - name: move paperless-ng + command: + cmd: "cp -a {{ tempdir.path }}/paperless-ng/. {{ paperlessng_directory }}" + - name: store commit hash of installed version + copy: + content: "{{ paperlessng_commit }}" + dest: "{{ paperlessng_directory }}/.installed_version" + owner: "{{ paperlessng_system_user }}" + group: "{{ paperlessng_system_group }}" + mode: "0440" + - name: remove temporary directory + file: + path: "{{ tempdir.path }}" + state: absent + when: not reconfigure_only + +- name: create paperless-ng directories and set permissions + file: + path: "{{ item }}" + state: directory + owner: "{{ paperlessng_system_user }}" + group: "{{ paperlessng_system_group }}" + mode: "750" + with_items: + - "{{ paperlessng_consumption_dir }}" + - "{{ paperlessng_data_dir }}" + - "{{ paperlessng_media_root }}" + - "{{ paperlessng_staticdir }}" + +- name: rename initial config + command: + cmd: "mv -f {{ paperlessng_directory }}/paperless.conf {{ paperlessng_directory }}/paperless.conf.template" + removes: "{{ paperlessng_directory }}/paperless.conf" + +- name: configure paperless-ng + lineinfile: + path: "{{ paperlessng_directory }}/paperless.conf.template" + regexp: "^#?{{ item.regexp }}=" + line: "{{ item.line }}" + with_items: + # Required services + - regexp: PAPERLESS_REDIS + line: "PAPERLESS_REDIS=redis://{{ paperlessng_redis_host }}:{{ paperlessng_redis_port }}" + # Paths and folders + - regexp: PAPERLESS_CONSUMPTION_DIR + line: "PAPERLESS_CONSUMPTION_DIR={{ paperlessng_consumption_dir }}" + - regexp: PAPERLESS_DATA_DIR + line: "PAPERLESS_DATA_DIR={{ paperlessng_data_dir }}" + - regexp: PAPERLESS_MEDIA_ROOT + line: "PAPERLESS_MEDIA_ROOT={{ paperlessng_media_root }}" + - regexp: PAPERLESS_STATICDIR + line: "PAPERLESS_STATICDIR={{ paperlessng_staticdir }}" + - regexp: PAPERLESS_FILENAME_FORMAT + line: "PAPERLESS_FILENAME_FORMAT={{ paperlessng_filename_format }}" + - regexp: PAPERLESS_LOGGING_DIR + line: "PAPERLESS_LOGGING_DIR={{ paperlessng_logging_dir }}" + # Hosting & Security + - regexp: PAPERLESS_SECRET_KEY + line: "PAPERLESS_SECRET_KEY={{ paperlessng_secret_key }}" + - regexp: PAPERLESS_ALLOWED_HOSTS + line: "PAPERLESS_ALLOWED_HOSTS={{ paperlessng_allowed_hosts }}" + - regexp: PAPERLESS_CORS_ALLOWED_HOSTS + line: "PAPERLESS_CORS_ALLOWED_HOSTS={{ paperlessng_cors_allowed_hosts }}" + - regexp: PAPERLESS_FORCE_SCRIPT_NAME + line: "PAPERLESS_FORCE_SCRIPT_NAME={{ paperlessng_force_script_name }}" + - regexp: PAPERLESS_STATIC_URL + line: "PAPERLESS_STATIC_URL={{ paperlessng_static_url }}" + - regexp: PAPERLESS_AUTO_LOGIN_USERNAME + line: "PAPERLESS_AUTO_LOGIN_USERNAME={{ paperlessng_auto_login_username }}" + - regexp: PAPERLESS_COOKIE_PREFIX + line: "PAPERLESS_COOKIE_PREFIX={{ paperlessng_cookie_prefix }}" + - regexp: PAPERLESS_ENABLE_HTTP_REMOTE_USER + line: "PAPERLESS_ENABLE_HTTP_REMOTE_USER={{ paperlessng_enable_http_remote_user }}" + # OCR settings + - regexp: PAPERLESS_OCR_LANGUAGE + line: "PAPERLESS_OCR_LANGUAGE={{ paperlessng_ocr_languages | join('+') | replace('-','_') }}" + - regexp: PAPERLESS_OCR_MODE + line: "PAPERLESS_OCR_MODE={{ paperlessng_ocr_mode }}" + - regexp: PAPERLESS_OCR_CLEAN + line: "PAPERLESS_OCR_CLEAN={{ paperlessng_ocr_clean }}" + - regexp: PAPERLESS_OCR_DESKEW + line: "PAPERLESS_OCR_DESKEW={{ paperlessng_ocr_deskew }}" + - regexp: PAPERLESS_OCR_ROTATE_PAGES + line: "PAPERLESS_OCR_ROTATE_PAGES={{ paperlessng_ocr_rotate_pages }}" + - regexp: PAPERLESS_OCR_ROTATE_PAGES_THRESHOLD + line: "PAPERLESS_OCR_ROTATE_PAGES_THRESHOLD={{ paperlessng_ocr_rotate_pages_threshold }}" + - regexp: PAPERLESS_OCR_OUTPUT_TYPE + line: "PAPERLESS_OCR_OUTPUT_TYPE={{ paperlessng_ocr_output_type }}" + - regexp: PAPERLESS_OCR_PAGES + line: "PAPERLESS_OCR_PAGES={{ paperlessng_ocr_pages }}" + - regexp: PAPERLESS_OCR_IMAGE_DPI + line: "PAPERLESS_OCR_IMAGE_DPI={{ paperlessng_ocr_image_dpi }}" + - regexp: PAPERLESS_OCR_USER_ARGS + line: "PAPERLESS_OCR_USER_ARGS={{ paperlessng_ocr_user_args | combine({'jbig2_lossy': true} if paperlessng_big2enc_lossy else {}) | to_json }}" + # Tika settings + - regexp: PAPERLESS_TIKA_ENABLED + line: "PAPERLESS_TIKA_ENABLED={{ paperlessng_tika_enabled }}" + - regexp: PAPERLESS_TIKA_ENDPOINT + line: "PAPERLESS_TIKA_ENDPOINT={{ paperlessng_tika_endpoint }}" + - regexp: PAPERLESS_TIKA_GOTENBERG_ENDPOINT + line: "PAPERLESS_TIKA_GOTENBERG_ENDPOINT={{ paperlessng_tika_gotenberg_endpoint }}" + # Software tweaks + - regexp: PAPERLESS_TIME_ZONE + line: "PAPERLESS_TIME_ZONE={{ paperlessng_time_zone }}" + - regexp: PAPERLESS_CONSUMER_POLLING + line: "PAPERLESS_CONSUMER_POLLING={{ paperlessng_consumer_polling }}" + - regexp: PAPERLESS_CONSUMER_DELETE_DUPLICATES + line: "PAPERLESS_CONSUMER_DELETE_DUPLICATES={{ paperlessng_consumer_delete_duplicates }}" + - regexp: PAPERLESS_CONSUMER_RECURSIVE + line: "PAPERLESS_CONSUMER_RECURSIVE={{ paperlessng_consumer_recursive }}" + - regexp: PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS + line: "PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS={{ paperlessng_consumer_subdirs_as_tags }}" + - regexp: PAPERLESS_CONVERT_MEMORY_LIMIT + line: "PAPERLESS_CONVERT_MEMORY_LIMIT={{ paperlessng_convert_memory_limit }}" + - regexp: PAPERLESS_CONVERT_TMPDIR + line: "PAPERLESS_CONVERT_TMPDIR={{ paperlessng_convert_tmpdir }}" + - regexp: PAPERLESS_OPTIMIZE_THUMBNAILS + line: "PAPERLESS_OPTIMIZE_THUMBNAILS={{ paperlessng_optimize_thumbnails }}" + - regexp: PAPERLESS_POST_CONSUME_SCRIPT + line: "PAPERLESS_POST_CONSUME_SCRIPT={{ paperlessng_post_consume_script }}" + - regexp: PAPERLESS_FILENAME_DATE_ORDER + line: "PAPERLESS_FILENAME_DATE_ORDER={{ paperlessng_filename_date_order }}" + - regexp: PAPERLESS_THUMBNAIL_FONT_NAME + line: "PAPERLESS_THUMBNAIL_FONT_NAME={{ paperlessng_thumbnail_font_name }}" + - regexp: PAPERLESS_IGNORE_DATES + line: "PAPERLESS_IGNORE_DATES={{ paperlessng_ignore_dates }}" + no_log: yes + +- name: configure paperless-ng database [sqlite] + lineinfile: + path: "{{ paperlessng_directory }}/paperless.conf.template" + regexp: "^#?PAPERLESS_DBHOST=(.*)$" + line: '#PAPERLESS_DBHOST=\1' + backrefs: yes + when: paperlessng_db_type == 'sqlite' + +- name: configure paperless-ng database [postgresql] + lineinfile: + path: "{{ paperlessng_directory }}/paperless.conf.template" + regexp: "^#?{{ item.regexp }}=" + line: "{{ item.line }}" + with_items: + - regexp: PAPERLESS_DBHOST + line: "PAPERLESS_DBHOST={{ paperlessng_db_host }}" + - regexp: PAPERLESS_DBPORT + line: "PAPERLESS_DBPORT={{ paperlessng_db_port }}" + - regexp: PAPERLESS_DBNAME + line: "PAPERLESS_DBNAME={{ paperlessng_db_name }}" + - regexp: PAPERLESS_DBUSER + line: "PAPERLESS_DBUSER={{ paperlessng_db_user }}" + - regexp: PAPERLESS_DBPASS + line: "PAPERLESS_DBPASS={{ paperlessng_db_pass }}" + - regexp: PAPERLESS_DBSSLMODE + line: "PAPERLESS_DBSSLMODE={{ paperlessng_db_sslmode }}" + when: paperlessng_db_type == 'postgresql' + no_log: yes + +- name: deploy paperless-ng configuration + copy: + src: "{{ paperlessng_directory }}/paperless.conf.template" + remote_src: yes + dest: /etc/paperless.conf + owner: root + group: root + mode: "0644" + register: configuration + +- name: create paperlessng venv + become: yes + become_user: "{{ paperlessng_system_user }}" + command: + cmd: "python3 -m virtualenv {{ paperlessng_virtualenv }} -p /usr/bin/python3" + creates: "{{ paperlessng_virtualenv }}" + register: venv + +- block: + - name: install paperlessng requirements + become: yes + become_user: "{{ paperlessng_system_user }}" + pip: + requirements: "{{ paperlessng_directory }}/requirements.txt" + executable: "{{ paperlessng_virtualenv }}/bin/pip3" + extra_args: --upgrade + - name: migrate database schema + become: yes + become_user: "{{ paperlessng_system_user }}" + command: "{{ paperlessng_virtualenv }}/bin/python3 {{ paperlessng_directory }}/src/manage.py migrate" + register: database_schema + changed_when: '"No migrations to apply." not in database_schema.stdout' + when: not reconfigure_only + +- name: configure paperless superuser + become: yes + become_user: "{{ paperlessng_system_user }}" + # "manage.py createsuperuser" only works on interactive TTYs + vars: + creation_script: | + from django.contrib.auth.models import User + from django.contrib.auth.hashers import get_hasher + + if User.objects.filter(username='{{ paperlessng_superuser_name }}').exists(): + user = User.objects.get(username='{{ paperlessng_superuser_name }}') + old = user.__dict__.copy() + + user.is_superuser = True + user.email = '{{ paperlessng_superuser_email }}' + user.set_password('{{ paperlessng_superuser_password }}') + user.save() + new = user.__dict__ + + algorithm, iterations, old_salt, old_hash = old['password'].split('$') + new_password_old_salt = get_hasher(algorithm).encode(password='{{ paperlessng_superuser_password }}', salt=old_salt, iterations=int(iterations)) + _, _, _, new_hash = new_password_old_salt.split('$') + if not (old_hash == new_hash and old['is_superuser'] == new['is_superuser'] and old['email'] == new['email']): + print('changed') + else: + User.objects.create_superuser('{{ paperlessng_superuser_name }}', '{{ paperlessng_superuser_email }}', '{{ paperlessng_superuser_password }}') + print('changed') + command: '{{ paperlessng_virtualenv }}/bin/python3 {{ paperlessng_directory }}/src/manage.py shell -c "{{ creation_script }}"' + register: superuser + changed_when: superuser.stdout == 'changed' + no_log: yes + +- name: set ownership and permissions on paperlessng venv + file: + path: "{{ paperlessng_virtualenv }}" + state: directory + recurse: yes + owner: "{{ paperlessng_system_user }}" + group: "{{ paperlessng_system_group }}" + mode: g-w,o-rwx + when: venv.changed or not reconfigure_only + +- name: configure ghostscript for PDF + lineinfile: + path: /etc/ImageMagick-6/policy.xml + regexp: '(\s+)' + line: '\1' + backrefs: yes + +- name: configure gunicorn web server + lineinfile: + path: "{{ paperlessng_directory }}/gunicorn.conf.py" + regexp: '^bind = ' + line: "bind = '{{ paperlessng_listen_address }}:{{ paperlessng_listen_port }}'" + +- name: configure systemd services + ini_file: + path: "{{ paperlessng_directory }}/scripts/{{ item[0] }}" + section: "Service" + option: "{{ item[1].option }}" + value: "{{ item[1].value }}" + with_nested: + - [ + paperless-consumer.service, + paperless-scheduler.service, + paperless-webserver.service, + ] + - [ + # https://www.freedesktop.org/software/systemd/man/systemd.exec.html + { option: "User", value: "{{ paperlessng_system_user }}" }, + { option: "Group", value: "{{ paperlessng_system_group }}" }, + { option: "WorkingDirectory", value: "{{ paperlessng_directory }}/src" }, + { option: "ProtectSystem", value: "full" }, + { option: "NoNewPrivileges", value: "true" }, + { option: "PrivateUsers", value: "true" }, + { option: "PrivateDevices", value: "true" }, + ] + +- name: configure paperless-consumer service + ini_file: + path: "{{ paperlessng_directory }}/scripts/paperless-consumer.service" + section: "Service" + option: "ExecStart" + value: "{{ paperlessng_virtualenv }}/bin/python3 manage.py document_consumer" + +- name: configure paperless-scheduler service + ini_file: + path: "{{ paperlessng_directory }}/scripts/paperless-scheduler.service" + section: "Service" + option: "ExecStart" + value: "{{ paperlessng_virtualenv }}/bin/python3 manage.py qcluster" + +- name: configure paperless-webserver service + ini_file: + path: "{{ paperlessng_directory }}/scripts/paperless-webserver.service" + section: "Service" + option: "ExecStart" + value: "{{ paperlessng_virtualenv }}/bin/gunicorn -c {{ paperlessng_directory }}/gunicorn.conf.py paperless.asgi:application" + +- name: copy systemd services + copy: + src: "{{ paperlessng_directory }}/scripts/{{ item }}" + remote_src: yes + dest: "/etc/systemd/system/{{ item }}" + with_items: + - paperless-consumer.service + - paperless-scheduler.service + - paperless-webserver.service + register: paperless_services + +- name: reload systemd daemon + systemd: + name: "{{ item }}" + state: restarted + daemon_reload: yes + with_items: + - paperless-consumer + - paperless-scheduler + - paperless-webserver + when: paperless_services.changed or configuration.changed + +- name: enable paperlessng services + systemd: + name: "{{ item }}" + enabled: yes + masked: no + state: started + with_items: + - paperless-consumer + - paperless-scheduler + - paperless-webserver diff --git a/compile-frontend.sh b/compile-frontend.sh new file mode 100755 index 000000000..98b88d033 --- /dev/null +++ b/compile-frontend.sh @@ -0,0 +1,7 @@ +#!/bin/bash + +set -e + +cd src-ui +npm install +./node_modules/.bin/ng build --prod diff --git a/crowdin.yml b/crowdin.yml new file mode 100644 index 000000000..ddc915f6d --- /dev/null +++ b/crowdin.yml @@ -0,0 +1,6 @@ +commit_message: '[ci skip]' +files: + - source: /src/locale/en_US/LC_MESSAGES/django.po + translation: /src/locale/%locale_with_underscore%/LC_MESSAGES/django.po + - source: /src-ui/messages.xlf + translation: /src-ui/src/locale/messages.%locale_with_underscore%.xlf diff --git a/docker/compose/.env b/docker/compose/.env new file mode 100644 index 000000000..0bd5be6ff --- /dev/null +++ b/docker/compose/.env @@ -0,0 +1 @@ +COMPOSE_PROJECT_NAME=paperless diff --git a/docker/docker-compose.env b/docker/compose/docker-compose.env similarity index 95% rename from docker/docker-compose.env rename to docker/compose/docker-compose.env index 4271bce6e..e9e8ecb06 100644 --- a/docker/docker-compose.env +++ b/docker/compose/docker-compose.env @@ -7,7 +7,7 @@ # Additional languages to install for text recognition, separated by a # whitespace. Note that this is # different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines the -# default language used when guessing the language from the OCR output. +# language used for OCR. # The container installs English, German, Italian, Spanish and French by # default. # See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster diff --git a/docker/compose/docker-compose.portainer.yml b/docker/compose/docker-compose.portainer.yml new file mode 100644 index 000000000..12988c636 --- /dev/null +++ b/docker/compose/docker-compose.portainer.yml @@ -0,0 +1,94 @@ +# docker-compose file for running paperless from the Docker Hub. +# This file contains everything paperless needs to run. +# Paperless supports amd64, arm and arm64 hardware. +# +# All compose files of paperless configure paperless in the following way: +# +# - Paperless is (re)started on system boot, if it was running before shutdown. +# - Docker volumes for storing data are managed by Docker. +# - Folders for importing and exporting files are created in the same directory +# as this file and mounted to the correct folders inside the container. +# - Paperless listens on port 8010. +# +# In addition to that, this docker-compose file adds the following optional +# configurations: +# +# - Instead of SQLite (default), PostgreSQL is used as the database server. +# +# To install and update paperless with this file, do the following: +# +# - Open portainer Stacks list and click 'Add stack' +# - Paste the contents of this file and assign a name, e.g. 'Paperless' +# - Click 'Deploy the stack' and wait for it to be deployed +# - Open the list of containers, select paperless_webserver_1 +# - Click 'Console' and then 'Connect' to open the command line inside the container +# - Run 'python3 manage.py createsuperuser' to create a user +# - Exit the console +# +# For more extensive installation and update instructions, refer to the +# documentation. + +version: "3.4" +services: + broker: + image: redis:6.0 + restart: unless-stopped + + db: + image: postgres:13 + restart: unless-stopped + volumes: + - pgdata:/var/lib/postgresql/data + environment: + POSTGRES_DB: paperless + POSTGRES_USER: paperless + POSTGRES_PASSWORD: paperless + + webserver: + image: jonaswinkler/paperless-ng:latest + restart: unless-stopped + depends_on: + - db + - broker + ports: + - 8010:8000 + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8000"] + interval: 30s + timeout: 10s + retries: 5 + volumes: + - data:/usr/src/paperless/data + - media:/usr/src/paperless/media + - ./export:/usr/src/paperless/export + - ./consume:/usr/src/paperless/consume + environment: + PAPERLESS_REDIS: redis://broker:6379 + PAPERLESS_DBHOST: db +# The UID and GID of the user used to run paperless in the container. Set this +# to your UID and GID on the host so that you have write access to the +# consumption directory. + USERMAP_UID: 1000 + USERMAP_GID: 100 +# Additional languages to install for text recognition, separated by a +# whitespace. Note that this is +# different from PAPERLESS_OCR_LANGUAGE (default=eng), which defines the +# language used for OCR. +# The container installs English, German, Italian, Spanish and French by +# default. +# See https://packages.debian.org/search?keywords=tesseract-ocr-&searchon=names&suite=buster +# for available languages. + #PAPERLESS_OCR_LANGUAGES: tur ces +# Adjust this key if you plan to make paperless available publicly. It should +# be a very long sequence of random characters. You don't need to remember it. + #PAPERLESS_SECRET_KEY: change-me +# Use this variable to set a timezone for the Paperless Docker containers. If not specified, defaults to UTC. + #PAPERLESS_TIME_ZONE: America/Los_Angeles +# The default language to use for OCR. Set this to the language most of your +# documents are written in. + #PAPERLESS_OCR_LANGUAGE: eng + +volumes: + data: + media: + pgdata: diff --git a/docker/compose/docker-compose.postgres-tika.yml b/docker/compose/docker-compose.postgres-tika.yml new file mode 100644 index 000000000..93fff4afc --- /dev/null +++ b/docker/compose/docker-compose.postgres-tika.yml @@ -0,0 +1,90 @@ +# docker-compose file for running paperless from the Docker Hub. +# This file contains everything paperless needs to run. +# Paperless supports amd64, arm and arm64 hardware. +# +# All compose files of paperless configure paperless in the following way: +# +# - Paperless is (re)started on system boot, if it was running before shutdown. +# - Docker volumes for storing data are managed by Docker. +# - Folders for importing and exporting files are created in the same directory +# as this file and mounted to the correct folders inside the container. +# - Paperless listens on port 8000. +# +# In addition to that, this docker-compose file adds the following optional +# configurations: +# +# - Instead of SQLite (default), PostgreSQL is used as the database server. +# - Apache Tika and Gotenberg servers are started with paperless and paperless +# is configured to use these services. These provide support for consuming +# Office documents (Word, Excel, Power Point and their LibreOffice counter- +# parts. +# +# To install and update paperless with this file, do the following: +# +# - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' +# and '.env' into a folder. +# - Run 'docker-compose pull'. +# - Run 'docker-compose run --rm webserver createsuperuser' to create a user. +# - Run 'docker-compose up -d'. +# +# For more extensive installation and update instructions, refer to the +# documentation. + +version: "3.4" +services: + broker: + image: redis:6.0 + restart: unless-stopped + + db: + image: postgres:13 + restart: unless-stopped + volumes: + - pgdata:/var/lib/postgresql/data + environment: + POSTGRES_DB: paperless + POSTGRES_USER: paperless + POSTGRES_PASSWORD: paperless + + webserver: + image: jonaswinkler/paperless-ng:latest + restart: unless-stopped + depends_on: + - db + - broker + - gotenberg + - tika + ports: + - 8000:8000 + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8000"] + interval: 30s + timeout: 10s + retries: 5 + volumes: + - data:/usr/src/paperless/data + - media:/usr/src/paperless/media + - ./export:/usr/src/paperless/export + - ./consume:/usr/src/paperless/consume + env_file: docker-compose.env + environment: + PAPERLESS_REDIS: redis://broker:6379 + PAPERLESS_DBHOST: db + PAPERLESS_TIKA_ENABLED: 1 + PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 + PAPERLESS_TIKA_ENDPOINT: http://tika:9998 + + gotenberg: + image: thecodingmachine/gotenberg + restart: unless-stopped + environment: + DISABLE_GOOGLE_CHROME: 1 + + tika: + image: apache/tika + restart: unless-stopped + +volumes: + data: + media: + pgdata: diff --git a/docker/compose/docker-compose.postgres.yml b/docker/compose/docker-compose.postgres.yml new file mode 100644 index 000000000..da33b4c25 --- /dev/null +++ b/docker/compose/docker-compose.postgres.yml @@ -0,0 +1,72 @@ +# docker-compose file for running paperless from the Docker Hub. +# This file contains everything paperless needs to run. +# Paperless supports amd64, arm and arm64 hardware. +# +# All compose files of paperless configure paperless in the following way: +# +# - Paperless is (re)started on system boot, if it was running before shutdown. +# - Docker volumes for storing data are managed by Docker. +# - Folders for importing and exporting files are created in the same directory +# as this file and mounted to the correct folders inside the container. +# - Paperless listens on port 8000. +# +# In addition to that, this docker-compose file adds the following optional +# configurations: +# +# - Instead of SQLite (default), PostgreSQL is used as the database server. +# +# To install and update paperless with this file, do the following: +# +# - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' +# and '.env' into a folder. +# - Run 'docker-compose pull'. +# - Run 'docker-compose run --rm webserver createsuperuser' to create a user. +# - Run 'docker-compose up -d'. +# +# For more extensive installation and update instructions, refer to the +# documentation. + +version: "3.4" +services: + broker: + image: redis:6.0 + restart: unless-stopped + + db: + image: postgres:13 + restart: unless-stopped + volumes: + - pgdata:/var/lib/postgresql/data + environment: + POSTGRES_DB: paperless + POSTGRES_USER: paperless + POSTGRES_PASSWORD: paperless + + webserver: + image: jonaswinkler/paperless-ng:latest + restart: unless-stopped + depends_on: + - db + - broker + ports: + - 8000:8000 + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8000"] + interval: 30s + timeout: 10s + retries: 5 + volumes: + - data:/usr/src/paperless/data + - media:/usr/src/paperless/media + - ./export:/usr/src/paperless/export + - ./consume:/usr/src/paperless/consume + env_file: docker-compose.env + environment: + PAPERLESS_REDIS: redis://broker:6379 + PAPERLESS_DBHOST: db + + +volumes: + data: + media: + pgdata: diff --git a/docker/compose/docker-compose.sqlite-tika.yml b/docker/compose/docker-compose.sqlite-tika.yml new file mode 100644 index 000000000..5dfff0830 --- /dev/null +++ b/docker/compose/docker-compose.sqlite-tika.yml @@ -0,0 +1,78 @@ +# docker-compose file for running paperless from the Docker Hub. +# This file contains everything paperless needs to run. +# Paperless supports amd64, arm and arm64 hardware. +# +# All compose files of paperless configure paperless in the following way: +# +# - Paperless is (re)started on system boot, if it was running before shutdown. +# - Docker volumes for storing data are managed by Docker. +# - Folders for importing and exporting files are created in the same directory +# as this file and mounted to the correct folders inside the container. +# - Paperless listens on port 8000. +# +# SQLite is used as the database. The SQLite file is stored in the data volume. +# +# In addition to that, this docker-compose file adds the following optional +# configurations: +# +# - Apache Tika and Gotenberg servers are started with paperless and paperless +# is configured to use these services. These provide support for consuming +# Office documents (Word, Excel, Power Point and their LibreOffice counter- +# parts. +# +# To install and update paperless with this file, do the following: +# +# - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' +# and '.env' into a folder. +# - Run 'docker-compose pull'. +# - Run 'docker-compose run --rm webserver createsuperuser' to create a user. +# - Run 'docker-compose up -d'. +# +# For more extensive installation and update instructions, refer to the +# documentation. + +version: "3.4" +services: + broker: + image: redis:6.0 + restart: unless-stopped + + webserver: + image: jonaswinkler/paperless-ng:latest + restart: unless-stopped + depends_on: + - broker + - gotenberg + - tika + ports: + - 8000:8000 + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8000"] + interval: 30s + timeout: 10s + retries: 5 + volumes: + - data:/usr/src/paperless/data + - media:/usr/src/paperless/media + - ./export:/usr/src/paperless/export + - ./consume:/usr/src/paperless/consume + env_file: docker-compose.env + environment: + PAPERLESS_REDIS: redis://broker:6379 + PAPERLESS_TIKA_ENABLED: 1 + PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 + PAPERLESS_TIKA_ENDPOINT: http://tika:9998 + + gotenberg: + image: thecodingmachine/gotenberg + restart: unless-stopped + environment: + DISABLE_GOOGLE_CHROME: 1 + + tika: + image: apache/tika + restart: unless-stopped + +volumes: + data: + media: diff --git a/docker/compose/docker-compose.sqlite.yml b/docker/compose/docker-compose.sqlite.yml new file mode 100644 index 000000000..f49e1d1e9 --- /dev/null +++ b/docker/compose/docker-compose.sqlite.yml @@ -0,0 +1,56 @@ +# docker-compose file for running paperless from the Docker Hub. +# This file contains everything paperless needs to run. +# Paperless supports amd64, arm and arm64 hardware. +# +# All compose files of paperless configure paperless in the following way: +# +# - Paperless is (re)started on system boot, if it was running before shutdown. +# - Docker volumes for storing data are managed by Docker. +# - Folders for importing and exporting files are created in the same directory +# as this file and mounted to the correct folders inside the container. +# - Paperless listens on port 8000. +# +# SQLite is used as the database. The SQLite file is stored in the data volume. +# +# To install and update paperless with this file, do the following: +# +# - Copy this file as 'docker-compose.yml' and the files 'docker-compose.env' +# and '.env' into a folder. +# - Run 'docker-compose pull'. +# - Run 'docker-compose run --rm webserver createsuperuser' to create a user. +# - Run 'docker-compose up -d'. +# +# For more extensive installation and update instructions, refer to the +# documentation. + +version: "3.4" +services: + broker: + image: redis:6.0 + restart: unless-stopped + + webserver: + image: jonaswinkler/paperless-ng:latest + restart: unless-stopped + depends_on: + - broker + ports: + - 8000:8000 + healthcheck: + test: ["CMD", "curl", "-f", "http://localhost:8000"] + interval: 30s + timeout: 10s + retries: 5 + volumes: + - data:/usr/src/paperless/data + - media:/usr/src/paperless/media + - ./export:/usr/src/paperless/export + - ./consume:/usr/src/paperless/consume + env_file: docker-compose.env + environment: + PAPERLESS_REDIS: redis://broker:6379 + + +volumes: + data: + media: diff --git a/docker/docker-entrypoint.sh b/docker/docker-entrypoint.sh old mode 100644 new mode 100755 index 13a0ba035..e494eac75 --- a/docker/docker-entrypoint.sh +++ b/docker/docker-entrypoint.sh @@ -4,87 +4,42 @@ set -e # Source: https://github.com/sameersbn/docker-gitlab/ map_uidgid() { - USERMAP_ORIG_UID=$(id -u paperless) - USERMAP_ORIG_GID=$(id -g paperless) - USERMAP_NEW_UID=${USERMAP_UID:-$USERMAP_ORIG_UID} - USERMAP_NEW_GID=${USERMAP_GID:-${USERMAP_ORIG_GID:-$USERMAP_NEW_UID}} - if [[ ${USERMAP_NEW_UID} != "${USERMAP_ORIG_UID}" || ${USERMAP_NEW_GID} != "${USERMAP_ORIG_GID}" ]]; then - echo "Mapping UID and GID for paperless:paperless to $USERMAP_NEW_UID:$USERMAP_NEW_GID" - usermod -u "${USERMAP_NEW_UID}" paperless - groupmod -o -g "${USERMAP_NEW_GID}" paperless - fi -} - - -wait_for_postgres() { - attempt_num=1 - max_attempts=5 - - echo "Waiting for PostgreSQL to start..." - - host="${PAPERLESS_DBHOST}" - port="${PAPERLESS_DBPORT}" - - if [[ -z $port ]] ; - then - port="5432" + USERMAP_ORIG_UID=$(id -u paperless) + USERMAP_ORIG_GID=$(id -g paperless) + USERMAP_NEW_UID=${USERMAP_UID:-$USERMAP_ORIG_UID} + USERMAP_NEW_GID=${USERMAP_GID:-${USERMAP_ORIG_GID:-$USERMAP_NEW_UID}} + if [[ ${USERMAP_NEW_UID} != "${USERMAP_ORIG_UID}" || ${USERMAP_NEW_GID} != "${USERMAP_ORIG_GID}" ]]; then + echo "Mapping UID and GID for paperless:paperless to $USERMAP_NEW_UID:$USERMAP_NEW_GID" + usermod -u "${USERMAP_NEW_UID}" paperless + groupmod -o -g "${USERMAP_NEW_GID}" paperless fi - - while !/usr/src/paperless/data/migration_lock - } initialize() { map_uidgid for dir in export data data/index media media/documents media/documents/originals media/documents/thumbnails; do - if [[ ! -d "../$dir" ]] - then - echo "creating directory ../$dir" + if [[ ! -d "../$dir" ]]; then + echo "Creating directory ../$dir" mkdir ../$dir fi done - chown -R paperless:paperless ../ + echo "Creating directory /tmp/paperless" + mkdir -p /tmp/paperless - migrations + set +e + echo "Adjusting permissions of paperless files. This may take a while." + chown -R paperless:paperless /tmp/paperless + find .. -not \( -user paperless -and -group paperless \) -exec chown paperless:paperless {} + + set -e + gosu paperless /sbin/docker-prepare.sh } install_languages() { + echo "Installing languages..." + local langs="$1" read -ra langs <<<"$langs" @@ -95,40 +50,43 @@ install_languages() { apt-get update for lang in "${langs[@]}"; do - pkg="tesseract-ocr-$lang" - # English is installed by default - #if [[ "$lang" == "eng" ]]; then - # continue - #fi + pkg="tesseract-ocr-$lang" + # English is installed by default + #if [[ "$lang" == "eng" ]]; then + # continue + #fi - if dpkg -s $pkg &> /dev/null; then - echo "package $pkg already installed!" - continue - fi + if dpkg -s $pkg &>/dev/null; then + echo "Package $pkg already installed!" + continue + fi - if ! apt-cache show $pkg &> /dev/null; then - echo "package $pkg not found! :(" - continue - fi + if ! apt-cache show $pkg &>/dev/null; then + echo "Package $pkg not found! :(" + continue + fi - echo "Installing package $pkg..." - if ! apt-get -y install "$pkg" &> /dev/null; then - echo "Could not install $pkg" - exit 1 - fi - done + echo "Installing package $pkg..." + if ! apt-get -y install "$pkg" &>/dev/null; then + echo "Could not install $pkg" + exit 1 + fi + done } +echo "Paperless-ng docker container starting..." + # Install additional languages if specified -if [[ ! -z "$PAPERLESS_OCR_LANGUAGES" ]]; then - install_languages "$PAPERLESS_OCR_LANGUAGES" +if [[ ! -z "$PAPERLESS_OCR_LANGUAGES" ]]; then + install_languages "$PAPERLESS_OCR_LANGUAGES" fi initialize if [[ "$1" != "/"* ]]; then - exec sudo -HEu paperless python3 manage.py "$@" + echo Executing management command "$@" + exec gosu paperless python3 manage.py "$@" else + echo Executing "$@" exec "$@" fi - diff --git a/docker/docker-prepare.sh b/docker/docker-prepare.sh new file mode 100755 index 000000000..28ddda55a --- /dev/null +++ b/docker/docker-prepare.sh @@ -0,0 +1,72 @@ +#!/usr/bin/env bash + +wait_for_postgres() { + attempt_num=1 + max_attempts=5 + + echo "Waiting for PostgreSQL to start..." + + host="${PAPERLESS_DBHOST}" + port="${PAPERLESS_DBPORT}" + + if [[ -z $port ]]; then + port="5432" + fi + + while ! /usr/src/paperless/data/migration_lock +} + +search_index() { + index_version=1 + index_version_file=/usr/src/paperless/data/.index_version + + if [[ (! -f "$index_version_file") || $(<$index_version_file) != "$index_version" ]]; then + echo "Search index out of date. Updating..." + python3 manage.py document_index reindex + echo $index_version | tee $index_version_file >/dev/null + fi +} + +superuser() { + if [[ -n "${PAPERLESS_ADMIN_USER}" ]]; then + python3 manage.py manage_superuser + fi +} + +do_work() { + if [[ -n "${PAPERLESS_DBHOST}" ]]; then + wait_for_postgres + fi + + migrations + + search_index + + superuser + +} + +do_work diff --git a/docker/hub/docker-compose.postgres.yml b/docker/hub/docker-compose.postgres.yml deleted file mode 100644 index 24f0e118f..000000000 --- a/docker/hub/docker-compose.postgres.yml +++ /dev/null @@ -1,44 +0,0 @@ -version: "3.4" -services: - broker: - image: redis:6.0 - restart: always - - db: - image: postgres:13 - restart: always - volumes: - - pgdata:/var/lib/postgresql/data - environment: - POSTGRES_DB: paperless - POSTGRES_USER: paperless - POSTGRES_PASSWORD: paperless - - webserver: - image: jonaswinkler/paperless-ng:0.9.6 - restart: always - depends_on: - - db - - broker - ports: - - 8000:8000 - healthcheck: - test: ["CMD", "curl", "-f", "http://localhost:8000"] - interval: 30s - timeout: 10s - retries: 5 - volumes: - - data:/usr/src/paperless/data - - media:/usr/src/paperless/media - - ./export:/usr/src/paperless/export - - ./consume:/usr/src/paperless/consume - env_file: docker-compose.env - environment: - PAPERLESS_REDIS: redis://broker:6379 - PAPERLESS_DBHOST: db - - -volumes: - data: - media: - pgdata: diff --git a/docker/hub/docker-compose.sqlite.yml b/docker/hub/docker-compose.sqlite.yml deleted file mode 100644 index 6ae619fd6..000000000 --- a/docker/hub/docker-compose.sqlite.yml +++ /dev/null @@ -1,31 +0,0 @@ -version: "3.4" -services: - broker: - image: redis:6.0 - restart: always - - webserver: - image: jonaswinkler/paperless-ng:0.9.6 - restart: always - depends_on: - - broker - ports: - - 8000:8000 - healthcheck: - test: ["CMD", "curl", "-f", "http://localhost:8000"] - interval: 30s - timeout: 10s - retries: 5 - volumes: - - data:/usr/src/paperless/data - - media:/usr/src/paperless/media - - ./export:/usr/src/paperless/export - - ./consume:/usr/src/paperless/consume - env_file: docker-compose.env - environment: - PAPERLESS_REDIS: redis://broker:6379 - - -volumes: - data: - media: diff --git a/docker/install_management_commands.sh b/docker/install_management_commands.sh new file mode 100755 index 000000000..468711d3c --- /dev/null +++ b/docker/install_management_commands.sh @@ -0,0 +1,6 @@ +for command in document_archiver document_exporter document_importer mail_fetcher document_create_classifier document_index document_renamer document_retagger document_thumbnails document_sanity_checker manage_superuser; +do + echo "installing $command..." + sed "s/management_command/$command/g" management_script.sh > /usr/local/bin/$command + chmod +x /usr/local/bin/$command +done diff --git a/docker/local/Dockerfile b/docker/local/Dockerfile deleted file mode 100644 index 461b9e4fc..000000000 --- a/docker/local/Dockerfile +++ /dev/null @@ -1,67 +0,0 @@ -FROM python:3.7-slim - -WORKDIR /usr/src/paperless/ - -COPY requirements.txt ./ - -#Dependencies -RUN apt-get update \ - && apt-get -y --no-install-recommends install \ - build-essential \ - curl \ - ghostscript \ - gnupg \ - icc-profiles-free \ - imagemagick \ - libatlas-base-dev \ - liblept5 \ - libmagic-dev \ - libpoppler-cpp-dev \ - libpq-dev \ - libqpdf-dev \ - libxml2 \ - optipng \ - pngquant \ - qpdf \ - sudo \ - tesseract-ocr \ - tesseract-ocr-eng \ - tesseract-ocr-deu \ - tesseract-ocr-fra \ - tesseract-ocr-ita \ - tesseract-ocr-spa \ - tzdata \ - unpaper \ - zlib1g \ - && pip3 install --upgrade supervisor setuptools \ - && pip install --no-cache-dir -r requirements.txt \ - && apt-get -y purge build-essential libqpdf-dev \ - && apt-get -y autoremove --purge \ - && rm -rf /var/lib/apt/lists/* \ - && mkdir /var/log/supervisord /var/run/supervisord - -# copy scripts -# this fixes issues with imagemagick and PDF -COPY docker/imagemagick-policy.xml /etc/ImageMagick-6/policy.xml -COPY docker/gunicorn.conf.py ./ -COPY docker/supervisord.conf /etc/supervisord.conf -COPY docker/docker-entrypoint.sh /sbin/docker-entrypoint.sh - -# copy app -COPY src/ ./src/ - -# add users, setup scripts -RUN addgroup --gid 1000 paperless \ - && useradd --uid 1000 --gid paperless --home-dir /usr/src/paperless paperless \ - && chown -R paperless:paperless . \ - && chmod 755 /sbin/docker-entrypoint.sh - -WORKDIR /usr/src/paperless/src/ - -RUN sudo -HEu paperless python3 manage.py collectstatic --clear --no-input - -VOLUME ["/usr/src/paperless/data", "/usr/src/paperless/media", "/usr/src/paperless/consume", "/usr/src/paperless/export"] -ENTRYPOINT ["/sbin/docker-entrypoint.sh"] -CMD ["/usr/local/bin/supervisord", "-c", "/etc/supervisord.conf"] - -LABEL maintainer="Jonas Winkler " diff --git a/docker/local/docker-compose.postgres.yml b/docker/local/docker-compose.postgres.yml deleted file mode 100644 index 59ae28939..000000000 --- a/docker/local/docker-compose.postgres.yml +++ /dev/null @@ -1,44 +0,0 @@ -version: "3.4" -services: - broker: - image: redis:6.0 - restart: always - - db: - image: postgres:13 - restart: always - volumes: - - pgdata:/var/lib/postgresql/data - environment: - POSTGRES_DB: paperless - POSTGRES_USER: paperless - POSTGRES_PASSWORD: paperless - - webserver: - build: . - restart: always - depends_on: - - db - - broker - ports: - - 8000:8000 - healthcheck: - test: ["CMD", "curl", "-f", "http://localhost:8000"] - interval: 30s - timeout: 10s - retries: 5 - volumes: - - data:/usr/src/paperless/data - - media:/usr/src/paperless/media - - ./export:/usr/src/paperless/export - - ./consume:/usr/src/paperless/consume - env_file: docker-compose.env - environment: - PAPERLESS_REDIS: redis://broker:6379 - PAPERLESS_DBHOST: db - - -volumes: - data: - media: - pgdata: diff --git a/docker/local/docker-compose.sqlite.yml b/docker/local/docker-compose.sqlite.yml deleted file mode 100644 index 2d21fa3f1..000000000 --- a/docker/local/docker-compose.sqlite.yml +++ /dev/null @@ -1,31 +0,0 @@ -version: "3.4" -services: - broker: - image: redis:6.0 - restart: always - - webserver: - build: . - restart: always - depends_on: - - broker - ports: - - 8000:8000 - healthcheck: - test: ["CMD", "curl", "-f", "http://localhost:8000"] - interval: 30s - timeout: 10s - retries: 5 - volumes: - - data:/usr/src/paperless/data - - media:/usr/src/paperless/media - - ./export:/usr/src/paperless/export - - ./consume:/usr/src/paperless/consume - env_file: docker-compose.env - environment: - PAPERLESS_REDIS: redis://broker:6379 - - -volumes: - data: - media: diff --git a/docker/management_script.sh b/docker/management_script.sh new file mode 100755 index 000000000..a0d38f2bc --- /dev/null +++ b/docker/management_script.sh @@ -0,0 +1,15 @@ +#!/bin/bash + +set -e + +cd /usr/src/paperless/src/ + +if [[ $(id -u) == 0 ]] ; +then + gosu paperless python3 manage.py management_command "$@" +elif [[ $(id -un) == "paperless" ]] ; +then + python3 manage.py management_command "$@" +else + echo "Unknown user." +fi diff --git a/docker/supervisord.conf b/docker/supervisord.conf index ebe0f005d..fca66c83c 100644 --- a/docker/supervisord.conf +++ b/docker/supervisord.conf @@ -8,7 +8,7 @@ loglevel=info ; log level; default info; others: debug,warn,trace user=root [program:gunicorn] -command=gunicorn -c /usr/src/paperless/gunicorn.conf.py -b 0.0.0.0:8000 paperless.wsgi +command=gunicorn -c /usr/src/paperless/gunicorn.conf.py paperless.asgi:application user=paperless stdout_logfile=/dev/stdout diff --git a/docs/administration.rst b/docs/administration.rst index 8885b7322..43710bd91 100644 --- a/docs/administration.rst +++ b/docs/administration.rst @@ -20,6 +20,14 @@ Options available to any installation of paperless: metadata to a specific folder. You may import your documents into a fresh instance of paperless again or store your documents in another DMS with this export. +* The document exporter is also able to update an already existing export. + Therefore, incremental backups with ``rsync`` are entirely possible. + +.. caution:: + + You cannot import the export generated with one version of paperless in a + different version of paperless. The export contains an exact image of the + database, and migrations may change the database layout. Options available to docker installations: @@ -48,16 +56,16 @@ Options available to bare-metal and non-docker installations: Restoring ========= - - - .. _administration-updating: -Updating paperless +Updating Paperless ################## +Docker Route +============ + If a new release of paperless-ng is available, upgrading depends on how you -installed paperless-ng in the first place. The releases are available at +installed paperless-ng in the first place. The releases are available at the `release page `_. First of all, ensure that paperless is stopped. @@ -69,106 +77,157 @@ First of all, ensure that paperless is stopped. After that, :ref:`make a backup `. -A. If you used the dockerfiles archive, simply download the files of the new release, - adjust the settings in the files (i.e., the path to your consumption directory), - and replace your existing docker-compose files. Then start paperless as usual, - which will pull the new image, and update your database, if necessary: +A. If you pull the image from the docker hub, all you need to do is: .. code:: shell-session - $ cd /path/to/paperless + $ docker-compose pull $ docker-compose up - If you see everything working, you can start paperless-ng with "-d" to have it - run in the background. + The docker-compose files refer to the ``latest`` version, which is always the latest + stable release. - .. hint:: - - The released docker-compose files specify exact versions to be pulled from the hub. - This is to ensure that if the docker-compose files should change at some point - (i.e., services updates/configured differently), you wont run into trouble due to - docker pulling the ``latest`` image and running it in an older environment. - -B. If you built the image yourself, grab the new archive and replace your current - paperless folder with the new contents. - - After that, make the necessary adjustments to the docker-compose.yml (i.e., - adjust your consumption directory). - - Build and start the new image with: +B. If you built the image yourself, do the following: .. code:: shell-session - $ cd /path/to/paperless + $ git pull + $ ./compile-frontend.sh $ docker-compose build $ docker-compose up - If you see everything working, you can start paperless-ng with "-d" to have it - run in the background. +Running ``docker-compose up`` will also apply any new database migrations. +If you see everything working, press CTRL+C once to gracefully stop paperless. +Then you can start paperless-ng with ``-d`` to have it run in the background. -.. hint:: + .. note:: - You can usually keep your ``docker-compose.env`` file, since this file will - never include mandatory configuration options. However, it is worth checking - out the new version of this file, since it might have new recommendations - on what to configure. + In version 0.9.14, the update process was changed. In 0.9.13 and earlier, the + docker-compose files specified exact versions and pull won't automatically + update to newer versions. In order to enable updates as described above, either + get the new ``docker-compose.yml`` file from `here `_ + or edit the ``docker-compose.yml`` file, find the line that says + .. code:: -Updating paperless without docker -================================= + image: jonaswinkler/paperless-ng:0.9.x + + and replace the version with ``latest``: + + .. code:: + + image: jonaswinkler/paperless-ng:latest + +Bare Metal Route +================ After grabbing the new release and unpacking the contents, do the following: 1. Update dependencies. New paperless version may require additional - dependencies. The dependencies required are listed in the section about + dependencies. The dependencies required are listed in the section about :ref:`bare metal installations `. -2. Update python requirements. If you use Pipenv, this is done with the following steps. +2. Update python requirements. Keep in mind to activate your virtual environment + before that, if you use one. .. code:: shell-session - $ pip install --upgrade pipenv - $ cd /path/to/paperless - $ pipenv clean - $ pipenv install + $ pip install -r requirements.txt - This creates a new virtual environment (or uses your existing environment) - and installs all dependencies into it. - -3. Collect static files. +3. Migrate the database. .. code:: shell-session $ cd src - $ pipenv run python3 manage.py collectstatic --clear - -4. Migrate the database. + $ python3 manage.py migrate + + This might not actually do anything. Not every new paperless version comes with new + database migrations. + +Ansible Route +============= + +Most of the update process is automated when using the ansible role. + +1. Update the role to the target release tag to make sure the ansible scripts are compatible: .. code:: shell-session - $ cd src - $ pipenv run python3 manage.py migrate + $ ansible-galaxy install git+https://github.com/jonaswinkler/paperless-ng.git,master --force + +2. Update the role variable definitions ``vars/paperless-ng.yml`` (where appropriate). + +3. Run the ansible playbook you created created during :ref:`installation ` again: + + .. note:: + + When ansible detects that an update run is in progress, it backs up the entire ``paperlessng_directory`` to ``paperlessng_directory-TIMESTAMP``. + Updates can be rolled back by simply moving the timestamped folder back to the original location. + If the update succeeds and you want to continue using the new release, please don't forget to delete the backup folder. + + .. code:: shell-session + + $ ansible-playbook playbook.yml + + +Downgrading Paperless +##################### + +Downgrades are possible. However, some updates also contain database migrations (these change the layout of the database and may move data). +In order to move back from a version that applied database migrations, you'll have to revert the database migration *before* downgrading, +and then downgrade paperless. + +This table lists the compatible versions for each database migration number. + ++------------------+-----------------+ +| Migration number | Version range | ++------------------+-----------------+ +| 1011 | 1.0.0 | ++------------------+-----------------+ +| 1012 | 1.1.0 - 1.2.1 | ++------------------+-----------------+ +| 1014 | 1.3.0 - 1.3.1 | ++------------------+-----------------+ +| 1016 | 1.3.2 - current | ++------------------+-----------------+ + +Execute the following management command to migrate your database: + +.. code:: shell-session + + $ python3 manage.py migrate documents + +.. note:: + + Some migrations cannot be undone. The command will issue errors if that happens. + +.. _utilities-management-commands: - Management utilities #################### Paperless comes with some management commands that perform various maintenance -tasks on your paperless instance. You can invoke these commands either by +tasks on your paperless instance. You can invoke these commands in the following way: + +With docker-compose, while paperless is running: .. code:: shell-session $ cd /path/to/paperless - $ docker-compose run --rm webserver + $ docker-compose exec webserver -or +With docker, while paperless is running: + +.. code:: shell-session + + $ docker exec -it + +Bare metal: .. code:: shell-session $ cd /path/to/paperless/src - $ pipenv run python manage.py - -depending on whether you use docker or not. + $ python3 manage.py All commands have built-in help, which can be accessed by executing them with the argument ``--help``. @@ -181,9 +240,16 @@ Document exporter The document exporter exports all your data from paperless into a folder for backup or migration to another DMS. +If you use the document exporter within a cronjob to backup your data you might use the ``-T`` flag behind exec to suppress "The input device is not a TTY" errors. For example: ``docker-compose exec -T webserver document_exporter ../export`` + .. code:: - document_exporter target + document_exporter target [-c] [-f] [-d] + + optional arguments: + -c, --compare-checksums + -f, --use-filename-format + -d, --delete ``target`` is a folder to which the data gets written. This includes documents, thumbnails and a ``manifest.json`` file. The manifest contains all metadata from @@ -193,6 +259,24 @@ When you use the provided docker compose script, specify ``../export`` as the target. This path inside the container is automatically mounted on your host on the folder ``export``. +If the target directory already exists and contains files, paperless will assume +that the contents of the export directory are a previous export and will attempt +to update the previous export. Paperless will only export changed and added files. +Paperless determines whether a file has changed by inspecting the file attributes +"date/time modified" and "size". If that does not work out for you, specify +``--compare-checksums`` and paperless will attempt to compare file checksums instead. +This is slower. + +Paperless will not remove any existing files in the export directory. If you want +paperless to also remove files that do not belong to the current export such as files +from deleted documents, specify ``--delete``. Be careful when pointing paperless to +a directory that already contains other files. + +The filenames generated by this command follow the format +``[date created] [correspondent] [title].[extension]``. +If you want paperless to use ``PAPERLESS_FILENAME_FORMAT`` for exported filenames +instead, specify ``--use-filename-format``. + .. _utilities-importer: @@ -323,6 +407,34 @@ the naming scheme. The command takes no arguments and processes all your documents at once. +.. _utilities-sanity-checker: + +Sanity checker +============== + +Paperless has a built-in sanity checker that inspects your document collection for issues. + +The issues detected by the sanity checker are as follows: + +* Missing original files. +* Missing archive files. +* Inaccessible original files due to improper permissions. +* Inaccessible archive files due to improper permissions. +* Corrupted original documents by comparing their checksum against what is stored in the database. +* Corrupted archive documents by comparing their checksum against what is stored in the database. +* Missing thumbnails. +* Inaccessible thumbnails due to improper permissions. +* Documents without any content (warning). +* Orphaned files in the media directory (warning). These are files that are not referenced by any document im paperless. + + +.. code:: + + document_sanity_checker + +The command takes no arguments. Depending on the size of your document archive, this may take some time. + + Fetching e-mail =============== @@ -387,7 +499,7 @@ Documents can be stored in Paperless using GnuPG encryption. Furthermore, the entire text content of the documents is stored plain in the database, even if your documents are encrypted. Filenames are not encrypted as well. - + Also, the web server provides transparent access to your encrypted documents. Consider running paperless on an encrypted filesystem instead, which will then @@ -410,6 +522,3 @@ Basic usage to disable encryption of your document store: .. code:: decrypt_documents [--passphrase SECR3TP4SSPHRA$E] - - -.. _Pipenv: https://pipenv.pypa.io/en/latest/ \ No newline at end of file diff --git a/docs/advanced_usage.rst b/docs/advanced_usage.rst index b5ae254b3..b7d46e063 100644 --- a/docs/advanced_usage.rst +++ b/docs/advanced_usage.rst @@ -5,98 +5,18 @@ Advanced topics Paperless offers a couple features that automate certain tasks and make your life easier. -Guesswork -######### - - -Any document you put into the consumption directory will be consumed, but if -you name the file right, it'll automatically set some values in the database -for you. This is is the logic the consumer follows: - -1. Try to find the correspondent, title, and tags in the file name following - the pattern: ``Date - Correspondent - Title - tag,tag,tag.pdf``. Note that - the format of the date is **rigidly defined** as ``YYYYMMDDHHMMSSZ`` or - ``YYYYMMDDZ``. The ``Z`` refers "Zulu time" AKA "UTC". - The tags are optional, so the format ``Date - Correspondent - Title.pdf`` - works as well. -2. If that doesn't work, we skip the date and try this pattern: - ``Correspondent - Title - tag,tag,tag.pdf``. -3. If that doesn't work, we try to find the correspondent and title in the file - name following the pattern: ``Correspondent - Title.pdf``. -4. If that doesn't work, just assume that the name of the file is the title. - -So given the above, the following examples would work as you'd expect: - -* ``20150314000700Z - Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` -* ``20150314Z - Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` -* ``Some Company Name - Invoice 2016-01-01 - money,invoices.pdf`` -* ``Another Company - Letter of Reference.jpg`` -* ``Dad's Recipe for Pancakes.png`` - -These however wouldn't work: - -* ``2015-03-14 00:07:00 UTC - Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` -* ``2015-03-14 - Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` -* ``Some Company Name, Invoice 2016-01-01, money, invoices.pdf`` -* ``Another Company- Letter of Reference.jpg`` - -Do I have to be so strict about naming? -======================================= - -Rather than using the strict document naming rules, one can also set the option -``PAPERLESS_FILENAME_DATE_ORDER`` in ``paperless.conf`` to any date order -that is accepted by dateparser_. Doing so will cause ``paperless`` to default -to any date format that is found in the title, instead of a date pulled from -the document's text, without requiring the strict formatting of the document -filename as described above. - -.. _dateparser: https://github.com/scrapinghub/dateparser/blob/v0.7.0/docs/usage.rst#settings - -.. _advanced-transforming_filenames: - -Transforming filenames for parsing -================================== - -Some devices can't produce filenames that can be parsed by the default -parser. By configuring the option ``PAPERLESS_FILENAME_PARSE_TRANSFORMS`` in -``paperless.conf`` one can add transformations that are applied to the filename -before it's parsed. - -The option contains a list of dictionaries of regular expressions (key: -``pattern``) and replacements (key: ``repl``) in JSON format, which are -applied in order by passing them to ``re.subn``. Transformation stops -after the first match, so at most one transformation is applied. The general -syntax is - -.. code:: python - - [{"pattern":"pattern1", "repl":"repl1"}, {"pattern":"pattern2", "repl":"repl2"}, ..., {"pattern":"patternN", "repl":"replN"}] - -The example below is for a Brother ADS-2400N, a scanner that allows -different names to different hardware buttons (useful for handling -multiple entities in one instance), but insists on adding ``_`` -to the filename. - -.. code:: python - - # Brother profile configuration, support "Name_Date_Count" (the default - # setting) and "Name_Count" (use "Name" as tag and "Count" as title). - PAPERLESS_FILENAME_PARSE_TRANSFORMS=[{"pattern":"^([a-z]+)_(\\d{8})_(\\d{6})_([0-9]+)\\.", "repl":"\\2\\3Z - \\4 - \\1."}, {"pattern":"^([a-z]+)_([0-9]+)\\.", "repl":" - \\2 - \\1."}] - - .. _advanced-matching: Matching tags, correspondents and document types ################################################ -After the consumer has tried to figure out what it could from the file name, -it starts looking at the content of the document itself. It will compare the -matching algorithms defined by every tag and correspondent already set in your -database to see if they apply to the text in that document. In other words, -if you defined a tag called ``Home Utility`` that had a ``match`` property of -``bc hydro`` and a ``matching_algorithm`` of ``literal``, Paperless will -automatically tag your newly-consumed document with your ``Home Utility`` tag -so long as the text ``bc hydro`` appears in the body of the document somewhere. +Paperless will compare the matching algorithms defined by every tag and +correspondent already set in your database to see if they apply to the text in +a document. In other words, if you defined a tag called ``Home Utility`` +that had a ``match`` property of ``bc hydro`` and a ``matching_algorithm`` of +``literal``, Paperless will automatically tag your newly-consumed document with +your ``Home Utility`` tag so long as the text ``bc hydro`` appears in the body +of the document somewhere. The matching logic is quite powerful, and supports searching the text of your document with different algorithms, and as such, some experimentation may be @@ -263,10 +183,10 @@ using the identifier which it has assigned to each document. You will end up get files like ``0000123.pdf`` in your media directory. This isn't necessarily a bad thing, because you normally don't have to access these files manually. However, if you wish to name your files differently, you can do that by adjusting the -``PAPERLESS_FILENAME_FORMAT`` settings variable. +``PAPERLESS_FILENAME_FORMAT`` configuration option. -This variable allows you to configure the filename (folders are allowed!) using -placeholders. For example, setting +This variable allows you to configure the filename (folders are allowed) using +placeholders. For example, configuring this to .. code:: bash @@ -277,17 +197,16 @@ will create a directory structure as follows: .. code:: 2019/ - my_bank/ - statement-january-0000001.pdf - statement-february-0000002.pdf + My bank/ + Statement January.pdf + Statement February.pdf 2020/ - my_bank/ - statement-january-0000003.pdf - shoe_store/ - my_new_shoes-0000004.pdf - -Paperless appends the unique identifier of each document to the filename. This -avoids filename clashes. + My bank/ + Statement January.pdf + Letter.pdf + Letter_01.pdf + Shoe store/ + My new shoes.pdf .. danger:: @@ -297,8 +216,10 @@ avoids filename clashes. Paperless provides the following placeholders withing filenames: +* ``{asn}``: The archive serial number of the document, or "none". * ``{correspondent}``: The name of the correspondent, or "none". * ``{document_type}``: The name of the document type, or "none". +* ``{tag_list}``: A comma separated list of all tags assigned to the document. * ``{title}``: The title of the document. * ``{created}``: The full date and time the document was created. * ``{created_year}``: Year created only. @@ -309,8 +230,14 @@ Paperless provides the following placeholders withing filenames: * ``{added_month}``: Month added only (number 1-12). * ``{added_day}``: Day added only (number 1-31). -Paperless will convert all values for the placeholders into values which are safe -for use in filenames. + +Paperless will try to conserve the information from your database as much as possible. +However, some characters that you can use in document titles and correspondent names (such +as ``: \ /`` and a couple more) are not allowed in filenames and will be replaced with dashes. + +If paperless detects that two documents share the same filename, paperless will automatically +append ``_01``, ``_02``, etc to the filename. This happens if all the placeholders in a filename +evaluate to the same value. .. hint:: diff --git a/docs/api.rst b/docs/api.rst index d352758fa..3a9d244c5 100644 --- a/docs/api.rst +++ b/docs/api.rst @@ -147,98 +147,57 @@ The REST api provides three different forms of authentication. Searching for documents ####################### -Paperless-ng offers API endpoints for full text search. These are as follows: +Full text searching is available on the ``/api/documents/`` endpoint. Two specific +query parameters cause the API to return full text search results: -``/api/search/`` -================ +* ``/api/documents/?query=your%20search%20query``: Search for a document using a full text query. + For details on the syntax, see :ref:`basic-usage_searching`. -Get search results based on a query. +* ``/api/documents/?more_like=1234``: Search for documents similar to the document with id 1234. -Query parameters: +Pagination works exactly the same as it does for normal requests on this endpoint. -* ``query``: The query string. See - `here `_ - for details on the syntax. -* ``page``: Specify the page you want to retrieve. Each page - contains 10 search results and the first page is ``page=1``, which - is the default if this is omitted. +Certain limitations apply to full text queries: -Result list object returned by the endpoint: +* Results are always sorted by search score. The results matching the query best will show up first. -.. code:: json +* Only a small subset of filtering parameters are supported. + +Furthermore, each returned document has an additional ``__search_hit__`` attribute with various information +about the search results: + +.. code:: { - "count": 1, - "page": 1, - "page_count": 1, - "corrected_query": "", + "count": 31, + "next": "http://localhost:8000/api/documents/?page=2&query=test", + "previous": null, "results": [ + ... + + { + "id": 123, + "title": "title", + "content": "content", + + ... + + "__search_hit__": { + "score": 0.343, + "highlights": "text Test text", + "rank": 23 + } + }, + + ... + ] } -* ``count``: The approximate total number of results. -* ``page``: The page returned to you. This might be different from - the page you requested, if you requested a page that is behind - the last page. In that case, the last page is returned. -* ``page_count``: The total number of pages. -* ``corrected_query``: Corrected version of the query string. Can be null. - If not null, can be used verbatim to start a new query. -* ``results``: A list of result objects on the current page. - -Result object: - -.. code:: json - - { - "id": 1, - "highlights": [ - - ], - "score": 6.34234, - "rank": 23, - "document": { - - } - } - -* ``id``: the primary key of the found document -* ``highlights``: an object containing parsable highlights for the result. - See below. -* ``score``: The score assigned to the document. A higher score indicates a - better match with the query. Search results are sorted descending by score. -* ``rank``: the position of the document within the entire search results list. -* ``document``: The full json of the document, as returned by - ``/api/documents//``. - -Highlights object: - -Highlights are provided as a list of fragments. A fragment is a longer section of -text from the original document. -Each fragment contains a list of strings, and some of them are marked as a highlight. - -.. code:: json - - [ - [ - {"text": "This is a sample text with a "}, - {"text": "highlighted", "term": 0}, - {"text": " word."} - ], - [ - {"text": "Another", "term": 1}, - {"text": " fragment with a highlight."} - ] - ] - - - -When ``term`` is present within a string, the word within ``text`` should be highlighted. -The term index groups multiple matches together and words with the same index -should get identical highlighting. -A client may use this example to produce the following output: - -... This is a sample text with a **highlighted** word. ... **Another** fragment with a highlight. ... +* ``score`` is an indication how well this document matches the query relative to the other search results. +* ``highlights`` is an excerpt from the document content and highlights the search terms with ```` tags as shown above. +* ``rank`` is the index of the search results. The first result will have rank 0. ``/api/search/autocomplete/`` ============================= @@ -289,3 +248,53 @@ The endpoint supports the following optional form fields: The endpoint will immediately return "OK" if the document consumption process was started successfully. No additional status information about the consumption process itself is available, since that happens in a different process. + + +.. _api-versioning: + +API Versioning +############## + +The REST API is versioned since Paperless-ng 1.3.0. + +* Versioning ensures that changes to the API don't break older clients. +* Clients specify the specific version of the API they wish to use with every request and Paperless will handle the request using the specified API version. +* Even if the underlying data model changes, older API versions will always serve compatible data. +* If no version is specified, Paperless will serve version 1 to ensure compatibility with older clients that do not request a specific API version. + +API versions are specified by submitting an additional HTTP ``Accept`` header with every request: + +.. code:: + + Accept: application/json; version=6 + +If an invalid version is specified, Paperless 1.3.0 will respond with "406 Not Acceptable" and an error message in the body. +Earlier versions of Paperless will serve API version 1 regardless of whether a version is specified via the ``Accept`` header. + +If a client wishes to verify whether it is compatible with any given server, the following procedure should be performed: + +1. Perform an *authenticated* request against any API endpoint. If the server is on version 1.3.0 or newer, the server will + add two custom headers to the response: + + .. code:: + + X-Api-Version: 2 + X-Version: 1.3.0 + +2. Determine whether the client is compatible with this server based on the presence/absence of these headers and their values if present. + + +API Changelog +============= + +Version 1 +--------- + +Initial API version. + +Version 2 +--------- + +* Added field ``Tag.color``. This read/write string field contains a hex color such as ``#a6cee3``. +* Added read-only field ``Tag.text_color``. This field contains the text color to use for a specific tag, which is either black or white depending on the brightness of ``Tag.color``. +* Removed field ``Tag.colour``. diff --git a/docs/changelog.rst b/docs/changelog.rst index a50fc31d5..4da732757 100644 --- a/docs/changelog.rst +++ b/docs/changelog.rst @@ -5,6 +5,570 @@ Changelog ********* +paperless-ng 1.5.0 +################## + +Support for Python 3.6 was dropped. + +* Updated python dependencies. +* Base image of the docker image changed from Debian Buster to Debian Bullseye due to its recent release. +* The docker image now uses python 3.9. +* Added the Luxembourgish locale. Thanks for translating! +* `Daniel Albers`_ added support for making the files and folders ignored by the paperless consume folder scanner configurable. See ``PAPERLESS_CONSUMER_IGNORE_PATTERNS``. + +paperless-ng 1.4.5 +################## + +This is a maintenance release. + +* Updated Python and Angular dependencies. +* Changed the algorithm that changes permissions during startup. This is still fast, + and will hopefully cause less issues. +* Fixed an issue that would sometimes cause paperless to write an incomplete + classification model file to disk. +* Fixed an issue with the OCRmyPDF parser that would always try to extract text + with PDFminer even from non-PDF files. + +paperless-ng 1.4.4 +################## + +* Drastically decreased the startup time of the docker container. The startup script adjusts file permissions of all data only if changes are required. +* Paperless mail: Added ability to specify the character set for each server. +* Document consumption: Ignore Mac OS specific files such as ``.DS_STORE`` and ``._XXXXX.pdf``. +* Fixed an issue with the automatic matching algorithm that prevents paperless from consuming new files. +* Updated translations. + +paperless-ng 1.4.3 +################## + +* Additions and changes + + * Added Swedish locale. + * `Stéphane Brunner`_ added an option to disable the progress bars of all management commands. + * `Jo Vandeginste`_ added support for RTF documents to the Apache TIKA parser. + * `Michael Shamoon`_ added dark mode for the login and logout pages. + * `Alexander Menk`_ added additional stylesheets for printing. You can now print any page of paperless and the print result will hide the page header, sidebar, and action buttons. + * Added support for sorting when using full text search. + +* Fixes + + * `puuu`_ fixed ``PAPERLESS_FORCE_SCRIPT_NAME``. You can now host paperless on sub paths such as ``https://localhost:8000/paperless/``. + * Fixed an issue with the document consumer crashing on certain documents due to issues with pdfminer.six. This library is used for PDF text extraction. + +paperless-ng 1.4.2 +################## + +* Fixed an issue with ``sudo`` that caused paperless to not start on many Raspberry Pi devices. Thank you `WhiteHatTux`_! + +paperless-ng 1.4.1 +################## + +* Added Polish locale. + +* Changed some parts of the Dockerfile to hopefully restore functionality on certain ARM devices. + +* Updated python dependencies. + +* `Michael Shamoon`_ added a sticky filter / bulk edit bar. + +* `sbrl`_ changed the docker-entrypoint.sh script to increase compatibility with NFS shares. + +* `Chris Nagy`_ added support for creating a super user by passing ``PAPERLESS_ADMIN_USER`` and + ``PAPERLESS_ADMIN_PASSWORD`` as environment variables to the docker container. + +paperless-ng 1.4.0 +################## + +* Docker images now use tesseract 4.1.1, which should fix a series of issues with OCR. + +* The full text search now displays results using the default document list. This enables + selection, filtering and bulk edit on search results. + +* Changes + + * Firefox only: Highlight search query in PDF previews. + + * New URL pattern for accessing documents by ASN directly (http:///asn/123) + + * Added logging when executing pre- and post-consume scripts. + + * Better error logging during document consumption. + + * Updated python dependencies. + + * Automatically inserts typed text when opening "Create new" dialogs on the document details page. + +* Fixes + + * Fixed an issue with null characters in the document content. + +.. note:: + + The changed to the full text searching require you to reindex your documents. + *The docker image does this automatically, you don't need to do anything.* + To do this, execute the ``document_index reindex`` management command + (see :ref:`administration-index`). + +paperless-ng 1.3.2 +################## + +* Added translation into Portuguese. + +* Changes + + * The exporter now exports user accounts, mail accounts, mail rules and saved views as well. + +* Fixes + + * Minor layout issues with document cards and the log viewer. + + * Fixed an issue with any/all/exact matching when characters used in regular expressions were used for the match. + +paperless-ng 1.3.1 +################## + +* Added translation into Spanish and Russian. + +* Other changes + + * ISO-8601 date format will now always show years with 4 digits. + + * Added the ability to search for a document with a specific ASN. + + * The document cards now display ASN, types and dates in a more organized way. + + * Added document previews when hovering over the preview button. + +* Fixes + + * The startup check for write permissions now works properly on NFS shares. + + * Fixed an issue with the search results score indicator. + + * Paperless was unable to generate thumbnails for encrypted PDF files and failed. Paperless will now generate a default thumbnail for these files. + + * Fixed ``AUTO_LOGIN_USERNAME``: Unable to perform POST/PUT/DELETE requests and unable to receive WebSocket messages. + +paperless-ng 1.3.0 +################## + +This release contains new database migrations. + +* Changes + + * The REST API is versioned from this point onwards. This will allow me to make changes without breaking existing clients. See the documentation about :ref:`api-versioning` for details. + + * Added a color picker for tag colors. + + * Added the ability to use the filter for searching the document content as well. + + * Added translations into Italian and Romanian. Thank you! + + * Close individual documents from the sidebar. Thanks to `Michael Shamoon`_. + + * `BolkoSchreiber `_ added an option to disable/enable thumbnail inversion in dark mode. + + * `Simon Taddiken `_ added the ability to customize the header used for remote user authentication with SSO applications. + +* Bug fixes + + * Fixed an issue with the auto matching algorithm when more than 256 tags were used. + + +paperless-ng 1.2.1 +################## + +* `Rodrigo Avelino `_ translated Paperless into Portuguese (Brazil)! + +* The date input fields now respect the currently selected date format. + +* Added a fancy icon when adding paperless to the home screen on iOS devices. Thanks to `Joel Nordell `_. + +* When using regular expression matching, the regular expression is now validated before saving the tag/correspondent/type. + +* Regression fix: Dates on the front end did not respect date locale settings in some cases. + +paperless-ng 1.2.0 +################## + +* Changes to the OCRmyPDF integration + + * Added support for deskewing and automatic rotation of incorrectly rotated pages. This is enabled by default, see :ref:`configuration-ocr`. + * Better support for encrypted files. + * Better support for various other PDF files: Paperless will now attempt to force OCR with safe options when OCR fails with the configured options. + * Added an explicit option to skip cleaning with ``unpaper``. + +* Download multiple selected documents as a zip archive. + +* The document list now remembers the current page. + +* Improved responsiveness when switching between saved views and the document list. + +* Increased the default wait time when observing files in the consumption folder + with polling from 1 to 5 seconds. This will decrease the likelihood of paperless + consuming partially written files. + +* Fixed a crash of the document archiver management command when trying to process documents with unknown mime types. + +* Paperless no longer depends on ``libpoppler-cpp-dev``. + +paperless-ng 1.1.4 +################## + +* Added English (GB) locale. + +* Added ISO-8601 date display option. + +paperless-ng 1.1.3 +################## + +* Added a docker-specific configuration option to adjust the number of + worker processes of the web server. See :ref:`configuration-docker`. + +* Some more memory usage optimizations. + +* Don't show inbox statistics if no inbox tag is defined. + +paperless-ng 1.1.2 +################## + +* Always show top left corner of thumbnails, even for extra wide documents. + +* Added a management command for executing the sanity checker directly. + See :ref:`utilities-sanity-checker`. + +* The weekly sanity check now reports messages in the log files. + +* Fixed an issue with the metadata tab not reporting anything in case of missing files. + +* Reverted a change from 1.1.0 that caused huge memory usage due to redis caching. + +* Some memory usage optimizations. + +paperless-ng 1.1.1 +################## + +This release contains new database migrations. + +* Fixed a bug in the sanity checker that would cause it to display "x not in list" errors instead of actual issues. + +* Fixed a bug with filename generation for archive filenames that would cause the archive files of two documents to overlap. + + * This happened when ``PAPERLESS_FILENAME_FORMAT`` is used and the filenames of two or more documents are the same, except for the file extension. + * Paperless will now store the archive filename in the database as well instead of deriving it from the original filename, and use the + same logic for detecting and avoiding filename clashes that's also used for original filenames. + * The migrations will repair any missing archive files. If you're using tika, ensure that tika is running while performing the migration. Docker-compose will take care of that. + +* Fixed a bug with thumbnail regeneration when TIKA integration was used. + +* Added ASN as a placeholder field to the filename format. + +* The docker image now comes with built-in shortcuts for most management commands. These are now the recommended way to execute management commands, since these + also ensure that they're always executed as the paperless user and you're less likely to run into permission issues. See :ref:`utilities-management-commands`. + +paperless-ng 1.1.0 +################## + +* Document processing status + + * Paperless now shows the status of processing documents on the dashboard in real time. + * Status notifications when + + * New documents are detected in the consumption folder, in mails, uploaded on the front end, + or added with one of the mobile apps. + * Documents are successfully added to paperless. + * Document consumption failed (with error messages) + + * Configuration options to enable/disable individual notifications. + +* Live updates to document lists and saved views when new documents are added. + + .. hint:: + + For status notifications and live updates to work, paperless now requires an `ASGI `_-enabled + web server. The docker images uses ``gunicorn`` and an ASGI-enabled worker called `uvicorn `_, + and there is no need to configure anything. + + For bare metal installations, changes are required for the notifications to work. Adapt the service ``paperless-webserver.service`` + to use the supplied ``gunicorn.conf.py`` configuration file and adapt the reference to the ASGI application as follows: + + .. code:: + + ExecStart=/opt/paperless/.local/bin/gunicorn -c /opt/paperless/gunicorn.conf.py paperless.asgi:application + + Paperless will continue to work with WSGI, but you will not get any status notifications. + + Apache ``mod_wsgi`` users, see :ref:`this note `. + +* Paperless now offers suggestions for tags, correspondents and types on the document detail page. + +* Added an interactive easy install script that automatically downloads, configures and starts paperless with docker. + +* Official support for Python 3.9. + +* Other changes and fixes + + * Adjusted the default parallelization settings to run more than one task in parallel on systems with 4 or less cores. + This addresses issues with paperless not consuming any new files when other tasks are running. + + * Fixed a rare race condition that would cause paperless to process incompletely written files when using the upload on the dashboard. + + * The document classifier no longer issues warnings and errors when auto matching is not used at all. + + * Better icon for document previews. + + * Better info section in the side bar. + + * Paperless no longer logs to the database. Instead, logs are written to rotating log files. This solves many "database is locked" + issues on Raspberry Pi, especially when SQLite is used. + + * By default, log files are written to ``PAPERLESS_DATA_DIR/log/``. Logging settings can be adjusted with + ``PAPERLESS_LOGGING_DIR``, ``PAPERLESS_LOGROTATE_MAX_SIZE`` and + ``PAPERLESS_LOGROTATE_MAX_BACKUPS``. + +paperless-ng 1.0.0 +################## + +Nothing special about this release, but since there are relatively few bug reports coming in, I think that this is reasonably stable. + +* Document export + + * The document exporter has been rewritten to support updating an already existing export in place. + This enables incremental backups with ``rsync``. + * The document exporter supports naming exported files according to ``PAPERLESS_FILENAME_FORMAT``. + * The document exporter locks the media directory and the database during execution to ensure that + the resulting export is consistent. + * See the :ref:`updated documentation ` for more details. + +* Other changes and additions + + * Added a language selector to the settings. + * Added date format options to the settings. + * Range selection with shift clicking is now possible in the document list. + * Filtering correspondent, type and tag management pages by name. + * Focus "Name" field in dialogs by default. + + +paperless-ng 0.9.14 +################### + +Starting with this version, releases are getting built automatically. This release also comes with changes on how to install and +update paperless. + +* Paperless now uses GitHub Actions to make releases and build docker images. + + * Docker images are available for amd64, armhf, and aarch64. + * When you pull an image from Docker Hub, Docker will automatically select the correct image for you. + +* Changes to docker installations and updates + + * The ``-dockerfiles.tar.xz`` release archive is gone. Instead, simply grab the docker files from ``/docker/compose`` in the repository + if you wish to install paperless by pulling from the hub. + * The docker compose files in ``/docker/compose`` were changed to always use the ``latest`` version automatically. In order to do further + updates, simply do a ``docker-compose pull``. The documentation has been updated. + * The docker compose files were changed to restart paperless on system boot only if it was running before shutdown. + * Documentation of the docker-compose files about what they do. + +* Changes to bare metal installations and updates + + * The release archive is built exactly like before. However, the release now comes with already compiled translation messages and + collected static files. Therefore, the update steps ``compilemessages`` and ``collectstatic`` are now obsolete. + +* Other changes + + * A new configuration option ``PAPERLESS_IGNORE_DATES`` was added by `jayme-github`_. This can be used to instruct paperless to ignore + certain dates (such as your date of birth) when guessing the date from the document content. This was actually introduced in 0.9.12, + I just forgot to mention it in the changelog. + * The filter drop downs now display selected entries on top of all other entries. + * The PostgreSQL client now supports setting an explicit ``sslmode`` to force encryption of the connection to PostgreSQL. + * The docker images now come with ``jbig2enc``, which is a lossless image encoder for PDF documents and decreases the size of certain + PDF/A documents. + * When using any of the manual matching algorithms, paperless now logs messages about when and why these matching algorithms matched. + * The default settings for parallelization in paperless were adjusted to always leave one CPU core free. + * Added an option to the frontend to choose which method to use for displaying PDF documents. + +* Fixes + + * An issue with the tika parser not picking up files from the consumption directory was fixed. + * A couple changes to the dark mode and fixes to several other layout issues. + * An issue with the drop downs for correspondents, tags and types not properly supporting filtering with special characters was fixed. + * Fixed an issue with filenames of downloaded files: Dates where off by one day due to timezone issues. + * Searching will continue to work even when the index returns non-existing documents. This resulted in "Document does not exist" errors + before. Instead, a warning is logged, indicating the issue. + * An issue with the consumer crashing when invalid regular expression were used was fixed. + +paperless-ng 0.9.13 +################### + +* Fixed an issue with Paperless not starting due to the new Tika integration when ``USERMAP_UID`` and ``USERMAP_GID`` was used + in the ``docker-compose.env`` file. + +paperless-ng 0.9.12 +################### + +* Paperless localization + + * Thanks to the combined efforts of many users, Paperless is now available in English, Dutch, French and German. + +* Thanks to `Jo Vandeginste`_, Paperless has optional support for Office documents such as .docx, .doc, .odt and more. + + * See the :ref:`configuration` on how to enable this feature. This feature requires two additional services + (one for parsing Office documents and metadata extraction and another for converting Office documents to PDF), and is therefore + not enabled on default installations. + * As with all other documents, paperless converts Office documents to PDF and stores both the original as well as the archived PDF. + +* Dark mode + + * Thanks to `Michael Shamoon`_, paperless now has a dark mode. Configuration is available in the settings. + +* Other changes and additions + + * The PDF viewer now uses a local copy of some dependencies instead of fetching them from the internet. Thanks to `slorenz`_. + * Revamped search bar styling thanks to `Michael Shamoon`_. + * Sorting in the document list by clicking on table headers. + * A button was added to the document detail page that assigns a new ASN to a document. + * Form field validation: When providing invalid input in a form (such as a duplicate ASN or no name), paperless now has visual + indicators and clearer error messages about what's wrong. + * Paperless disables buttons with network actions (such as save and delete) when a network action is active. This indicates that + something is happening and prevents double clicking. + * When using "Save & next", the title field is focussed automatically to better support keyboard editing. + * E-Mail: Added filter rule parameters to allow inline attachments (watch out for mails with inlined images!) and attachment filename filters + with wildcards. + * Support for remote user authentication thanks to `Michael Shamoon`_. This is useful for hiding Paperless behind single sign on applications + such as `authelia `_. + * "Clear filters" has been renamed to "Reset filters" and now correctly restores the default filters on saved views. Thanks to `Michael Shamoon`_ + +* Fixes + + * Paperless was unable to save views when "Not assigned" was chosen in one of the filter dropdowns. + * Clearer error messages when pre and post consumption scripts do not exist. + * The post consumption script is executed later in the consumption process. Before the change, an ID was passed to the script referring to + a document that did not yet exist in the database. + +paperless-ng 0.9.11 +################### + +* Fixed an issue with the docker image not starting at all due to a configuration change of the web server. + + +paperless-ng 0.9.10 +################### + +* Bulk editing + + * Thanks to `Michael Shamoon`_, we've got a new interface for the bulk editor. + * There are some configuration options in the settings to alter the behavior. + +* Other changes and additions + + * Thanks to `zjean`_, paperless now publishes a webmanifest, which is useful for adding the application to home screens on mobile devices. + * The Paperless-ng logo now navigates to the dashboard. + * Filter for documents that don't have any correspondents, types or tags assigned. + * Tags, types and correspondents are now sorted case insensitive. + * Lots of preparation work for localization support. + +* Fixes + + * Added missing dependencies for Raspberry Pi builds. + * Fixed an issue with plain text file consumption: Thumbnail generation failed due to missing fonts. + * An issue with the search index reporting missing documents after bulk deletes was fixed. + * Issue with the tag selector not clearing input correctly. + * The consumer used to stop working when encountering an incomplete classifier model file. + +.. note:: + + The bulk delete operations did not update the search index. Therefore, documents that you deleted remained in the index and + caused the search to return messages about missing documents when searching. Further bulk operations will properly update + the index. + + However, this change is not retroactive: If you used the delete method of the bulk editor, you need to reindex your search index + by :ref:`running the management command document_index with the argument reindex `. + +paperless-ng 0.9.9 +################## + +Christmas release! + +* Bulk editing + + * Paperless now supports bulk editing. + * The following operations are available: Add and remove correspondents, tags, document types from selected documents, as well as mass-deleting documents. + * We've got a more fancy UI in the works that makes these features more accessible, but that's not quite ready yet. + +* Searching + + * Paperless now supports searching for similar documents ("More like this") both from the document detail page as well as from individual search results. + * A search score indicates how well a document matches the search query, or how similar a document is to a given reference document. + +* Other additions and changes + + * Clarification in the UI that the fields "Match" and "Is insensitive" are not relevant for the Auto matching algorithm. + * New select interface for tags, types and correspondents allows filtering. This also improves tag selection. Thanks again to `Michael Shamoon`_! + * Page navigation controls for the document viewer, thanks to `Michael Shamoon`_. + * Layout changes to the small cards document list. + * The dashboard now displays the username (or full name if specified in the admin) on the dashboard. + +* Fixes + + * An error that caused the document importer to crash was fixed. + * An issue with changes not being possible when ``PAPERLESS_COOKIE_PREFIX`` is used was fixed. + * The date selection filters now allow manual entry of dates. + +* Feature Removal + + * Most of the guesswork features have been removed. Paperless no longer tries to extract correspondents and tags from file names. + +paperless-ng 0.9.8 +################## + +This release addresses two severe issues with the previous release. + +* The delete buttons for document types, correspondents and tags were not working. +* The document section in the admin was causing internal server errors (500). + + +paperless-ng 0.9.7 +################## + + +* Front end + + * Thanks to the hard work of `Michael Shamoon`_, paperless now comes with a much more streamlined UI for + filtering documents. + + * `Michael Shamoon`_ replaced the document preview with another component. This should fix compatibility with Safari browsers. + + * Added buttons to the management pages to quickly show all documents with one specific tag, correspondent, or title. + + * Paperless now stores your saved views on the server and associates them with your user account. + This means that you can access your views on multiple devices and have separate views for different users. + You will have to recreate your views. + + * The GitHub and documentation links now open in new tabs/windows. Thanks to `rYR79435`_. + + * Paperless now generates default saved view names when saving views with certain filter rules. + + * Added a small version indicator to the front end. + +* Other additions and changes + + * The new filename format field ``{tag_list}`` inserts a list of tags into the filename, separated by comma. + * The ``document_retagger`` no longer removes inbox tags or tags without matching rules. + * The new configuration option ``PAPERLESS_COOKIE_PREFIX`` allows you to run multiple instances of paperless on different ports. + This option enables you to be logged in into multiple instances by specifying different cookie names for each instance. + +* Fixes + + * Sometimes paperless would assign dates in the future to newly consumed documents. + * The filename format fields ``{created_month}`` and ``{created_day}`` now use a leading zero for single digit values. + * The filename format field ``{tags}`` can no longer be used without arguments. + * Paperless was not able to consume many images (especially images from mobile scanners) due to missing DPI information. + Paperless now assumes A4 paper size for PDF generation if no DPI information is present. + * Documents with empty titles could not be opened from the table view due to the link being empty. + * Fixed an issue with filenames containing special characters such as ``:`` not being accepted for upload. + * Fixed issues with thumbnail generation for plain text files. + + paperless-ng 0.9.6 ################## @@ -841,6 +1405,16 @@ bulk of the work on this big change. * Initial release +.. _Alexander Menk: https://github.com/amenk +.. _puuu: https://github.com/puuu +.. _WhiteHatTux: https://github.com/WhiteHatTux +.. _Chris Nagy: https://github.com/what-name +.. _sbrl: https://github.com/sbrl +.. _slorenz: https://github.com/sisao +.. _Jo Vandeginste: https://github.com/jovandeginste +.. _zjean: https://github.com/zjean +.. _rYR79435: https://github.com/rYR79435 +.. _Michael Shamoon: https://github.com/shamoon .. _jayme-github: http://github.com/jayme-github .. _Brian Conn: https://github.com/TheConnMan .. _Christopher Luu: https://github.com/nuudles @@ -908,6 +1482,7 @@ bulk of the work on this big change. .. _JOKer: https://github.com/MasterofJOKers .. _Brian Cribbs: https://github.com/cribbstechnolog .. _Brendan M. Sleight: https://github.com/bmsleight +.. _Daniel Albers: https://github.com/AlD .. _#20: https://github.com/the-paperless-project/paperless/issues/20 .. _#44: https://github.com/the-paperless-project/paperless/issues/44 diff --git a/docs/conf.py b/docs/conf.py index b2442ddc9..71abe2079 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -28,7 +28,7 @@ master_doc = 'index' # General information about the project. project = u'Paperless-ng' -copyright = u'2015, Daniel Quinn' +copyright = u'2021, Daniel Quinn, Jonas Winkler' # The version info for the project you're documenting, acts as replacement for # |version| and |release|, also used in various other places throughout the diff --git a/docs/configuration.rst b/docs/configuration.rst index d3f47215b..27ace497b 100644 --- a/docs/configuration.rst +++ b/docs/configuration.rst @@ -53,6 +53,12 @@ PAPERLESS_DBPASS= Defaults to "paperless". +PAPERLESS_DBSSLMODE= + SSL mode to use when connecting to PostgreSQL. + + See `the official documentation about sslmode `_. + + Default is ``prefer``. Paths and folders ################# @@ -66,13 +72,13 @@ PAPERLESS_CONSUMPTION_DIR= container. Change the local consumption directory in the docker-compose.yml file instead. - Defaults to "../consume", relative to the "src" directory. + Defaults to "../consume/", relative to the "src" directory. PAPERLESS_DATA_DIR= This is where paperless stores all its data (search index, SQLite database, classification model, etc). - Defaults to "../data", relative to the "src" directory. + Defaults to "../data/", relative to the "src" directory. PAPERLESS_MEDIA_ROOT= This is where your documents and thumbnails are stored. @@ -80,7 +86,7 @@ PAPERLESS_MEDIA_ROOT= You can set this and PAPERLESS_DATA_DIR to the same folder to have paperless store all its data within the same volume. - Defaults to "../media", relative to the "src" directory. + Defaults to "../media/", relative to the "src" directory. PAPERLESS_STATICDIR= Override the default STATIC_ROOT here. This is where all static files @@ -88,7 +94,7 @@ PAPERLESS_STATICDIR= Unless you're doing something fancy, there is no need to override this. - Defaults to "../static", relative to the "src" directory. + Defaults to "../static/", relative to the "src" directory. PAPERLESS_FILENAME_FORMAT= Changes the filenames paperless uses to store documents in the media directory. @@ -96,6 +102,25 @@ PAPERLESS_FILENAME_FORMAT= Default is none, which disables this feature. +PAPERLESS_LOGGING_DIR= + This is where paperless will store log files. + + Defaults to "``PAPERLESS_DATA_DIR``/log/". + + +Logging +####### + +PAPERLESS_LOGROTATE_MAX_SIZE= + Maximum file size for log files before they are rotated, in bytes. + + Defaults to 1 MiB. + +PAPERLESS_LOGROTATE_MAX_BACKUPS= + Number of rotated log files to keep. + + Defaults to 20. + Hosting & Security ################## @@ -129,10 +154,6 @@ PAPERLESS_FORCE_SCRIPT_NAME= To host paperless under a subpath url like example.com/paperless you set this value to /paperless. No trailing slash! - .. note:: - - I don't know if this works in paperless-ng. Probably not. - Defaults to none, which hosts paperless at "/". PAPERLESS_STATIC_URL= @@ -152,6 +173,30 @@ PAPERLESS_AUTO_LOGIN_USERNAME= Defaults to none, which disables this feature. +PAPERLESS_ADMIN_USER= + If this environment variable is specified, Paperless automatically creates + a superuser with the provided username at start. This is useful in cases + where you can not run the `createsuperuser` command seperately, such as Kubernetes + or AWS ECS. + + Requires `PAPERLESS_ADMIN_PASSWORD` to be set. + + .. note:: + + This will not change an existing [super]user's password, nor will + it recreate a user that already exists. You can leave this throughout + the lifecycle of the containers. + +PAPERLESS_ADMIN_MAIL= + (Optional) Specify superuser email address. Only used when + `PAPERLESS_ADMIN_USER` is set. + + Defaults to ``root@localhost``. + +PAPERLESS_ADMIN_PASSWORD= + Only used when `PAPERLESS_ADMIN_USER` is set. + This will be the password of the automatically created superuser. + PAPERLESS_COOKIE_PREFIX= Specify a prefix that is added to the cookies used by paperless to identify @@ -162,6 +207,39 @@ PAPERLESS_COOKIE_PREFIX= Defaults to ``""``, which does not alter the cookie names. +PAPERLESS_ENABLE_HTTP_REMOTE_USER= + Allows authentication via HTTP_REMOTE_USER which is used by some SSO + applications. + + .. warning:: + + This will allow authentication by simply adding a ``Remote-User: `` header + to a request. Use with care! You especially *must* ensure that any such header is not + passed from your proxy server to paperless. + + If you're exposing paperless to the internet directly, do not use this. + + Also see the warning `in the official documentation `. + + Defaults to `false` which disables this feature. + +PAPERLESS_HTTP_REMOTE_USER_HEADER_NAME= + If `PAPERLESS_ENABLE_HTTP_REMOTE_USER` is enabled, this property allows to + customize the name of the HTTP header from which the authenticated username + is extracted. Values are in terms of + [HttpRequest.META](https://docs.djangoproject.com/en/3.1/ref/request-response/#django.http.HttpRequest.META). + Thus, the configured value must start with `HTTP_` followed by the + normalized actual header name. + + Defaults to `HTTP_REMOTE_USER`. + +PAPERLESS_LOGOUT_REDIRECT_URL= + URL to redirect the user to after a logout. This can be used together with + `PAPERLESS_ENABLE_HTTP_REMOTE_USER` to redirect the user back to the SSO + application's logout page. + + Defaults to None, which disables this feature. + .. _configuration-ocr: OCR settings @@ -171,7 +249,6 @@ Paperless uses `OCRmyPDF `_ for performing OCR on documents and images. Paperless uses sensible defaults for most settings, but all of them can be configured to your needs. - PAPERLESS_OCR_LANGUAGE= Customize the language that paperless will attempt to use when parsing documents. @@ -188,6 +265,8 @@ PAPERLESS_OCR_LANGUAGE= Defaults to "eng". + Note: If your language contains a '-' such as chi-sim, you must use chi_sim + PAPERLESS_OCR_MODE= Tell paperless when and how to perform ocr on your documents. Four modes are available: @@ -210,20 +289,68 @@ PAPERLESS_OCR_MODE= into images and puts the OCRed text on top. This works for all documents, however, the resulting document may be significantly larger and text won't appear as sharp when zoomed in. - + The default is ``skip``, which only performs OCR when necessary and always creates archived documents. + Read more about this in the `OCRmyPDF documentation `_. + +PAPERLESS_OCR_CLEAN= + Tells paperless to use ``unpaper`` to clean any input document before + sending it to tesseract. This uses more resources, but generally results + in better OCR results. The following modes are available: + + * ``clean``: Apply unpaper. + * ``clean-final``: Apply unpaper, and use the cleaned images to build the + output file instead of the original images. + * ``none``: Do not apply unpaper. + + Defaults to ``clean``. + + .. note:: + + ``clean-final`` is incompatible with ocr mode ``redo``. When both + ``clean-final`` and the ocr mode ``redo`` is configured, ``clean`` + is used instead. + +PAPERLESS_OCR_DESKEW= + Tells paperless to correct skewing (slight rotation of input images mainly + due to improper scanning) + + Defaults to ``true``, which enables this feature. + + .. note:: + + Deskewing is incompatible with ocr mode ``redo``. Deskewing will get + disabled automatically if ``redo`` is used as the ocr mode. + +PAPERLESS_OCR_ROTATE_PAGES= + Tells paperless to correct page rotation (90°, 180° and 270° rotation). + + If you notice that paperless is not rotating incorrectly rotated + pages (or vice versa), try adjusting the threshold up or down (see below). + + Defaults to ``true``, which enables this feature. + + +PAPERLESS_OCR_ROTATE_PAGES_THRESHOLD= + Adjust the threshold for automatic page rotation by ``PAPERLESS_OCR_ROTATE_PAGES``. + This is an arbitrary value reported by tesseract. "15" is a very conservative value, + whereas "2" is a very aggressive option and will often result in correctly rotated pages + being rotated as well. + + Defaults to "12". + PAPERLESS_OCR_OUTPUT_TYPE= Specify the the type of PDF documents that paperless should produce. - + * ``pdf``: Modify the PDF document as little as possible. * ``pdfa``: Convert PDF documents into PDF/A-2b documents, which is a subset of the entire PDF specification and meant for storing documents long term. * ``pdfa-1``, ``pdfa-2``, ``pdfa-3`` to specify the exact version of PDF/A you wish to use. - + If not specified, ``pdfa`` is used. Remember that paperless also keeps the original input file as well as the archived version. @@ -240,7 +367,6 @@ PAPERLESS_OCR_PAGES= Defaults to 0, which disables this feature and always uses all pages. - PAPERLESS_OCR_IMAGE_DPI= Paperless will OCR any images you put into the system and convert them into PDF documents. This is useful if your scanner produces images. @@ -251,17 +377,17 @@ PAPERLESS_OCR_IMAGE_DPI= Set this to the DPI your scanner produces images at. - Default is none, which causes paperless to fail if no DPI information is - present in an image. + Default is none, which will automatically calculate image DPI so that + the produced PDF documents are A4 sized. -PAPERLESS_OCR_USER_ARG= +PAPERLESS_OCR_USER_ARGS= OCRmyPDF offers many more options. Use this parameter to specify any additional arguments you wish to pass to OCRmyPDF. Since Paperless uses the API of OCRmyPDF, you have to specify these in a format that can be passed to the API. See `the API reference of OCRmyPDF `_ for valid parameters. All command line options are supported, but they - use underscores instead of dashed. + use underscores instead of dashes. .. caution:: @@ -275,9 +401,69 @@ PAPERLESS_OCR_USER_ARG= .. code:: json - {"deskew": true, "optimize": 3, "unpaper_args": "--pre-rotate 90"} - - + {"deskew": true, "optimize": 3, "unpaper_args": "--pre-rotate 90"} + +.. _configuration-tika: + +Tika settings +############# + +Paperless can make use of `Tika `_ and +`Gotenberg `_ for parsing and +converting "Office" documents (such as ".doc", ".xlsx" and ".odt"). If you +wish to use this, you must provide a Tika server and a Gotenberg server, +configure their endpoints, and enable the feature. + +PAPERLESS_TIKA_ENABLED= + Enable (or disable) the Tika parser. + + Defaults to false. + +PAPERLESS_TIKA_ENDPOINT= + Set the endpoint URL were Paperless can reach your Tika server. + + Defaults to "http://localhost:9998". + +PAPERLESS_TIKA_GOTENBERG_ENDPOINT= + Set the endpoint URL were Paperless can reach your Gotenberg server. + + Defaults to "http://localhost:3000". + +If you run paperless on docker, you can add those services to the docker-compose +file (see the provided ``docker-compose.tika.yml`` file for reference). The changes +requires are as follows: + +.. code:: yaml + + services: + # ... + + webserver: + # ... + + environment: + # ... + + PAPERLESS_TIKA_ENABLED: 1 + PAPERLESS_TIKA_GOTENBERG_ENDPOINT: http://gotenberg:3000 + PAPERLESS_TIKA_ENDPOINT: http://tika:9998 + + # ... + + gotenberg: + image: thecodingmachine/gotenberg + restart: unless-stopped + environment: + DISABLE_GOOGLE_CHROME: 1 + + tika: + image: apache/tika + restart: unless-stopped + +Add the configuration variables to the environment of the webserver (alternatively +put the configuration in the ``docker-compose.env`` file) and add the additional +services below the webserver service. Watch out for indentation. + Software tweaks ############### @@ -304,8 +490,25 @@ PAPERLESS_THREADS_PER_WORKER= use a higher thread per worker count. The default is a balance between the two, according to your CPU core count, - with a slight favor towards threads per worker, and using as much cores as - possible. + with a slight favor towards threads per worker: + + +----------------+---------+---------+ + | CPU core count | Workers | Threads | + +----------------+---------+---------+ + | 1 | 1 | 1 | + +----------------+---------+---------+ + | 2 | 2 | 1 | + +----------------+---------+---------+ + | 4 | 2 | 2 | + +----------------+---------+---------+ + | 6 | 2 | 3 | + +----------------+---------+---------+ + | 8 | 2 | 4 | + +----------------+---------+---------+ + | 12 | 3 | 4 | + +----------------+---------+---------+ + | 16 | 4 | 4 | + +----------------+---------+---------+ If you only specify PAPERLESS_TASK_WORKERS, paperless will adjust PAPERLESS_THREADS_PER_WORKER automatically. @@ -319,11 +522,14 @@ PAPERLESS_TIME_ZONE= Defaults to UTC. +.. _configuration-polling: + PAPERLESS_CONSUMER_POLLING= If paperless won't find documents added to your consume folder, it might not be able to automatically detect filesystem changes. In that case, specify a polling interval in seconds here, which will then cause paperless - to periodically check your consumption directory for changes. + to periodically check your consumption directory for changes. This will also + disable listening for file system changes with ``inotify``. Defaults to 0, which disables polling and uses filesystem notifications. @@ -348,6 +554,9 @@ PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS= E.g. /foo/bar/file.pdf will add the tags "foo" and "bar" to the consumed file. Paperless will create any tags that don't exist yet. + This is useful for sorting documents with certain tags such as ``car`` or + ``todo`` prior to consumption. These folders won't be deleted. + PAPERLESS_CONSUMER_RECURSIVE must be enabled for this to work. Defaults to false. @@ -400,11 +609,42 @@ PAPERLESS_FILENAME_DATE_ORDER= Defaults to none, which disables this feature. -PAPERLESS_FILENAME_PARSE_TRANSFORMS - Transforms filenames before they are processed by paperless. See - :ref:`advanced-transforming_filenames` for details. +PAPERLESS_THUMBNAIL_FONT_NAME= + Paperless creates thumbnails for plain text files by rendering the content + of the file on an image and uses a predefined font for that. This + font can be changed here. - Defaults to none, which disables this feature. + Note that this won't have any effect on already generated thumbnails. + + Defaults to ``/usr/share/fonts/liberation/LiberationSerif-Regular.ttf``. + +PAPERLESS_IGNORE_DATES= + Paperless parses a documents creation date from filename and file content. + You may specify a comma separated list of dates that should be ignored during + this process. This is useful for special dates (like date of birth) that appear + in documents regularly but are very unlikely to be the documents creation date. + + You may specify dates in a multitude of formats supported by dateparser (see + https://dateparser.readthedocs.io/en/latest/#popular-formats) but as the dates + need to be comma separated, the options are limited. + Example: "2020-12-02,22.04.1999" + + Defaults to an empty string to not ignore any dates. + +PAPERLESS_DATE_ORDER= + Paperless will try to determine the document creation date from its contents. + Specify the date format Paperless should expect to see within your documents. + + This option defaults to DMY which translates to day first, month second, and year + last order. Characters D, M, or Y can be shuffled to meet the required order. + +PAPERLESS_CONSUMER_IGNORE_PATTERNS= + By default, paperless ignores certain files and folders in the consumption + directory, such as system files created by the Mac OS. + + This can be adjusted by configuring a custom json array with patterns to exclude. + + Defautls to ``[".DS_STORE/*", "._*", ".stfolder/*"]``. Binaries ######## @@ -424,3 +664,65 @@ PAPERLESS_GS_BINARY= PAPERLESS_OPTIPNG_BINARY= Defaults to "/usr/bin/optipng". + + +.. _configuration-docker: + +Docker-specific options +####################### + +These options don't have any effect in ``paperless.conf``. These options adjust +the behavior of the docker container. Configure these in `docker-compose.env`. + +PAPERLESS_WEBSERVER_WORKERS= + The number of worker processes the webserver should spawn. More worker processes + usually result in the front end to load data much quicker. However, each worker process + also loads the entire application into memory separately, so increasing this value + will increase RAM usage. + + Consider configuring this to 1 on low power devices with limited amount of RAM. + + Defaults to 2. + +USERMAP_UID= + The ID of the paperless user in the container. Set this to your actual user ID on the + host system, which you can get by executing + + .. code:: shell-session + + $ id -u + + Paperless will change ownership on its folders to this user, so you need to get this right + in order to be able to write to the consumption directory. + + Defaults to 1000. + +USERMAP_GID= + The ID of the paperless Group in the container. Set this to your actual group ID on the + host system, which you can get by executing + + .. code:: shell-session + + $ id -g + + Paperless will change ownership on its folders to this group, so you need to get this right + in order to be able to write to the consumption directory. + + Defaults to 1000. + +PAPERLESS_OCR_LANGUAGES= + Additional OCR languages to install. By default, paperless comes with + English, German, Italian, Spanish and French. If your language is not in this list, install + additional languages with this configuration option: + + .. code:: bash + + PAPERLESS_OCR_LANGUAGES=tur ces + + To actually use these languages, also set the default OCR language of paperless: + + .. code:: bash + + PAPERLESS_OCR_LANGUAGE=tur + + Defaults to none, which does not install any additional languages. diff --git a/docs/contributing.rst b/docs/contributing.rst index 30eb9779a..ec1a20921 100644 --- a/docs/contributing.rst +++ b/docs/contributing.rst @@ -6,7 +6,7 @@ Contributing to Paperless .. warning:: This section is not updated to paperless-ng yet. - + Maybe you've been using Paperless for a while and want to add a feature or two, or maybe you've come across a bug that you have some ideas how to solve. The beauty of Free software is that you can see what's wrong and help to get it diff --git a/docs/extending.rst b/docs/extending.rst index 28da1f56b..11f340b27 100644 --- a/docs/extending.rst +++ b/docs/extending.rst @@ -5,28 +5,82 @@ Paperless development This section describes the steps you need to take to start development on paperless-ng. -1. Check out the source from github. The repository is organized in the following way: +Check out the source from github. The repository is organized in the following way: - * ``master`` always represents the latest release and will only see changes - when a new release is made. - * ``dev`` contains the code that will be in the next release. - * ``feature-X`` contain bigger changes that will be in some release, but not - necessarily the next one. - - Apart from that, the folder structure is as follows: +* ``master`` always represents the latest release and will only see changes + when a new release is made. +* ``dev`` contains the code that will be in the next release. +* ``feature-X`` contain bigger changes that will be in some release, but not + necessarily the next one. - * ``docs/`` - Documentation. - * ``src-ui/`` - Code of the front end. - * ``src/`` - Code of the back end. - * ``scripts/`` - Various scripts that help with different parts of development. - * ``docker/`` - Files required to build the docker image. +When making functional changes to paperless, *always* make your changes on the ``dev`` branch. -2. Install some dependencies. +Apart from that, the folder structure is as follows: - * Python 3.6. - * All dependencies listed in the :ref:`Bare metal route ` - * redis. You can either install redis or use the included scritps/start-redis.sh - to use docker to fire up a redis instance. +* ``docs/`` - Documentation. +* ``src-ui/`` - Code of the front end. +* ``src/`` - Code of the back end. +* ``scripts/`` - Various scripts that help with different parts of development. +* ``docker/`` - Files required to build the docker image. + +Initial setup and first start +============================= + +After you forked and cloned the code from github you need to perform a first-time setup. +To do the setup you need to perform the steps from the following chapters in a certain order: + +1. Install prerequisites + pipenv as mentioned in :ref:`Bare metal route ` +2. Copy ``paperless.conf.example`` to ``paperless.conf`` and enable debug mode. +3. Install the Angular CLI interface: + + .. code:: shell-session + + $ npm install -g @angular/cli + +4. Create ``consume`` and ``media`` folders in the cloned root folder. + + .. code:: shell-session + + mkdir -p consume media + +5. You can now either ... + + * install redis or + * use the included scripts/start-services.sh to use docker to fire up a redis instance (and some other services such as tika, gotenberg and a postgresql server) or + * spin up a bare redis container + + .. code:: shell-session + + docker run -d -p 6379:6379 -restart unless-stopped redis:latest + +6. Install the python dependencies by performing in the src/ directory. + + .. code:: shell-session + + pipenv install --dev + +7. Generate the static UI so you can perform a login to get session that is required for frontend development (this needs to be done one time only). From root folder: + + .. code:: shell-session + + compile-frontend.sh + +8. Apply migrations and create a superuser for your dev instance: + + .. code:: shell-session + + python3 manage.py migrate + python3 manage.py createsuperuser + +9. Now spin up the dev backend. Depending on which part of paperless you're developing for, you need to have some or all of them running. + + .. code:: shell-session + + python3 manage.py runserver & python3 manage.py document_consumer & python3 manage.py qcluster + +10. Login with the superuser credentials provided in step 8 at ``http://localhost:8000`` to create a session that enables you to use the backend. + +Backend development environment is now ready, to start Frontend development go to ``/src-ui`` and run ``ng serve``. From there you can use ``http://localhost:4200`` for a preview. Back end development ==================== @@ -34,21 +88,18 @@ Back end development The backend is a django application. I use PyCharm for development, but you can use whatever you want. -Install the python dependencies by performing ``pipenv install --dev`` in the src/ directory. -This will also create a virtual environment, which you can enter with ``pipenv shell`` or -execute one-shot commands in with ``pipenv run``. - -In ``src/paperless.conf``, enable debug mode. - Configure the IDE to use the src/ folder as the base source folder. Configure the following launch configurations in your IDE: * python3 manage.py runserver * python3 manage.py qcluster -* python3 manage.py consumer +* python3 manage.py document_consumer -Depending on which part of paperless you're developing for, you need to have some or all of -them running. +To start them all: + +.. code:: shell-session + + python3 manage.py runserver & python3 manage.py document_consumer & python3 manage.py qcluster Testing and code style: @@ -61,7 +112,7 @@ Testing and code style: The line length rule E501 is generally useful for getting multiple source files next to each other on the screen. However, in some cases, its just not possible - to make some lines fit, especially complicated IF cases. Append `` # NOQA: E501`` + to make some lines fit, especially complicated IF cases. Append ``# NOQA: E501`` to disable this check for certain lines. Front end development @@ -102,18 +153,138 @@ In order to build the front end and serve it as part of django, execute .. code:: shell-session - $ ng build --prod --output-path ../src/documents/static/frontend/ + $ ng build --prod This will build the front end and put it in a location from which the Django server will serve it as static content. This way, you can verify that authentication is working. -Making a release -================ -Execute the ``make-release.sh `` script. +Localization +============ -This will test and assemble everything and also build and tag a docker image. +Paperless is available in many different languages. Since paperless consists both of a django +application and an Angular front end, both these parts have to be translated separately. +Front end localization +---------------------- + +* The Angular front end does localization according to the `Angular documentation `_. +* The source language of the project is "en_US". +* The source strings end up in the file "src-ui/messages.xlf". +* The translated strings need to be placed in the "src-ui/src/locale/" folder. +* In order to extract added or changed strings from the source files, call ``ng xi18n --ivy``. + +Adding new languages requires adding the translated files in the "src-ui/src/locale/" folder and adjusting a couple files. + +1. Adjust "src-ui/angular.json": + + .. code:: json + + "i18n": { + "sourceLocale": "en-US", + "locales": { + "de": "src/locale/messages.de.xlf", + "nl-NL": "src/locale/messages.nl_NL.xlf", + "fr": "src/locale/messages.fr.xlf", + "en-GB": "src/locale/messages.en_GB.xlf", + "pt-BR": "src/locale/messages.pt_BR.xlf", + "language-code": "language-file" + } + } + +2. Add the language to the available options in "src-ui/src/app/services/settings.service.ts": + + .. code:: typescript + + getLanguageOptions(): LanguageOption[] { + return [ + {code: "en-us", name: $localize`English (US)`, englishName: "English (US)", dateInputFormat: "mm/dd/yyyy"}, + {code: "en-gb", name: $localize`English (GB)`, englishName: "English (GB)", dateInputFormat: "dd/mm/yyyy"}, + {code: "de", name: $localize`German`, englishName: "German", dateInputFormat: "dd.mm.yyyy"}, + {code: "nl", name: $localize`Dutch`, englishName: "Dutch", dateInputFormat: "dd-mm-yyyy"}, + {code: "fr", name: $localize`French`, englishName: "French", dateInputFormat: "dd/mm/yyyy"}, + {code: "pt-br", name: $localize`Portuguese (Brazil)`, englishName: "Portuguese (Brazil)", dateInputFormat: "dd/mm/yyyy"} + // Add your new language here + ] + } + + ``dateInputFormat`` is a special string that defines the behavior of the date input fields and absolutely needs to contain "dd", "mm" and "yyyy". + +3. Import and register the Angular data for this locale in "src-ui/src/app/app.module.ts": + + .. code:: typescript + + import localeDe from '@angular/common/locales/de'; + registerLocaleData(localeDe) + +Back end localization +--------------------- + +A majority of the strings that appear in the back end appear only when the admin is used. However, +some of these are still shown on the front end (such as error messages). + +* The django application does localization according to the `django documentation `_. +* The source language of the project is "en_US". +* Localization files end up in the folder "src/locale/". +* In order to extract strings from the application, call ``python3 manage.py makemessages -l en_US``. This is important after making changes to translatable strings. +* The message files need to be compiled for them to show up in the application. Call ``python3 manage.py compilemessages`` to do this. The generated files don't get + committed into git, since these are derived artifacts. The build pipeline takes care of executing this command. + +Adding new languages requires adding the translated files in the "src/locale/" folder and adjusting the file "src/paperless/settings.py" to include the new language: + +.. code:: python + + LANGUAGES = [ + ("en-us", _("English (US)")), + ("en-gb", _("English (GB)")), + ("de", _("German")), + ("nl-nl", _("Dutch")), + ("fr", _("French")), + ("pt-br", _("Portuguese (Brazil)")), + # Add language here. + ] + + +Building the documentation +========================== + +The documentation is built using sphinx. I've configured ReadTheDocs to automatically build +the documentation when changes are pushed. If you want to build the documentation locally, +this is how you do it: + +1. Install python dependencies. + + .. code:: shell-session + + $ cd /path/to/paperless + $ pipenv install --dev + +2. Build the documentation + + .. code:: shell-session + + $ cd /path/to/paperless/docs + $ pipenv run make clean html + +This will build the HTML documentation, and put the resulting files in the ``_build/html`` +directory. + +Building the Docker image +========================= + +Building the docker image from source requires the following two steps: + +1. Build the front end. + + .. code:: shell-session + + ./compile-frontend.sh + +2. Build the docker image. + + .. code:: shell-session + + docker build . -t Extending Paperless =================== @@ -149,7 +320,7 @@ methods ``parse`` and ``get_thumbnail``. You can provide your own implementation # The content of the document. self.text = "content" - + # Optional: path to a PDF document that you created from the original. self.archive_path = os.path.join(self.tempdir, "archived.pdf") diff --git a/docs/faq.rst b/docs/faq.rst index 6eac18617..74b32cf5c 100644 --- a/docs/faq.rst +++ b/docs/faq.rst @@ -52,6 +52,8 @@ out of that folder to use them elsewhere. Here are a couple notes about that. * PDF documents, PNG images, JPEG images, TIFF images and GIF images are processed with OCR and converted into PDF documents. * Plain text documents are supported as well and are added verbatim to paperless. +* With the optional Tika integration enabled (see :ref:`Configuration `), Paperless also supports various + Office documents (.docx, .doc, odt, .ppt, .pptx, .odp, .xls, .xlsx, .ods). Paperless determines the type of a file by inspecting its content. The file extensions do not matter. @@ -60,23 +62,34 @@ file extensions do not matter. **A:** The short answer is yes. I've tested it on a Raspberry Pi 3 B. The long answer is that certain parts of -Paperless will run very slow, such as the tesseract OCR. On Raspberry Pi, +Paperless will run very slow, such as the OCR. On Raspberry Pi, try to OCR documents before feeding them into paperless so that paperless can -reuse the text. The web interface should be a lot snappier, since it runs +reuse the text. The web interface is a lot snappier, since it runs in your browser and paperless has to do much less work to serve the data. .. note:: - + You can adjust some of the settings so that paperless uses less processing power. See :ref:`setup-less_powerful_devices` for details. - + **Q:** *How do I install paperless-ng on Raspberry Pi?* -**A:** There is no docker image for ARM available. If you know how to build -that automatically, I'm all ears. For now, you have to grab the latest release -archive from the project page and build the image yourself. The release comes -with the front end already compiled, so you don't have to do this on the Pi. +**A:** Docker images are available for arm and arm64 hardware, so just follow +the docker-compose instructions. Apart from more required disk space compared to +a bare metal installation, docker comes with close to zero overhead, even on +Raspberry Pi. + +If you decide to got with the bare metal route, be aware that some of the +python requirements do not have precompiled packages for ARM / ARM64. Installation +of these will require additional development libraries and compilation will take +a long time. + +**Q:** *How do I run this on unRaid?* + +**A:** Head over to ``_, +`Uli Fahrer `_ created a container template for that. +I don't exactly know how to use that though, since I don't use unRaid. **Q:** *How do I run this on my toaster?* @@ -89,12 +102,21 @@ occasionally build the image on and see if it works. **Q:** *How do I proxy this with NGINX?* -.. code:: +**A:** See :ref:`here `. - location / { - proxy_pass http://localhost:8000/ - } +.. _faq-mod_wsgi: -And that's about it. Paperless serves everything, including static files by itself -when running the docker image. If you want to do anything fancy, you have to -install paperless bare metal. +**Q:** *How do I get WebSocket support with Apache mod_wsgi*? + +**A:** ``mod_wsgi`` by itself does not support ASGI. Paperless will continue +to work with WSGI, but certain features such as status notifications about +document consumption won't be available. + +If you want to continue using ``mod_wsgi``, you will have to run an ASGI-enabled +web server as well that processes WebSocket connections, and configure Apache to +redirect WebSocket connections to this server. Multiple options for ASGI servers +exist: + +* ``gunicorn`` with ``uvicorn`` as the worker implementation (the default of paperless) +* ``daphne`` as a standalone server, which is the reference implementation for ASGI. +* ``uvicorn`` as a standalone server diff --git a/docs/index.rst b/docs/index.rst index a083fb3d1..f30895508 100644 --- a/docs/index.rst +++ b/docs/index.rst @@ -49,7 +49,7 @@ resources in the documentation: paperless-ng. * Paperless is now integrated with a :ref:`task processing queue ` that tells you - at a glance when and why something is not working. + at a glance when and why something is not working. * The :ref:`changelog ` contains a detailed list of all changes in paperless-ng. @@ -70,8 +70,8 @@ Contents configuration api faq - extending troubleshooting + extending contributing scanners screenshots diff --git a/src-ui/src/app/components/common/delete-dialog/delete-dialog.component.scss b/docs/requirements.txt similarity index 100% rename from src-ui/src/app/components/common/delete-dialog/delete-dialog.component.scss rename to docs/requirements.txt diff --git a/docs/scanners.rst b/docs/scanners.rst index d4ad4dfb1..f29cb0328 100644 --- a/docs/scanners.rst +++ b/docs/scanners.rst @@ -10,34 +10,91 @@ scanner you use, but sometimes finding a scanner that will write to an FTP, NFS, or SMB server can be difficult. This page is here to help you find one that works right for you based on recommendations from other Paperless users. -+---------+----------------+-----+-----+-----+----------------+ -| Brand | Model | Supports | Recommended By | -+---------+----------------+-----+-----+-----+----------------+ -| | | FTP | NFS | SMB | | -+=========+================+=====+=====+=====+================+ -| Brother | `ADS-1500W`_ | yes | no | yes | `danielquinn`_ | -+---------+----------------+-----+-----+-----+----------------+ -| Brother | `MFC-J6930DW`_ | yes | | | `ayounggun`_ | -+---------+----------------+-----+-----+-----+----------------+ -| Brother | `MFC-J5910DW`_ | yes | | | `bmsleight`_ | -+---------+----------------+-----+-----+-----+----------------+ -| Brother | `MFC-9142CDN`_ | yes | | yes | `REOLDEV`_ | -+---------+----------------+-----+-----+-----+----------------+ -| Fujitsu | `ix500`_ | yes | | yes | `eonist`_ | -+---------+----------------+-----+-----+-----+----------------+ -| Fujitsu | `S1300i`_ | yes | | yes | `jonaswinkler`_| -+---------+----------------+-----+-----+-----+----------------+ +Physical scanners +================= ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brand | Model | Supports | Recommended By | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| | | FTP | NFS | SMB | SMTP | API [1]_ | | ++=========+================+=====+=====+=====+======+==========+================+ +| Brother | `ADS-1700W`_ | yes | no | yes | yes | |`holzhannes`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `ADS-1600W`_ | yes | no | yes | yes | |`holzhannes`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `ADS-1500W`_ | yes | no | yes | yes | |`danielquinn`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `ADS-1100W`_ | yes | no | no | no | |`ytzelf`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `MFC-J6930DW`_ | yes | | | | |`ayounggun`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `MFC-L5850DW`_ | yes | | | yes | |`holzhannes`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `MFC-J5910DW`_ | yes | | | | |`bmsleight`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Brother | `MFC-9142CDN`_ | yes | | yes | | |`REOLDEV`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Fujitsu | `ix500`_ | yes | | yes | | |`eonist`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Epson | `WF-7710DWF`_ | yes | | yes | | |`Skylinar`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Fujitsu | `S1300i`_ | yes | | yes | | |`jonaswinkler`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ +| Doxie | `Q2`_ | no | no | no | no | yes |`Unkn0wnCat`_ | ++---------+----------------+-----+-----+-----+------+----------+----------------+ + +.. _MFC-L5850DW: https://www.brother-usa.com/products/mfcl5850dw +.. _ADS-1700W: https://www.brother-usa.com/products/ads1700w +.. _ADS-1600W: https://www.brother-usa.com/products/ads1600w .. _ADS-1500W: https://www.brother.ca/en/p/ads1500w +.. _ADS-1100W: https://support.brother.com/g/b/downloadtop.aspx?c=fr&lang=fr&prod=ads1100w_eu_as_cn .. _MFC-J6930DW: https://www.brother.ca/en/p/MFCJ6930DW .. _MFC-J5910DW: https://www.brother.co.uk/printers/inkjet-printers/mfcj5910dw .. _MFC-9142CDN: https://www.brother.co.uk/printers/laser-printers/mfc9140cdn -.. _ix500: https://www.fujitsu.com/global/products/computing/peripheral/scanners/scansnap/ix500/ +.. _ix500: http://www.fujitsu.com/us/products/computing/peripheral/scanners/scansnap/ix500/ +.. _WF-7710DWF: https://www.epson.de/en/products/printers/inkjet-printers/for-home/workforce-wf-7710dwf .. _S1300i: https://www.fujitsu.com/global/products/computing/peripheral/scanners/soho/s1300i/ +.. _Q2: https://www.getdoxie.com/product/doxie-q/ + .. _danielquinn: https://github.com/danielquinn .. _ayounggun: https://github.com/ayounggun .. _bmsleight: https://github.com/bmsleight .. _eonist: https://github.com/eonist .. _REOLDEV: https://github.com/REOLDEV +.. _Skylinar: https://github.com/Skylinar .. _jonaswinkler: https://github.com/jonaswinkler +.. _holzhannes: https://github.com/holzhannes +.. _ytzelf: https://github.com/ytzelf +.. _Unkn0wnCat: https://github.com/Unkn0wnCat + +.. [1] Scanners with API Integration allow to push scanned documents directly to :ref:`Paperless API `, sometimes referred to as Webhook or Document POST. + +Mobile phone software +===================== + +You can use your phone to "scan" documents. The regular camera app will work, but may have too low contrast for OCR to work well. Apps specifically for scanning are recommended. + ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ +| Name | OS | Supports | Recommended By | ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ +| | | FTP | NFS | SMB | Email | WebDav | | ++===================+================+=====+=====+=====+=======+========+==================+ +| `Office Lens`_ | Android | ? | ? | ? | ? | ? | `jonaswinkler`_ | ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ +| `Genius Scan`_ | Android | yes | no | yes | yes | yes | `hannahswain`_ | ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ +| `OpenScan`_ | Android | no | no | no | no | no | `benjaminfrank`_ | ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ +| `Quick Scan`_ | iOS | no | no | no | no | no | `holzhannes`_ | ++-------------------+----------------+-----+-----+-----+-------+--------+------------------+ + +On Android, you can use these applications in combination with one of the :ref:`Paperless-ng compatible apps ` to "Share" the documents produced by these scanner apps with paperless. On iOS, you can share the scanned documents via iOS-Sharing to other mail, WebDav or FTP apps. + +.. _Office Lens: https://play.google.com/store/apps/details?id=com.microsoft.office.officelens +.. _Genius Scan: https://play.google.com/store/apps/details?id=com.thegrizzlylabs.geniusscan.free +.. _Quick Scan: https://apps.apple.com/us/app/quickscan-scanner-text-ocr/id1513790291 +.. _OpenScan: https://github.com/Ethereal-Developers-Inc/OpenScan + +.. _hannahswain: https://github.com/hannahswain +.. _benjaminfrank: https://github.com/benjaminfrank diff --git a/docs/setup.rst b/docs/setup.rst index e5e6526ea..99c910cf2 100644 --- a/docs/setup.rst +++ b/docs/setup.rst @@ -3,35 +3,6 @@ Setup ***** -Download -######## - -Go to the project page on GitHub and download the -`latest release `_. -There are multiple options available. - -* Download the dockerfiles archive if you want to pull paperless from - Docker Hub. - -* Download the dist archive and extract it if you want to build the docker image - yourself or want to install paperless without docker. - -.. hint:: - - In contrast to paperless, the recommended way to get and update paperless-ng - is not to pull the entire git repository. Paperless-ng includes artifacts - that need to be compiled, and that's already done for you in the release. - -.. admonition:: Want to try out paperless-ng before migrating? - - The release contains a file ``.env`` which sets the docker-compose project - name to "paperless", which is the same as before and instructs docker-compose - to reuse and upgrade your paperless volumes. - - Just rename the project name in that file to anything else and docker-compose - will create fresh volumes for you! - - Overview of Paperless-ng ######################## @@ -49,45 +20,45 @@ Paperless consists of the following components: .. code:: shell-session $ cd /path/to/paperless/src/ - $ pipenv run gunicorn -c /usr/src/paperless/gunicorn.conf.py -b 0.0.0.0:8000 paperless.wsgi + $ gunicorn -c ../gunicorn.conf.py paperless.wsgi or by any other means such as Apache ``mod_wsgi``. * **The consumer:** This is what watches your consumption folder for documents. - However, the consumer itself does not consume really consume your documents anymore. - It rather notifies a task processor that a new file is ready for consumption. + However, the consumer itself does not really consume your documents. + Now it notifies a task processor that a new file is ready for consumption. I suppose it should be named differently. - This also used to check your emails, but that's now gone elsewhere as well. + This was also used to check your emails, but that's now done elsewhere as well. Start the consumer with the management command ``document_consumer``: .. code:: shell-session $ cd /path/to/paperless/src/ - $ pipenv run python3 manage.py document_consumer + $ python3 manage.py document_consumer .. _setup-task_processor: * **The task processor:** Paperless relies on `Django Q `_ - for doing much of the heavy lifting. This is a task queue that accepts tasks from - multiple sources and processes tasks in parallel. It also comes with a scheduler that executes + for doing most of the heavy lifting. This is a task queue that accepts tasks from + multiple sources and processes these in parallel. It also comes with a scheduler that executes certain commands periodically. This task processor is responsible for: * Consuming documents. When the consumer finds new documents, it notifies the task processor to start a consumption task. - * Consuming emails. It periodically checks your configured accounts for new mails and - produces consumption tasks for any documents it finds. * The task processor also performs the consumption of any documents you upload through the web interface. - * Maintain the search index and the automatic matching algorithm. These are things that paperless + * Consuming emails. It periodically checks your configured accounts for new emails and + notifies the task processor to consume the attachment of an email. + * Maintaining the search index and the automatic matching algorithm. These are things that paperless needs to do from time to time in order to operate properly. This allows paperless to process multiple documents from your consumption folder in parallel! On - a modern multi core system, consumption with full ocr is blazing fast. + a modern multi core system, this makes the consumption process with full OCR blazingly fast. - The task processor comes with a built-in admin interface that you can use to see whenever any of the + The task processor comes with a built-in admin interface that you can use to check whenever any of the tasks fail and inspect the errors (i.e., wrong email credentials, errors during consuming a specific file, etc). @@ -96,11 +67,11 @@ Paperless consists of the following components: .. code:: shell-session $ cd /path/to/paperless/src/ - $ pipenv run python3 manage.py qcluster + $ python3 manage.py qcluster * A `redis `_ message broker: This is a really lightweight service that is responsible - for getting the tasks from the webserver and consumer to the task scheduler. These run in different - processes (maybe even on different machines!), and therefore, this is necessary. + for getting the tasks from the webserver and the consumer to the task scheduler. These run in a different + process (maybe even on different machines!), and therefore, this is necessary. * Optional: A database server. Paperless supports both PostgreSQL and SQLite for storing its data. @@ -108,48 +79,84 @@ Paperless consists of the following components: Installation ############ -You can go multiple routes with setting up and running Paperless: +You can go multiple routes to setup and run Paperless: -* The `docker route`_ -* The `bare metal route`_ +* :ref:`Use the easy install docker script ` +* :ref:`Pull the image from Docker Hub ` +* :ref:`Build the Docker image yourself ` +* :ref:`Install Paperless directly on your system manually (bare metal) ` +* :ref:`Use ansible to install Paperless on your system automatically (bare metal) ` -The `docker route`_ is quick & easy. This is the recommended route. This configures all the stuff -from above automatically so that it just works and uses sensible defaults for all configuration options. +The Docker routes are quick & easy. These are the recommended routes. This configures all the stuff +from the above automatically so that it just works and uses sensible defaults for all configuration options. +Here you find a cheat-sheet for docker beginners: `CLI Basics `_ -The `bare metal route`_ is more complicated to setup but makes it easier +The bare metal route is complicated to setup but makes it easier should you want to contribute some code back. You need to configure and run the above mentioned components yourself. -Docker Route -============ +The ansible route combines benefits of both options: +the setup process is fully automated, reproducible and `idempotent `_, +it includes the same sensible defaults, and it simultaneously provides the flexibility of a bare metal installation. -1. Install `Docker`_ and `docker-compose`_. [#compose]_ +.. _CLI Basics: https://sehn.tech/post/devops-with-docker/ +.. _idempotent: https://docs.ansible.com/ansible/latest/reference_appendices/glossary.html#Idempotency - .. caution:: +.. _setup-docker_script: - If you want to use the included ``docker-compose.*.yml`` file, you - need to have at least Docker version **17.09.0** and docker-compose - version **1.17.0**. +Install Paperless from Docker Hub using the installation script +=============================================================== - See the `Docker installation guide`_ on how to install the current - version of Docker for your operating system or Linux distribution of - choice. To get an up-to-date version of docker-compose, follow the - `docker-compose installation guide`_ if your package repository doesn't - include it. +Paperless provides an interactive installation script. This script will ask you +for a couple configuration options, download and create the necessary configuration files, pull the docker image, start paperless and create your user account. This script essentially +performs all the steps described in :ref:`setup-docker_hub` automatically. - .. _Docker installation guide: https://docs.docker.com/engine/installation/ - .. _docker-compose installation guide: https://docs.docker.com/compose/install/ +1. Make sure that docker and docker-compose are installed. +2. Download and run the installation script: -2. Copy either ``docker-compose.sqlite.yml`` or ``docker-compose.postgres.yml`` to - ``docker-compose.yml``, depending on which database backend you want to use. + .. code:: shell-session + + $ curl -L https://raw.githubusercontent.com/jonaswinkler/paperless-ng/master/install-paperless-ng.sh | bash + +.. _setup-docker_hub: + +Install Paperless from Docker Hub +================================= + +1. Login with your user and create a folder in your home-directory `mkdir -v ~/paperless-ng` to have a place for your configuration files and consumption directory. + +2. Go to the `/docker/compose directory on the project page `_ + and download one of the `docker-compose.*.yml` files, depending on which database backend you + want to use. Rename this file to `docker-compose.yml`. + If you want to enable optional support for Office documents, download a file with `-tika` in the file name. + Download the ``docker-compose.env`` file and the ``.env`` file as well and store them + in the same directory. .. hint:: For new installations, it is recommended to use PostgreSQL as the database backend. -2. Modify ``docker-compose.yml`` to your preferences. You may want to change the path - to the consumption directory in this file. Find the line that specifies where +3. Install `Docker`_ and `docker-compose`_. + + .. caution:: + + If you want to use the included ``docker-compose.*.yml`` file, you + need to have at least Docker version **17.09.0** and docker-compose + version **1.17.0**. + To check do: `docker-compose -v` or `docker -v` + + See the `Docker installation guide`_ on how to install the current + version of Docker for your operating system or Linux distribution of + choice. To get the latest version of docker-compose, follow the + `docker-compose installation guide`_ if your package repository doesn't + include it. + + .. _Docker installation guide: https://docs.docker.com/engine/installation/ + .. _docker-compose installation guide: https://docs.docker.com/compose/install/ + +4. Modify ``docker-compose.yml`` to your preferences. You may want to change the path + to the consumption directory. Find the line that specifies where to mount the consumption directory: .. code:: @@ -164,25 +171,53 @@ Docker Route Don't change the part after the colon or paperless wont find your documents. + You may also need to change the default port that the webserver will use + from the default (8000): -3. Modify ``docker-compose.env``, following the comments in the file. The + .. code:: + + ports: + - 8000:8000 + + Replace the part BEFORE the colon with a port of your choice: + + .. code:: + + ports: + - 8010:8000 + + Don't change the part after the colon or edit other lines that refer to + port 8000. Modifying the part before the colon will map requests on another + port to the webserver running on the default port. + +5. Modify ``docker-compose.env``, following the comments in the file. The most important change is to set ``USERMAP_UID`` and ``USERMAP_GID`` - to the uid and gid of your user on the host system. This ensures that + to the uid and gid of your user on the host system. Use ``id -u`` and + ``id -g`` to get these. + + This ensures that both the docker container and you on the host machine have write access to the consumption directory. If your UID and GID on the host system is 1000 (the default for the first normal user on most systems), it will - work out of the box without any modifications. + work out of the box without any modifications. `id "username"` to check. .. note:: - You can use any settings from the file ``paperless.conf`` in this file. - Have a look at :ref:`configuration` to see whats available. + You can copy any setting from the file ``paperless.conf.example`` and paste it here. + Have a look at :ref:`configuration` to see what's available. -4. Run ``docker-compose up -d``. This will create and start the necessary - containers. This will also build the image of paperless if you grabbed the - source archive. + .. caution:: -5. To be able to login, you will need a super user. To create it, execute the + Some file systems such as NFS network shares don't support file system + notifications with ``inotify``. When storing the consumption directory + on such a file system, paperless will not pick up new files + with the default configuration. You will need to use ``PAPERLESS_CONSUMER_POLLING``, + which will disable inotify. See :ref:`here `. + +6. Run ``docker-compose pull``, followed by ``docker-compose up -d``. + This will pull the image, create and start the necessary containers. + +7. To be able to login, you will need a super user. To create it, execute the following command: .. code-block:: shell-session @@ -190,21 +225,59 @@ Docker Route $ docker-compose run --rm webserver createsuperuser This will prompt you to set a username, an optional e-mail address and - finally a password. + finally a password (at least 8 characters). -6. The default ``docker-compose.yml`` exports the webserver on your local port - 8000. If you haven't adapted this, you should now be able to visit your - Paperless instance at ``http://127.0.0.1:8000``. You can login with the - user and password you just created. +8. The default ``docker-compose.yml`` exports the webserver on your local port + 8000. If you did not change this, you should now be able to visit your + Paperless instance at ``http://127.0.0.1:8000`` or your servers IP-Address:8000. + Use the login credentials you have created with the previous step. .. _Docker: https://www.docker.com/ .. _docker-compose: https://docs.docker.com/compose/install/ -.. [#compose] You of course don't have to use docker-compose, but it - simplifies deployment immensely. If you know your way around Docker, feel - free to tinker around without using compose! +.. _setup-docker_build: -.. _`setup-bare_metal`: +Build the Docker image yourself +=============================== + +1. Clone the entire repository of paperless: + + .. code:: shell-session + + git clone https://github.com/jonaswinkler/paperless-ng + + The master branch always reflects the latest stable version. + +2. Copy one of the ``docker/compose/docker-compose.*.yml`` to ``docker-compose.yml`` in the root folder, + depending on which database backend you want to use. Copy + ``docker-compose.env`` into the project root as well. + +3. In the ``docker-compose.yml`` file, find the line that instructs docker-compose to pull the paperless image from Docker Hub: + + .. code:: yaml + + webserver: + image: jonaswinkler/paperless-ng:latest + + and replace it with a line that instructs docker-compose to build the image from the current working directory instead: + + .. code:: yaml + + webserver: + build: . + +4. Run the ``compile-frontend.sh`` script. This requires ``node`` and ``npm >= v15``. + +5. Follow steps 3 to 8 of :ref:`setup-docker_hub`. When asked to run + ``docker-compose pull`` to pull the image, do + + .. code:: shell-session + + $ docker-compose build + + instead to build the image. + +.. _setup-bare_metal: Bare Metal Route ================ @@ -215,16 +288,23 @@ writing. Windows is not and will never be supported. 1. Install dependencies. Paperless requires the following packages. - * ``python3`` 3.6, 3.7, 3.8 (3.9 is untested). - * ``python3-pip``, optionally ``pipenv`` for package installation + * ``python3`` 3.6, 3.7, 3.8, 3.9 + * ``python3-pip`` * ``python3-dev`` + * ``fonts-liberation`` for generating thumbnails for plain text files * ``imagemagick`` >= 6 for PDF conversion - * ``optipng`` for optimising thumbnails + * ``optipng`` for optimizing thumbnails * ``gnupg`` for handling encrypted documents - * ``libpoppler-cpp-dev`` for PDF to text conversion - * ``libmagic-dev`` for mime type detection * ``libpq-dev`` for PostgreSQL + * ``libmagic-dev`` for mime type detection + * ``mime-support`` for mime type detection + + Use this list for your preferred package management: + + .. code:: + + python3 python3-pip python3-dev imagemagick fonts-liberation optipng gnupg libpq-dev libmagic-dev mime-support These dependencies are required for OCRmyPDF, which is used for text recognition. @@ -239,17 +319,28 @@ writing. Windows is not and will never be supported. * ``tesseract-ocr`` >= 4.0.0 for OCR * ``tesseract-ocr`` language packs (``tesseract-ocr-eng``, ``tesseract-ocr-deu``, etc) + Use this list for your preferred package management: + + .. code:: + + unpaper ghostscript icc-profiles-free qpdf liblept5 libxml2 pngquant zlib1g tesseract-ocr + + On Raspberry Pi, these libraries are required as well: + + * ``libatlas-base-dev`` + * ``libxslt1-dev`` + You will also need ``build-essential``, ``python3-setuptools`` and ``python3-wheel`` - for installing some of the python dependencies. You can remove that - again after installation. + for installing some of the python dependencies. 2. Install ``redis`` >= 5.0 and configure it to start automatically. 3. Optional. Install ``postgresql`` and configure a database, user and password for paperless. If you do not wish - to use PostgreSQL, SQLite is avialable as well. + to use PostgreSQL, SQLite is available as well. -4. Get the release archive. If you pull the git repo as it is, you also have to compile the front end by yourself. - Extract the frontend to a place from where you wish to execute it, such as ``/opt/paperless``. +4. Get the release archive from ``_. + If you clone the git repo as it is, you also have to compile the front end by yourself. + Extract the archive to a place from where you wish to execute it, such as ``/opt/paperless``. 5. Configure paperless. See :ref:`configuration` for details. Edit the included ``paperless.conf`` and adjust the settings to your needs. Required settings for getting paperless running are: @@ -262,58 +353,73 @@ writing. Windows is not and will never be supported. paperless stores its data. If you like, you can point both to the same directory. * ``PAPERLESS_SECRET_KEY`` should be a random sequence of characters. It's used for authentication. Failure to do so allows third parties to forge authentication credentials. - + Many more adjustments can be made to paperless, especially the OCR part. The following options are recommended for everyone: * Set ``PAPERLESS_OCR_LANGUAGE`` to the language most of your documents are written in. * Set ``PAPERLESS_TIME_ZONE`` to your local time zone. -6. Setup permissions. Create a system users under which you wish to run paperless. Ensure that these directories exist - and that the user has write permissions to the following directories - +6. Create a system user under which you wish to run paperless. + + .. code:: shell-session + + adduser paperless --system --home /opt/paperless --group + +7. Ensure that these directories exist + and that the paperless user has write permissions to the following directories: + * ``/opt/paperless/media`` * ``/opt/paperless/data`` * ``/opt/paperless/consume`` Adjust as necessary if you configured different folders. -7. Install python requirements. Paperless comes with both Pipfiles for ``pipenv`` as well as with a ``requirements.txt``. - Both will install exactly the same requirements. It is up to you if you wish to use a virtual environment or not. +8. Install python requirements from the ``requirements.txt`` file. + It is up to you if you wish to use a virtual environment or not. First you should update your pip, so it gets the actual packages. -8. Go to ``/opt/paperless/src``, and execute the following commands: + .. code:: shell-session + + sudo -Hu paperless pip3 install --upgrade pip + + .. code:: shell-session + + sudo -Hu paperless pip3 install -r requirements.txt + + This will install all python dependencies in the home directory of + the new paperless user. + +9. Go to ``/opt/paperless/src``, and execute the following commands: .. code:: bash - # This collects static files from paperless and django. - python3 manage.py collectstatic --clear --no-input - # This creates the database schema. - python3 manage.py migrate + sudo -Hu paperless python3 manage.py migrate # This creates your first paperless user - python3 manage.py createsuperuser + sudo -Hu paperless python3 manage.py createsuperuser -9. Optional: Test that paperless is working by executing +10. Optional: Test that paperless is working by executing .. code:: bash # This collects static files from paperless and django. - python3 manage.py runserver - + sudo -Hu paperless python3 manage.py runserver + and pointing your browser to http://localhost:8000/. .. warning:: This is a development server which should not be used in - production. + production. It is not audited for security and performance + is inferior to production ready web servers. .. hint:: This will not start the consumer. Paperless does this in a separate process. -10. Setup systemd services to run paperless automatically. You may +11. Setup systemd services to run paperless automatically. You may use the service definition files included in the ``scripts`` folder as a starting point. @@ -321,17 +427,15 @@ writing. Windows is not and will never be supported. ``consumer`` script to watch the input folder, and the ``scheduler`` script to run tasks such as email checking and document consumption. + You may need to adjust the path to the ``gunicorn`` executable. This + will be installed as part of the python dependencies, and is either located + in the ``bin`` folder of your virtual environment, or in ``~/.local/bin/`` if + no virtual environment is used. + These services rely on redis and optionally the database server, but don't need to be started in any particular order. The example files depend on redis being started. If you use a database server, you should - add additinal dependencies. - - .. hint:: - - You may optionally set up your preferred web server to serve - paperless as a wsgi application directly instead of running the - ``webserver`` service. The module containing the wsgi application - is named ``paperless.wsgi``. + add additional dependencies. .. caution:: @@ -340,10 +444,13 @@ writing. Windows is not and will never be supported. however, the documentation of GUnicorn states that you should use a proxy server in front of gunicorn instead. -11. Optional: Install a samba server and make the consumption folder + For instructions on how to use nginx for that, + :ref:`see the instructions below `. + +12. Optional: Install a samba server and make the consumption folder available as a network share. -12. Configure ImageMagick to allow processing of PDF documents. Most distributions have +13. Configure ImageMagick to allow processing of PDF documents. Most distributions have this disabled by default, since PDF documents can contain malware. If you don't do this, paperless will fall back to ghostscript for certain steps such as thumbnail generation. @@ -353,19 +460,137 @@ writing. Windows is not and will never be supported. .. code:: - + to .. code:: +14. Optional: Install the `jbig2enc `_ + encoder. This will reduce the size of generated PDF documents. You'll most likely need + to compile this by yourself, because this software has been patented until around 2017 and + binary packages are not available for most distributions. + +.. _setup-ansible: + +Install Paperless using ansible +=============================== + +.. note:: + + This role currently only supports Debian 10 Buster and Ubuntu 20.04 Focal or later as target hosts. + Additionally, only i386 or amd64 based hosts are supported right now, i.e. installation on arm hosts will fail. + +1. Install ansible 2.7+ on the management node. + This may be the target host paperless-ng is being installed on or any remote host which can access the target host. + For further details, check the ansible `inventory `_ documentation. + + On Debian and Ubuntu, the official repositories should provide a suitable version: + + .. code:: bash + + apt install ansible + ansible --version + + Alternatively, you can install the most recent ansible release using PyPI: + + .. code:: bash + + python3 -m pip install ansible + ansible --version + + Make sure your taget hosts are accessible: + + .. code:: sh + + ansible -m ping YourAnsibleTargetHostGoesHere + +2. Install the latest tag of the ansible role using ansible-galaxy + + .. code:: sh + + ansible-galaxy install git+https://github.com/jonaswinkler/paperless-ng.git,ng-1.4.2 + +3. Create an ansible ``playbook.yml`` in a directory of your choice: + + .. code:: yaml + + - hosts: YourAnsibleTargetHostGoesHere + become: yes + vars_files: + - vars/paperless-ng.yml + roles: + - paperless-ng + + Optional: If you also want to use PostgreSQL on the target system, install and add (for example) the `geerlingguy.postgresql `_ role: + + .. code:: sh + + ansible-galaxy install geerlingguy.postgresql + + .. code:: yaml + + - hosts: YourAnsibleTargetHostGoesHere + become: yes + vars_files: + - vars/paperless-ng.yml + roles: + - geerlingguy.postgresql + - paperless-ng + + Optional: If you also want to use a reverse proxy on the target system, install and add (for example) the `geerlingguy.nginx `_ role: + + .. code:: sh + + ansible-galaxy install geerlingguy.nginx + + .. code:: yaml + + - hosts: YourAnsibleTargetHostGoesHere + become: yes + vars_files: + - vars/paperless-ng.yml + roles: + - geerlingguy.postgresql + - paperless-ng + - geerlingguy.nginx + +4. Create ``vars/paperless-ng.yml`` to configure your ansible deployment: + + .. code:: yaml + + paperlessng_secret_key: PleaseGenerateAStrongKeyForThis + + paperlessng_superuser_name: YourUserName + paperlessng_superuser_email: name@domain.tld + paperlessng_superuser_password: YourDesiredPasswordUsedForFirstLogin + + paperlessng_ocr_languages: + - eng + - deu + + For all of the available options, please check ``ansible/README.md`` and :ref:`configuration`. + + Optional configurations for the above-mentioned PostgreSQL and nginx roles would also go here. + +5. Run the ansible playbook from the management node: + + .. code:: sh + + ansible-playbook playbook.yml + + When this step completes successfully, paperless-ng will be available on the target host at ``http://127.0.0.1:8000`` (or the address you configured). + Migration to paperless-ng ######################### At its core, paperless-ng is still paperless and fully compatible. However, some things have changed under the hood, so you need to adapt your setup depending on -how you installed paperless. The important things to keep in mind are as follows. +how you installed paperless. + +This setup describes how to update an existing paperless Docker installation. +The important things to keep in mind are as follows: * Read the :ref:`changelog ` and take note of breaking changes. * You should decide if you want to stick with SQLite or want to migrate your database @@ -393,32 +618,34 @@ Migration to paperless-ng is then performed in a few simple steps: paperless. 3. Download the latest release of paperless-ng. You can either go with the - docker-compose files or use the archive to build the image yourself. + docker-compose files from `here `__ + or clone the repository to build the image yourself (see :ref:`above `). You can either replace your current paperless folder or put paperless-ng in a different location. .. caution:: - The release include a ``.env`` file. This will set the - project name for docker compose to ``paperless`` so that paperless-ng will - automatically reuse your existing paperless volumes. When you start it, it - will migrate your existing data. After that, your old paperless installation - will be incompatible with the migrated volumes. + Paperless-ng includes a ``.env`` file. This will set the + project name for docker compose to ``paperless``, which will also define the name + of the volumes by paperless-ng. However, if you experience that paperless-ng + is not using your old paperless volumes, verify the names of your volumes with -4. Copy the ``docker-compose.sqlite.yml`` file to ``docker-compose.yml``. + .. code:: shell-session + + $ docker volume ls | grep _data + + and adjust the project name in the ``.env`` file so that it matches the name + of the volumes before the ``_data`` part. + + +4. Download the ``docker-compose.sqlite.yml`` file to ``docker-compose.yml``. If you want to switch to PostgreSQL, do that after you migrated your existing SQLite database. -5. Adjust ``docker-compose.yml`` and - ``docker-compose.env`` to your needs. - See `docker route`_ for details on which edits are advised. +5. Adjust ``docker-compose.yml`` and ``docker-compose.env`` to your needs. + See :ref:`setup-docker_hub` for details on which edits are advised. -6. Since ``docker-compose`` would just use the the old paperless image, we need to - manually build a new image: - - .. code:: shell-session - - $ docker-compose build +6. :ref:`Update paperless. ` 7. In order to find your existing documents with the new search feature, you need to invoke a one-time operation that will create the search index: @@ -426,7 +653,7 @@ Migration to paperless-ng is then performed in a few simple steps: .. code:: shell-session $ docker-compose run --rm webserver document_index reindex - + This will migrate your database and create the search index. After that, paperless will take care of maintaining the index by itself. @@ -439,7 +666,7 @@ Migration to paperless-ng is then performed in a few simple steps: This will run paperless in the background and automatically start it on system boot. 9. Paperless installed a permanent redirect to ``admin/`` in your browser. This - redirect is still in place and prevents access to the new UI. Clear + redirect is still in place and prevents access to the new UI. Clear your browsing cache in order to fix this. 10. Optionally, follow the instructions below to migrate your existing data to PostgreSQL. @@ -460,6 +687,15 @@ management commands as below. load data from an old database schema in SQLite into a newer database schema in PostgreSQL, you will run into trouble. +.. warning:: + + On some database fields, PostgreSQL enforces predefined limits on maximum + length, whereas SQLite does not. The fields in question are the title of documents + (128 characters), names of document types, tags and correspondents (128 characters), + and filenames (1024 characters). If you have data in these fields that surpasses these + limits, migration to PostgreSQL is not possible and will fail with an error. + + 1. Stop paperless, if it is running. 2. Tell paperless to use PostgreSQL: @@ -478,19 +714,17 @@ management commands as below. $ cd /path/to/paperless $ docker-compose run --rm webserver /bin/bash - + This will launch the container and initialize the PostgreSQL database. - - b) Without docker, open a shell in your virtual environment, switch to + + b) Without docker, remember to activate any virtual environment, switch to the ``src`` directory and create the database schema: .. code:: shell-session - $ cd /path/to/paperless - $ pipenv shell - $ cd src + $ cd /path/to/paperless/src $ python3 manage.py migrate - + This will not copy any data yet. 4. Dump your data from SQLite: @@ -498,14 +732,14 @@ management commands as below. .. code:: shell-session $ python3 manage.py dumpdata --database=sqlite --exclude=contenttypes --exclude=auth.Permission > data.json - + 5. Load your data into PostgreSQL: .. code:: shell-session $ python3 manage.py loaddata data.json -6. Exit the shell. +6. If operating inside Docker, you may exit the shell now. .. code:: shell-session @@ -549,12 +783,13 @@ as well. Considerations for less powerful devices ######################################## -Paperless runs on Raspberry Pi. However, some things are rather slow on the Pi and +Paperless runs on Raspberry Pi. However, some things are rather slow on the Pi and configuring some options in paperless can help improve performance immensely: * Stick with SQLite to save some resources. * Consider setting ``PAPERLESS_OCR_PAGES`` to 1, so that paperless will only OCR - the first page of your documents. + the first page of your documents. In most cases, this page contains enough + information to be able to find it. * ``PAPERLESS_TASK_WORKERS`` and ``PAPERLESS_THREADS_PER_WORKER`` are configured to use all cores. The Raspberry Pi models 3 and up have 4 cores, meaning that paperless will use 2 workers and 2 threads per worker. This may result in @@ -565,23 +800,69 @@ configuring some options in paperless can help improve performance immensely: your documents before feeding them into paperless. Some scanners are able to do this! You might want to even specify ``skip_noarchive`` to skip archive file generation for already ocr'ed documents entirely. +* If you want to perform OCR on the the device, consider using ``PAPERLESS_OCR_CLEAN=none``. + This will speed up OCR times and use less memory at the expense of slightly worse + OCR results. * Set ``PAPERLESS_OPTIMIZE_THUMBNAILS`` to 'false' if you want faster consumption times. Thumbnails will be about 20% larger. +* If using docker, consider setting ``PAPERLESS_WEBSERVER_WORKERS`` to + 1. This will save some memory. For details, refer to :ref:`configuration`. .. note:: - + Updating the :ref:`automatic matching algorithm ` takes quite a bit of time. However, the update mechanism checks if your - data has changed before doing the heavy lifting. If you experience the + data has changed before doing the heavy lifting. If you experience the algorithm taking too much cpu time, consider changing the schedule in the admin interface to daily. You can also manually invoke the task by changing the date and time of the next run to today/now. - The actual matching of the algorithm is fast and works on Raspberry Pi as + The actual matching of the algorithm is fast and works on Raspberry Pi as well as on any other device. - - .. _redis: https://redis.io/ + + +.. _setup-nginx: + +Using nginx as a reverse proxy +############################## + +If you want to expose paperless to the internet, you should hide it behind a +reverse proxy with SSL enabled. + +In addition to the usual configuration for SSL, +the following configuration is required for paperless to operate: + +.. code:: nginx + + http { + + # Adjust as required. This is the maximum size for file uploads. + # The default value 1M might be a little too small. + client_max_body_size 10M; + + server { + + location / { + + # Adjust host and port as required. + proxy_pass http://localhost:8000/; + + # These configuration options are required for WebSockets to work. + proxy_http_version 1.1; + proxy_set_header Upgrade $http_upgrade; + proxy_set_header Connection "upgrade"; + + proxy_redirect off; + proxy_set_header Host $host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Host $server_name; + } + } + } + +Also read `this `__, towards the end of the section. diff --git a/docs/troubleshooting.rst b/docs/troubleshooting.rst index dc5bf7f5d..25a1b7f6f 100644 --- a/docs/troubleshooting.rst +++ b/docs/troubleshooting.rst @@ -30,16 +30,28 @@ Consumer fails to pickup any new files ###################################### If you notice that the consumer will only pickup files in the consumption -directory at startup, but won't find any other files added later, check out -the configuration file and enable filesystem polling with the setting -``PAPERLESS_CONSUMER_POLLING``. +directory at startup, but won't find any other files added later, you will need to +enable filesystem polling with the configuration option +``PAPERLESS_CONSUMER_POLLING``, see :ref:`here `. + +This will disable listening to filesystem changes with inotify and paperless will +manually check the consumption directory for changes instead. + + +Paperless always redirects to /admin +#################################### + +You probably had the old paperless installed at some point. Paperless installed +a permanent redirect to /admin in your browser, and you need to clear your +browsing data / cache to fix that. + Operation not permitted ####################### You might see errors such as: -.. code:: +.. code:: shell-session chown: changing ownership of '../export': Operation not permitted @@ -48,4 +60,164 @@ required so that the user running paperless inside docker has write permissions to these folders. This happens when pointing these directories to NFS shares, for example. -Ensure that `chown` is possible on these directories. +Ensure that ``chown`` is possible on these directories. + + +Classifier error: No training data available +############################################ + +This indicates that the Auto matching algorithm found no documents to learn from. +This may have two reasons: + +* You don't use the Auto matching algorithm: The error can be safely ignored in this case. +* You are using the Auto matching algorithm: The classifier explicitly excludes documents + with Inbox tags. Verify that there are documents in your archive without inbox tags. + The algorithm will only learn from documents not in your inbox. + + +UserWarning in sklearn on every single document +############################################### + +You may encounter warnings like this: + +.. code:: + + /usr/local/lib/python3.7/site-packages/sklearn/base.py:315: + UserWarning: Trying to unpickle estimator CountVectorizer from version 0.23.2 when using version 0.24.0. + This might lead to breaking code or invalid results. Use at your own risk. + +This happens when certain dependencies of paperless that are responsible for the auto matching algorithm are +updated. After updating these, your current training data *might* not be compatible anymore. This can be ignored +in most cases. This warning will disappear automatically when paperless updates the training data. + +If you want to get rid of the warning or actually experience issues with automatic matching, delete +the file ``classification_model.pickle`` in the data directory and let paperless recreate it. + + +504 Server Error: Gateway Timeout when adding Office documents +############################################################## + +You may experience these errors when using the optional TIKA integration: + +.. code:: + + requests.exceptions.HTTPError: 504 Server Error: Gateway Timeout for url: http://gotenberg:3000/convert/office + +Gotenberg is a server that converts Office documents into PDF documents and has a default timeout of 10 seconds. +When conversion takes longer, Gotenberg raises this error. + +You can increase the timeout by configuring an environment variable for gotenberg (see also `here `__). +If using docker-compose, this is achieved by the following configuration change in the ``docker-compose.yml`` file: + +.. code:: yaml + + gotenberg: + image: thecodingmachine/gotenberg + restart: unless-stopped + environment: + DISABLE_GOOGLE_CHROME: 1 + DEFAULT_WAIT_TIMEOUT: 30 + +Permission denied errors in the consumption directory +##################################################### + +You might encounter errors such as: + +.. code:: shell-session + + The following error occured while consuming document.pdf: [Errno 13] Permission denied: '/usr/src/paperless/src/../consume/document.pdf' + +This happens when paperless does not have permission to delete files inside the consumption directory. +Ensure that ``USERMAP_UID`` and ``USERMAP_GID`` are set to the user id and group id you use on the host operating system, if these are +different from ``1000``. See :ref:`setup-docker_hub`. + +Also ensure that you are able to read and write to the consumption directory on the host. + + +OSError: [Errno 19] No such device when consuming files +####################################################### + +If you experience errors such as: + +.. code:: shell-session + + File "/usr/local/lib/python3.7/site-packages/whoosh/codec/base.py", line 570, in open_compound_file + return CompoundStorage(dbfile, use_mmap=storage.supports_mmap) + File "/usr/local/lib/python3.7/site-packages/whoosh/filedb/compound.py", line 75, in __init__ + self._source = mmap.mmap(fileno, 0, access=mmap.ACCESS_READ) + OSError: [Errno 19] No such device + + During handling of the above exception, another exception occurred: + + Traceback (most recent call last): + File "/usr/local/lib/python3.7/site-packages/django_q/cluster.py", line 436, in worker + res = f(*task["args"], **task["kwargs"]) + File "/usr/src/paperless/src/documents/tasks.py", line 73, in consume_file + override_tag_ids=override_tag_ids) + File "/usr/src/paperless/src/documents/consumer.py", line 271, in try_consume_file + raise ConsumerError(e) + +Paperless uses a search index to provide better and faster full text searching. This search index is stored inside +the ``data`` folder. The search index uses memory-mapped files (mmap). The above error indicates that paperless +was unable to create and open these files. + +This happens when you're trying to store the data directory on certain file systems (mostly network shares) +that don't support memory-mapped files. + + +Web-UI stuck at "Loading..." +############################ + +This might have multiple reasons. + + +1. If you built the docker image yourself or deployed using the bare metal route, + make sure that there are files in ``/static/frontend//``. + If there are no files, make sure that you executed ``collectstatic`` successfully, either + manually or as part of the docker image build. + + If the front end is still missing, make sure that the front end is compiled (files present in + ``src/documents/static/frontend``). If it is not, you need to compile the front end yourself + or download the release archive instead of cloning the repository. + +2. Check the output of the web server. You might see errors like this: + + + .. code:: + + [2021-01-25 10:08:04 +0000] [40] [ERROR] Socket error processing request. + Traceback (most recent call last): + File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 134, in handle + self.handle_request(listener, req, client, addr) + File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 190, in handle_request + util.reraise(*sys.exc_info()) + File "/usr/local/lib/python3.7/site-packages/gunicorn/util.py", line 625, in reraise + raise value + File "/usr/local/lib/python3.7/site-packages/gunicorn/workers/sync.py", line 178, in handle_request + resp.write_file(respiter) + File "/usr/local/lib/python3.7/site-packages/gunicorn/http/wsgi.py", line 396, in write_file + if not self.sendfile(respiter): + File "/usr/local/lib/python3.7/site-packages/gunicorn/http/wsgi.py", line 386, in sendfile + sent += os.sendfile(sockno, fileno, offset + sent, count) + OSError: [Errno 22] Invalid argument + + To fix this issue, add + + .. code:: + + SENDFILE=0 + + to your `docker-compose.env` file. + +Error while reading metadata +############################ + +You might find messages like these in your log files: + +.. code:: + + [WARNING] [paperless.parsing.tesseract] Error while reading metadata + +This indicates that paperless failed to read PDF metadata from one of your documents. This happens when you +open the affected documents in paperless for editing. Paperless will continue to work, and will simply not +show the invalid metadata. diff --git a/docs/usage_overview.rst b/docs/usage_overview.rst index bb9ecd452..b9f235957 100644 --- a/docs/usage_overview.rst +++ b/docs/usage_overview.rst @@ -57,9 +57,6 @@ Adding documents to paperless ############################# Once you've got Paperless setup, you need to start feeding documents into it. -Currently, there are four options: the consumption directory, the dashboard, IMAP (email), and -HTTP POST. - When adding documents to paperless, it will perform the following operations on your documents: @@ -74,7 +71,7 @@ your documents: This process can be configured to fit your needs. If you don't want paperless to create archived versions for digital documents, you can configure that by - configuring ``PAPERLESS_OCR_MODE=skip_noarchive``. Please read the + configuring ``PAPERLESS_OCR_MODE=skip_noarchive``. Please read the :ref:`relevant section in the documentation `. .. note:: @@ -112,6 +109,19 @@ Dashboard upload The dashboard has a file drop field to upload documents to paperless. Simply drag a file onto this field or select a file with the file dialog. Multiple files are supported. + +.. _usage-mobile_upload: + +Mobile upload +============= + +The mobile app over at ``_ allows Android users +to share any documents with paperless. This can be combined with any of the mobile +scanning apps out there, such as Office Lens. + +Furthermore, there is the `Paperless App `_ as well, +which not only has document upload, but also document browsing and download features. + .. _usage-email: IMAP (Email) @@ -245,6 +255,8 @@ Here are a couple examples of tags and types that you could use in your collecti * A tag ``missing_metadata`` when you still need to add some metadata to a document, but can't or don't want to do this right now. +.. _basic-usage_searching: + Searching ######### @@ -277,7 +289,7 @@ Matching specific tags, correspondents or types: Matching dates: .. code:: - + created:[2005 to 2009] added:yesterday modified:today @@ -294,11 +306,11 @@ Matching inexact words: auto complete and query correction. All of these constructs can be combined as you see fit. -If you want to learn more about the query language used by paperless, paperless uses Whoosh's default query language. +If you want to learn more about the query language used by paperless, paperless uses Whoosh's default query language. Head over to `Whoosh query language `_. For details on what date parsing utilities are available, see `Date parsing `_. - + .. _usage-recommended_workflow: @@ -373,7 +385,7 @@ Once you have scanned in a document, proceed in paperless as follows. 6. Remove inbox tags from the documents. .. hint:: - + You can setup manual matching rules for your correspondents and tags and paperless will assign them automatically. After consuming a couple documents, you can even ask paperless to *learn* when to assign tags and correspondents diff --git a/docker/gunicorn.conf.py b/gunicorn.conf.py similarity index 76% rename from docker/gunicorn.conf.py rename to gunicorn.conf.py index a2f456079..bcc12490e 100644 --- a/docker/gunicorn.conf.py +++ b/gunicorn.conf.py @@ -1,21 +1,9 @@ -bind = '127.0.0.1:8000' -backlog = 2048 -workers = 3 -worker_class = 'sync' -worker_connections = 1000 -timeout = 20 -keepalive = 2 -spew = False -daemon = False -pidfile = None -umask = 0 -user = None -group = None -tmp_upload_dir = None -loglevel = 'info' -errorlog = '-' -accesslog = '-' -proc_name = None +import os + +bind = '0.0.0.0:8000' +workers = int(os.getenv("PAPERLESS_WEBSERVER_WORKERS", 2)) +worker_class = 'paperless.workers.ConfigurableWorker' +timeout = 120 def pre_fork(server, worker): pass diff --git a/install-paperless-ng.sh b/install-paperless-ng.sh new file mode 100755 index 000000000..a31c9203f --- /dev/null +++ b/install-paperless-ng.sh @@ -0,0 +1,318 @@ +#!/bin/bash + +ask() { + while true ; do + if [[ -z $3 ]] ; then + read -p "$1 [$2]: " result + else + read -p "$1 ($3) [$2]: " result + fi + if [[ -z $result ]]; then + ask_result=$2 + return + fi + array=$3 + if [[ -z $3 || " ${array[@]} " =~ " ${result} " ]]; then + ask_result=$result + return + else + echo "Invalid option: $result" + fi + done +} + +ask_docker_folder() { + while true ; do + + read -p "$1 [$2]: " result + + if [[ -z $result ]]; then + ask_result=$2 + return + fi + + if [[ $result == /* || $result == ./* ]]; then + ask_result=$result + return + else + echo "Invalid folder: $result" + fi + + + done +} + +if [[ $(id -u) == "0" ]] ; then + echo "Do not run this script as root." + exit 1 +fi + +if [[ -z $(which wget) ]] ; then + echo "wget executable not found. Is wget installed?" + exit 1 +fi + +if [[ -z $(which docker) ]] ; then + echo "docker executable not found. Is docker installed?" + exit 1 +fi + +if [[ -z $(which docker-compose) ]] ; then + echo "docker-compose executable not found. Is docker-compose installed?" + exit 1 +fi + +# Check if user has permissions to run Docker by trying to get the status of Docker (docker status). +# If this fails, the user probably does not have permissions for Docker. +docker stats --no-stream 2>/dev/null 1>&2 +if [ $? -ne 0 ] ; then + echo "" + echo "WARN: It look like the current user does not have Docker permissions." + echo "WARN: Use 'sudo usermod -aG docker $USER' to assign Docker permissions to the user." + echo "" + sleep 3 +fi + +default_time_zone=$(timedatectl show -p Timezone --value) + +set -e + +echo "" +echo "############################################" +echo "### Paperless-ng docker installation ###" +echo "############################################" +echo "" +echo "This script will download, configure and start paperless-ng." + +echo "" +echo "1. Folder configuration" +echo "=======================" +echo "" +echo "The target folder is used to store the configuration files of " +echo "paperless. You can move this folder around after installing paperless." +echo "You will need this folder whenever you want to start, stop, update or " +echo "maintain your paperless instance." +echo "" + +ask "Target folder" "$(pwd)/paperless-ng" +TARGET_FOLDER=$ask_result + +echo "" +echo "The consume folder is where paperles will search for new documents." +echo "Point this to a folder where your scanner is able to put your scanned" +echo "documents." +echo "" +echo "CAUTION: You must specify an absolute path starting with / or a relative " +echo "path starting with ./ here. Examples:" +echo " /mnt/consume" +echo " ./consume" +echo "" + +ask_docker_folder "Consume folder" "$TARGET_FOLDER/consume" +CONSUME_FOLDER=$ask_result + +echo "" +echo "The media folder is where paperless stores your documents." +echo "Leave empty and docker will manage this folder for you." +echo "Docker usually stores managed folders in /var/lib/docker/volumes." +echo "" +echo "CAUTION: If specified, you must specify an absolute path starting with /" +echo "or a relative path starting with ./ here." +echo "" + +ask_docker_folder "Media folder" "" +MEDIA_FOLDER=$ask_result + +echo "" +echo "The data folder is where paperless stores other data, such as your" +echo "SQLite database (if used), the search index and other data." +echo "As with the media folder, leave empty to have this managed by docker." +echo "" +echo "CAUTION: If specified, you must specify an absolute path starting with /" +echo "or a relative path starting with ./ here." +echo "" + +ask_docker_folder "Data folder" "" +DATA_FOLDER=$ask_result + +echo "" +echo "2. Application configuration" +echo "============================" + +echo "" +echo "The port on which the paperless webserver will listen for incoming" +echo "connections." +echo "" + +ask "Port" "8000" +PORT=$ask_result + +echo "" +echo "Paperless requires you to configure the current time zone correctly." +echo "Otherwise, the dates of your documents may appear off by one day," +echo "depending on where you are on earth." +echo "" + +ask "Current time zone" "$default_time_zone" +TIME_ZONE=$ask_result + +echo "" +echo "Database backend: PostgreSQL and SQLite are available. Use PostgreSQL" +echo "if unsure. If you're running on a low-power device such as Raspberry" +echo "Pi, use SQLite to save resources." +echo "" + +ask "Database backend" "postgres" "postgres sqlite" +DATABASE_BACKEND=$ask_result + +echo "" +echo "Paperless is able to use Apache Tika to support Office documents such as" +echo "Word, Excel, Powerpoint, and Libreoffice equivalents. This feature" +echo "requires more resources due to the required services." +echo "" + +ask "Enable Apache Tika?" "no" "yes no" +TIKA_ENABLED=$ask_result + +echo "" +echo "Specify the default language that most of your documents are written in." +echo "Use ISO 639-2, (T) variant language codes: " +echo "https://www.loc.gov/standards/iso639-2/php/code_list.php" +echo "Common values: eng (English) deu (German) nld (Dutch) fra (French)" +echo "" + +ask "OCR language" "eng" +OCR_LANGUAGE=$ask_result + +echo "" +echo "Specify the user id and group id you wish to run paperless as." +echo "Paperless will also change ownership on the data, media and consume" +echo "folder to the specified values, so it's a good idea to supply the user id" +echo "and group id of your unix user account." +echo "If unsure, leave default." +echo "" + +ask "User ID" "$(id -u)" +USERMAP_UID=$ask_result + +ask "Group ID" "$(id -g)" +USERMAP_GID=$ask_result + +echo "" +echo "3. Login credentials" +echo "====================" +echo "" +echo "Specify initial login credentials. You can change these later." +echo "A mail address is required, however it is not used in paperless. You don't" +echo "need to provide an actual mail address." +echo "" + +ask "Paperless username" "$(whoami)" +USERNAME=$ask_result + +while true; do + read -sp "Paperless password: " PASSWORD + echo "" + + if [[ -z $PASSWORD ]] ; then + echo "Password cannot be empty." + continue + fi + + read -sp "Paperless password (again): " PASSWORD_REPEAT + echo "" + + if [[ ! "$PASSWORD" == "$PASSWORD_REPEAT" ]] ; then + echo "Passwords did not match" + else + break + fi +done + +ask "Email" "$USERNAME@localhost" +EMAIL=$ask_result + +echo "" +echo "Summary" +echo "=======" +echo "" + +echo "Target folder: $TARGET_FOLDER" +echo "Consume folder: $CONSUME_FOLDER" +if [[ -z $MEDIA_FOLDER ]] ; then + echo "Media folder: Managed by docker" +else + echo "Media folder: $MEDIA_FOLDER" +fi +if [[ -z $DATA_FOLDER ]] ; then + echo "Data folder: Managed by docker" +else + echo "Data folder: $DATA_FOLDER" +fi +echo "" +echo "Port: $PORT" +echo "Database: $DATABASE_BACKEND" +echo "Tika enabled: $TIKA_ENABLED" +echo "OCR language: $OCR_LANGUAGE" +echo "User id: $USERMAP_UID" +echo "Group id: $USERMAP_GID" +echo "" +echo "Paperless username: $USERNAME" +echo "Paperless email: $EMAIL" + +echo "" +read -p "Press any key to install." + +echo "" +echo "Installing paperless..." +echo "" + +mkdir -p "$TARGET_FOLDER" + +cd "$TARGET_FOLDER" + +DOCKER_COMPOSE_VERSION=$DATABASE_BACKEND + +if [[ $TIKA_ENABLED == "yes" ]] ; then + DOCKER_COMPOSE_VERSION="$DOCKER_COMPOSE_VERSION-tika" +fi + +wget "https://raw.githubusercontent.com/jonaswinkler/paperless-ng/master/docker/compose/docker-compose.$DOCKER_COMPOSE_VERSION.yml" -O docker-compose.yml +wget "https://raw.githubusercontent.com/jonaswinkler/paperless-ng/master/docker/compose/.env" -O .env + +SECRET_KEY=$(cat /dev/urandom | tr -dc 'a-zA-Z0-9' | fold -w 64 | head -n 1) + +DEFAULT_LANGUAGES="deu eng fra ita spa" + +{ + if [[ ! $USERMAP_UID == "1000" ]] ; then + echo "USERMAP_UID=$USERMAP_UID" + fi + if [[ ! $USERMAP_GID == "1000" ]] ; then + echo "USERMAP_GID=$USERMAP_GID" + fi + echo "PAPERLESS_TIME_ZONE=$TIME_ZONE" + echo "PAPERLESS_OCR_LANGUAGE=$OCR_LANGUAGE" + echo "PAPERLESS_SECRET_KEY=$SECRET_KEY" + if [[ ! " ${DEFAULT_LANGUAGES[@]} " =~ " ${OCR_LANGUAGE} " ]] ; then + echo "PAPERLESS_OCR_LANGUAGES=$OCR_LANGUAGE" + fi +} > docker-compose.env + +sed -i "s/- 8000:8000/- $PORT:8000/g" docker-compose.yml + +sed -i "s#- \./consume:/usr/src/paperless/consume#- $CONSUME_FOLDER:/usr/src/paperless/consume#g" docker-compose.yml + +if [[ -n $MEDIA_FOLDER ]] ; then + sed -i "s#- media:/usr/src/paperless/media#- $MEDIA_FOLDER:/usr/src/paperless/media#g" docker-compose.yml +fi + +if [[ -n $DATA_FOLDER ]] ; then + sed -i "s#- data:/usr/src/paperless/data#- $DATA_FOLDER:/usr/src/paperless/data#g" docker-compose.yml +fi + +docker-compose pull + +docker-compose run --rm -e DJANGO_SUPERUSER_PASSWORD="$PASSWORD" webserver createsuperuser --noinput --username "$USERNAME" --email "$EMAIL" + +docker-compose up -d diff --git a/paperless.conf.example b/paperless.conf.example index 910fc22a0..a62d92e05 100644 --- a/paperless.conf.example +++ b/paperless.conf.example @@ -13,6 +13,7 @@ #PAPERLESS_DBNAME=paperless #PAPERLESS_DBUSER=paperless #PAPERLESS_DBPASS=paperless +#PAPERLESS_DBSSLMODE=prefer # Paths and folders @@ -26,11 +27,12 @@ #PAPERLESS_SECRET_KEY=change-me #PAPERLESS_ALLOWED_HOSTS=example.com,www.example.com -#PAPERLESS_CORS_ALLOWED_HOSTS=localhost:8080,example.com,localhost:8000 +#PAPERLESS_CORS_ALLOWED_HOSTS=http://example.com,http://localhost:8000 #PAPERLESS_FORCE_SCRIPT_NAME= #PAPERLESS_STATIC_URL=/static/ #PAPERLESS_AUTO_LOGIN_USERNAME= #PAPERLESS_COOKIE_PREFIX= +#PAPERLESS_ENABLE_HTTP_REMOTE_USER=false # OCR settings @@ -39,7 +41,11 @@ #PAPERLESS_OCR_OUTPUT_TYPE=pdfa #PAPERLESS_OCR_PAGES=1 #PAPERLESS_OCR_IMAGE_DPI=300 -#PAPERLESS_OCR_USER_ARG={} +#PAPERLESS_OCR_CLEAN=clean +#PAPERLESS_OCR_DESKEW=true +#PAPERLESS_OCR_ROTATE_PAGES=true +#PAPERLESS_OCR_ROTATE_PAGES_THRESHOLD=12.0 +#PAPERLESS_OCR_USER_ARGS={} #PAPERLESS_CONVERT_MEMORY_LIMIT=0 #PAPERLESS_CONVERT_TMPDIR=/var/tmp/paperless @@ -50,10 +56,21 @@ #PAPERLESS_TIME_ZONE=UTC #PAPERLESS_CONSUMER_POLLING=10 #PAPERLESS_CONSUMER_DELETE_DUPLICATES=false +#PAPERLESS_CONSUMER_RECURSIVE=false +#PAPERLESS_CONSUMER_IGNORE_PATTERNS=[".DS_STORE/*", "._*", ".stfolder/*"] +#PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS=false #PAPERLESS_OPTIMIZE_THUMBNAILS=true #PAPERLESS_POST_CONSUME_SCRIPT=/path/to/an/arbitrary/script.sh #PAPERLESS_FILENAME_DATE_ORDER=YMD #PAPERLESS_FILENAME_PARSE_TRANSFORMS=[] +#PAPERLESS_THUMBNAIL_FONT_NAME= +#PAPERLESS_IGNORE_DATES= + +# Tika settings + +#PAPERLESS_TIKA_ENABLED=false +#PAPERLESS_TIKA_ENDPOINT=http://localhost:9998 +#PAPERLESS_TIKA_GOTENBERG_ENDPOINT=http://localhost:3000 # Binaries diff --git a/requirements.txt b/requirements.txt new file mode 100644 index 000000000..7fd7b98ca --- /dev/null +++ b/requirements.txt @@ -0,0 +1,103 @@ +# +# These requirements were autogenerated by pipenv +# To regenerate from the project's Pipfile, run: +# +# pipenv lock --requirements +# + +-i https://pypi.python.org/simple +--extra-index-url https://www.piwheels.org/simple +aioredis==1.3.1 +arrow==1.1.1; python_version >= '3.6' +asgiref==3.4.1; python_version >= '3.6' +async-timeout==3.0.1; python_full_version >= '3.5.3' +attrs==21.2.0; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' +autobahn==21.3.1; python_version >= '3.7' +automat==20.2.0 +backports.zoneinfo==0.2.1 +blessed==1.18.1; python_version >= '2.7' +certifi==2021.5.30 +cffi==1.14.6 +channels-redis==3.3.0 +channels==3.0.4 +chardet==4.0.0; python_version >= '3.1' +charset-normalizer==2.0.4; python_version >= '3' +click==8.0.1; python_version >= '3.6' +coloredlogs==15.0.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' +concurrent-log-handler==0.9.19 +constantly==15.1.0 +cryptography==3.4.7 +daphne==3.0.2; python_version >= '3.6' +dateparser==1.0.0 +django-cors-headers==3.8.0 +django-extensions==3.1.3 +django-filter==2.4.0 +django-picklefield==3.0.1; python_version >= '3' +django-q==1.3.9 +django==3.2.6 +djangorestframework==3.12.4 +filelock==3.0.12 +fuzzywuzzy[speedup]==0.18.0 +gunicorn==20.1.0 +h11==0.12.0; python_version >= '3.6' +hiredis==2.0.0; python_version >= '3.6' +httptools==0.2.0 +humanfriendly==9.2; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' +hyperlink==21.0.0 +idna==3.2; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3' +imap-tools==0.46.0 +img2pdf==0.4.1 +incremental==21.3.0 +inotify-simple==1.3.5; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3' +inotifyrecursive==0.3.5 +joblib==1.0.1; python_version >= '3.6' +langdetect==1.0.9 +lxml==4.6.3; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' +msgpack==1.0.2 +numpy==1.20.3 +ocrmypdf==12.3.2 +pathvalidate==2.4.1 +pdfminer.six==20201018 +pikepdf==2.16.1 +pillow==8.3.1 +pluggy==0.13.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3' +portalocker==2.3.0; python_version >= '3' +psycopg2-binary==2.9.1 +pyasn1-modules==0.2.8 +pyasn1==0.4.8 +pycparser==2.20; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3' +pyopenssl==20.0.1; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' +python-dateutil==2.8.2 +python-dotenv==0.19.0 +python-gnupg==0.4.7 +python-levenshtein==0.12.2 +python-magic==0.4.24 +pytz==2021.1 +pyyaml==5.4.1 +redis==3.5.3 +regex==2021.8.3 +reportlab==3.6.1; python_version >= '2.7' and python_version < '4' +requests==2.26.0; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4, 3.5' +scikit-learn==0.24.0 +scipy==1.7.1; python_version < '3.10' and python_version >= '3.7' +service-identity==21.1.0 +six==1.16.0; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3' +sortedcontainers==2.4.0 +sqlparse==0.4.1; python_version >= '3.5' +threadpoolctl==2.2.0; python_version >= '3.6' +tika==1.24 +tqdm==4.62.1 +twisted[tls]==21.7.0; python_full_version >= '3.6.7' +txaio==21.2.1; python_version >= '3.6' +typing-extensions==3.10.0.0 +tzlocal==3.0; python_version >= '3.6' +urllib3==1.26.6; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' and python_version < '4' +uvicorn[standard]==0.15.0 +uvloop==0.16.0 +watchdog==2.1.3 +watchgod==0.7 +wcwidth==0.2.5 +websockets==9.1 +whitenoise==5.3.0 +whoosh==2.7.4 +zope.interface==5.4.0; python_version >= '2.7' and python_version not in '3.0, 3.1, 3.2, 3.3, 3.4' diff --git a/resources/logo.txt b/resources/logo.txt new file mode 100644 index 000000000..75a1d2c75 --- /dev/null +++ b/resources/logo.txt @@ -0,0 +1,16 @@ + 9w + {@@N + Q@@@@H + G@@@@@@@\ + SilN@@@@@@@ + *Q *@@@@@@@@S /= = = = = = = = = = = = = = = = = =\ + *@ B@@@@@@@@N || || + N R$ A@@@@@@@@@@ || PAPERLESS-NG || +x@@ $U B@@@@@@@@@R || || +N@@N^ @ N@@@@@@@@@* \= = = = = = = = = = = = = = = = = =/ +|@@@u @ E@@@@@@@@l + Q@@@ \ Px@@@@@@P + 1@@S` @@@o' + z$ ; + v + / diff --git a/scripts/make-release.sh b/scripts/make-release.sh deleted file mode 100755 index 0a7bc7a9b..000000000 --- a/scripts/make-release.sh +++ /dev/null @@ -1,124 +0,0 @@ -#!/bin/bash - -# Release checklist -# - wait for travis build. -# adjust src/paperless/version.py -# changelog in the documentation -# adjust versions in docker/hub/* -# If docker-compose was modified: all compose files are the same. - -# Steps: -# run release script "dev", push -# if it works: new tag, merge into master -# on master: make release "lastest", push -# on master: make release "version-tag", push -# publish release files - -set -e - - -VERSION=$1 - -if [ -z "$VERSION" ] -then - echo "Need a version string." - exit 1 -fi - -# source root directory of paperless -PAPERLESS_ROOT=$(git rev-parse --show-toplevel) - -# output directory -PAPERLESS_DIST="$PAPERLESS_ROOT/dist" -PAPERLESS_DIST_APP="$PAPERLESS_DIST/paperless-ng" -PAPERLESS_DIST_DOCKERFILES="$PAPERLESS_DIST/paperless-ng-dockerfiles" - -if [ -d "$PAPERLESS_DIST" ] -then - echo "Removing $PAPERLESS_DIST" - rm "$PAPERLESS_DIST" -r -fi - -mkdir "$PAPERLESS_DIST" -mkdir "$PAPERLESS_DIST_APP" -mkdir "$PAPERLESS_DIST_APP/docker" -mkdir "$PAPERLESS_DIST_APP/scripts" -mkdir "$PAPERLESS_DIST_DOCKERFILES" - -# setup dependencies. - -cd "$PAPERLESS_ROOT" - -pipenv clean -pipenv install --dev -pipenv lock --keep-outdated -r > "$PAPERLESS_DIST_APP/requirements.txt" - -# test if the application works. - -cd "$PAPERLESS_ROOT/src" -pipenv run pytest --cov -pipenv run pycodestyle - -# make the documentation. - -cd "$PAPERLESS_ROOT/docs" -make clean html - -# copy stuff into place - -# the application itself - -cp "$PAPERLESS_ROOT/.env" \ - "$PAPERLESS_ROOT/.dockerignore" \ - "$PAPERLESS_ROOT/CONTRIBUTING.md" \ - "$PAPERLESS_ROOT/LICENSE" \ - "$PAPERLESS_ROOT/Pipfile" \ - "$PAPERLESS_ROOT/Pipfile.lock" \ - "$PAPERLESS_ROOT/README.md" "$PAPERLESS_DIST_APP" - -cp "$PAPERLESS_ROOT/paperless.conf.example" "$PAPERLESS_DIST_APP/paperless.conf" - -# copy python source, templates and static files. -cd "$PAPERLESS_ROOT" -find src -wholename '*/templates/*' -o -wholename '*/static/*' -o -name '*.py' | cpio -pdm "$PAPERLESS_DIST_APP" - -# build the front end. - -cd "$PAPERLESS_ROOT/src-ui" -ng build --prod --output-hashing none --sourceMap=false --output-path "$PAPERLESS_DIST_APP/src/documents/static/frontend" - -# documentation -cp "$PAPERLESS_ROOT/docs/_build/html/" "$PAPERLESS_DIST_APP/docs" -r - -# docker files for building the image yourself -cp "$PAPERLESS_ROOT/docker/local/"* "$PAPERLESS_DIST_APP" -cp "$PAPERLESS_ROOT/docker/docker-compose.env" "$PAPERLESS_DIST_APP" - -# docker files for pulling from docker hub -cp "$PAPERLESS_ROOT/docker/hub/"* "$PAPERLESS_DIST_DOCKERFILES" -cp "$PAPERLESS_ROOT/.env" "$PAPERLESS_DIST_DOCKERFILES" -cp "$PAPERLESS_ROOT/docker/docker-compose.env" "$PAPERLESS_DIST_DOCKERFILES" - -# auxiliary files required for the docker image -cp "$PAPERLESS_ROOT/docker/docker-entrypoint.sh" "$PAPERLESS_DIST_APP/docker/" -cp "$PAPERLESS_ROOT/docker/gunicorn.conf.py" "$PAPERLESS_DIST_APP/docker/" -cp "$PAPERLESS_ROOT/docker/imagemagick-policy.xml" "$PAPERLESS_DIST_APP/docker/" -cp "$PAPERLESS_ROOT/docker/supervisord.conf" "$PAPERLESS_DIST_APP/docker/" - -# auxiliary files for bare metal installs -cp "$PAPERLESS_ROOT/scripts/paperless-webserver.service" "$PAPERLESS_DIST_APP/scripts/" -cp "$PAPERLESS_ROOT/scripts/paperless-consumer.service" "$PAPERLESS_DIST_APP/scripts/" -cp "$PAPERLESS_ROOT/scripts/paperless-scheduler.service" "$PAPERLESS_DIST_APP/scripts/" - -# try to make the docker build. - -cd "$PAPERLESS_DIST_APP" - -docker build . -t "jonaswinkler/paperless-ng:$VERSION" - -# works. package the app! - -cd "$PAPERLESS_DIST" - -tar -cJf "paperless-ng-$VERSION.tar.xz" paperless-ng/ -tar -cJf "paperless-ng-$VERSION-dockerfiles.tar.xz" paperless-ng-dockerfiles/ diff --git a/scripts/paperless-scheduler.service b/scripts/paperless-scheduler.service index c565d8af4..b1c82a38e 100644 --- a/scripts/paperless-scheduler.service +++ b/scripts/paperless-scheduler.service @@ -1,5 +1,5 @@ [Unit] -Description=Paperless consumer +Description=Paperless scheduler Requires=redis.service [Service] diff --git a/scripts/paperless-webserver.service b/scripts/paperless-webserver.service index a43110be2..6a4655b6f 100644 --- a/scripts/paperless-webserver.service +++ b/scripts/paperless-webserver.service @@ -8,7 +8,7 @@ Requires=redis.service User=paperless Group=paperless WorkingDirectory=/opt/paperless/src -ExecStart=/opt/paperless/.local/bin/gunicorn paperless.wsgi -w 2 -b 0.0.0.0:8000 +ExecStart=/opt/paperless/.local/bin/gunicorn -c /opt/paperless/gunicorn.conf.py paperless.asgi:application [Install] WantedBy=multi-user.target diff --git a/scripts/push-release.sh b/scripts/push-release.sh deleted file mode 100755 index cfa63f5cf..000000000 --- a/scripts/push-release.sh +++ /dev/null @@ -1,23 +0,0 @@ -#!/bin/bash - -set -e - - -VERSION=$1 - -if [ -z "$VERSION" ] -then - echo "Need a version string." - exit 1 -fi - -# source root directory of paperless -PAPERLESS_ROOT=$(git rev-parse --show-toplevel) - -# output directory -PAPERLESS_DIST="$PAPERLESS_ROOT/dist" -PAPERLESS_DIST_APP="$PAPERLESS_DIST/paperless-ng" - -cd "$PAPERLESS_DIST_APP" - -docker push "jonaswinkler/paperless-ng:$VERSION" diff --git a/scripts/start_services.sh b/scripts/start_services.sh index e566f59b3..e2fc740a4 100755 --- a/scripts/start_services.sh +++ b/scripts/start_services.sh @@ -1,2 +1,4 @@ -docker run -p 5432:5432 -v paperless_pgdata:/var/lib/postgresql/data -d postgres:13 +docker run -p 5432:5432 -e POSTGRES_PASSWORD=password -v paperless_pgdata:/var/lib/postgresql/data -d postgres:13 docker run -d -p 6379:6379 redis:latest +docker run -p 3000:3000 -d thecodingmachine/gotenberg +docker run -p 9998:9998 -d apache/tika diff --git a/src-ui/angular.json b/src-ui/angular.json index 2ff1bb3b0..b9659fd07 100644 --- a/src-ui/angular.json +++ b/src-ui/angular.json @@ -13,6 +13,24 @@ "root": "", "sourceRoot": "src", "prefix": "app", + "i18n": { + "sourceLocale": "en-US", + "locales": { + "de-DE": "src/locale/messages.de_DE.xlf", + "nl-NL": "src/locale/messages.nl_NL.xlf", + "fr-FR": "src/locale/messages.fr_FR.xlf", + "en-GB": "src/locale/messages.en_GB.xlf", + "pt-BR": "src/locale/messages.pt_BR.xlf", + "pt-PT": "src/locale/messages.pt_PT.xlf", + "it-IT": "src/locale/messages.it_IT.xlf", + "ro-RO": "src/locale/messages.ro_RO.xlf", + "ru-RU": "src/locale/messages.ru_RU.xlf", + "es-ES": "src/locale/messages.es_ES.xlf", + "pl-PL": "src/locale/messages.pl_PL.xlf", + "sv-SE": "src/locale/messages.sv_SE.xlf", + "lb-LU": "src/locale/messages.lb_LU.xlf" + } + }, "architect": { "build": { "builder": "@angular-devkit/build-angular:browser", @@ -23,15 +41,25 @@ "main": "src/main.ts", "polyfills": "src/polyfills.ts", "tsConfig": "tsconfig.app.json", + "localize": true, "aot": true, "assets": [ "src/favicon.ico", - "src/assets" + "src/apple-touch-icon.png", + "src/assets", + "src/manifest.webmanifest", { + "glob": "pdf.worker.min.js", + "input": "node_modules/pdfjs-dist/build/", + "output": "/assets/js/" + } ], "styles": [ "src/styles.scss" ], - "scripts": [] + "scripts": [], + "allowedCommonJsDependencies": [ + "ng2-pdf-viewer" + ] }, "configurations": { "production": { @@ -41,10 +69,10 @@ "with": "src/environments/environment.prod.ts" } ], + "outputPath": "../src/documents/static/frontend/", "optimization": true, "outputHashing": "none", "sourceMap": false, - "extractCss": true, "namedChunks": false, "extractLicenses": true, "vendorChunk": false, @@ -61,13 +89,16 @@ "maximumError": "10kb" } ] + }, + "en-US": { + "localize": ["en-US"] } } }, "serve": { "builder": "@angular-devkit/build-angular:dev-server", "options": { - "browserTarget": "paperless-ui:build" + "browserTarget": "paperless-ui:build:en-US" }, "configurations": { "production": { @@ -78,7 +109,8 @@ "extract-i18n": { "builder": "@angular-devkit/build-angular:extract-i18n", "options": { - "browserTarget": "paperless-ui:build" + "browserTarget": "paperless-ui:build", + "ivy": true } }, "test": { @@ -90,7 +122,9 @@ "karmaConfig": "karma.conf.js", "assets": [ "src/favicon.ico", - "src/assets" + "src/apple-touch-icon.png", + "src/assets", + "src/manifest.webmanifest" ], "styles": [ "src/styles.scss" @@ -127,4 +161,4 @@ } }, "defaultProject": "paperless-ui" -} \ No newline at end of file +} diff --git a/src-ui/e2e/protractor.conf.js b/src-ui/e2e/protractor.conf.js index f238c0bbe..09a96840f 100644 --- a/src-ui/e2e/protractor.conf.js +++ b/src-ui/e2e/protractor.conf.js @@ -33,4 +33,4 @@ exports.config = { } })); } -}; \ No newline at end of file +}; diff --git a/src-ui/messages.xlf b/src-ui/messages.xlf new file mode 100644 index 000000000..a733dd809 --- /dev/null +++ b/src-ui/messages.xlf @@ -0,0 +1,2050 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + + + Open document + + src/app/app.component.ts + 51 + + + + Could not add : + + src/app/app.component.ts + 59 + + + + New document detected + + src/app/app.component.ts + 65 + + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + + + English (US) + + src/app/services/settings.service.ts + 90 + + + + English (GB) + + src/app/services/settings.service.ts + 91 + + + + German + + src/app/services/settings.service.ts + 92 + + + + Dutch + + src/app/services/settings.service.ts + 93 + + + + French + + src/app/services/settings.service.ts + 94 + + + + Portuguese + + src/app/services/settings.service.ts + 95 + + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + + + Italian + + src/app/services/settings.service.ts + 97 + + + + Romanian + + src/app/services/settings.service.ts + 98 + + + + Russian + + src/app/services/settings.service.ts + 99 + + + + Spanish + + src/app/services/settings.service.ts + 100 + + + + Polish + + src/app/services/settings.service.ts + 101 + + + + Swedish + + src/app/services/settings.service.ts + 102 + + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + + + Error + + src/app/services/toast.service.ts + 35 + + + + Information + + src/app/services/toast.service.ts + 39 + + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + + + Document type + + src/app/services/rest/document.service.ts + 20 + + + + Created + + src/app/services/rest/document.service.ts + 21 + + + + Added + + src/app/services/rest/document.service.ts + 22 + + + + Modified + + src/app/services/rest/document.service.ts + 23 + + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + + + Any word + + src/app/data/matching-model.ts + 12 + + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + + + All words + + src/app/data/matching-model.ts + 13 + + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + + + Exact match + + src/app/data/matching-model.ts + 14 + + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + + + Regular expression + + src/app/data/matching-model.ts + 15 + + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + + + + diff --git a/src-ui/package-lock.json b/src-ui/package-lock.json index 5eca0b3c0..da2df3597 100644 --- a/src-ui/package-lock.json +++ b/src-ui/package-lock.json @@ -1,366 +1,426 @@ { "name": "paperless-ui", "version": "0.0.0", - "lockfileVersion": 1, + "lockfileVersion": 2, "requires": true, - "dependencies": { - "@angular-devkit/architect": { - "version": "0.1001.5", - "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1001.5.tgz", - "integrity": "sha512-W8ZqtbxwDtHnzPoqVyeyDEq24i+H0/i0fjIBuJ+XAMtd3U9JtPALIRLdhnunLXO7OLxjtxjzh0qLxKgiXGEd3g==", - "dev": true, - "requires": { - "@angular-devkit/core": "10.1.5", - "rxjs": "6.6.2" - }, + "packages": { + "": { + "name": "paperless-ui", + "version": "0.0.0", "dependencies": { - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "tslib": { - "version": "1.14.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", - "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", - "dev": true - } + "@angular/animations": "~11.2.14", + "@angular/common": "~11.2.14", + "@angular/compiler": "~11.2.14", + "@angular/core": "~11.2.14", + "@angular/forms": "~11.2.14", + "@angular/localize": "~11.2.14", + "@angular/platform-browser": "~11.2.14", + "@angular/platform-browser-dynamic": "~11.2.14", + "@angular/router": "~11.2.14", + "@ng-bootstrap/ng-bootstrap": "^9.1.2", + "@ng-select/ng-select": "^7.0.0", + "bootstrap": "^4.5.0", + "file-saver": "^2.0.5", + "ng2-pdf-viewer": "^6.3.2", + "ngx-bootstrap": "^6.2.0", + "ngx-color": "^6.2.0", + "ngx-cookie-service": "^10.1.1", + "ngx-file-drop": "^11.1.0", + "ngx-infinite-scroll": "^9.1.0", + "rxjs": "~6.6.0", + "tslib": "^2.0.0", + "uuid": "^8.3.1", + "zone.js": "~0.11.4" + }, + "devDependencies": { + "@angular-devkit/build-angular": "~0.1102.13", + "@angular/cli": "~11.2.14", + "@angular/compiler-cli": "~11.2.14", + "@types/jasmine": "~3.6.0", + "@types/jasminewd2": "~2.0.3", + "@types/node": "^12.11.1", + "codelyzer": "^6.0.0", + "jasmine-core": "~3.6.0", + "jasmine-spec-reporter": "~5.0.0", + "karma": "~6.3.3", + "karma-chrome-launcher": "~3.1.0", + "karma-coverage-istanbul-reporter": "~3.0.2", + "karma-jasmine": "~4.0.0", + "karma-jasmine-html-reporter": "^1.5.0", + "protractor": "~7.0.0", + "ts-node": "~8.3.0", + "tslint": "~6.1.0", + "typescript": "~4.1.5" } }, - "@angular-devkit/build-angular": { - "version": "0.1002.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-0.1002.0.tgz", - "integrity": "sha512-cPkdp1GceokGHc79Wg0hACMqqmnJ4W3H9kY4c9qp1Xz18b3vk1aq09JNawOpfUN09S9vBCnn4glg22lRyqmJNA==", + "node_modules/@angular-devkit/architect": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1102.14.tgz", + "integrity": "sha512-965TVXuBtRb8RySgxRxUEO+YTd7mT0xiqVHSe+MHvMtUCmEE9vwRofFZl6axkK5ri4fiomiMnOVE19aw4spgNQ==", "dev": true, - "requires": { - "@angular-devkit/architect": "0.1002.0", - "@angular-devkit/build-optimizer": "0.1002.0", - "@angular-devkit/build-webpack": "0.1002.0", - "@angular-devkit/core": "10.2.0", - "@babel/core": "7.11.1", - "@babel/generator": "7.11.0", - "@babel/plugin-transform-runtime": "7.11.0", - "@babel/preset-env": "7.11.0", - "@babel/runtime": "7.11.2", - "@babel/template": "7.10.4", + "dependencies": { + "@angular-devkit/core": "11.2.14", + "rxjs": "6.6.3" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular-devkit/build-angular": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-0.1102.14.tgz", + "integrity": "sha512-SyX9SK3qfpk6xNIrxpxYi8zxP/cN2kny4I+XYbkKvgGiE3qhkrC/PRJE9OWj0sloekLD0CDfFWOvIiw3GMc4Tg==", + "dev": true, + "dependencies": { + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/build-optimizer": "0.1102.14", + "@angular-devkit/build-webpack": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "@babel/core": "7.12.10", + "@babel/generator": "7.12.11", + "@babel/plugin-transform-async-to-generator": "7.12.1", + "@babel/plugin-transform-runtime": "7.12.10", + "@babel/preset-env": "7.12.11", + "@babel/runtime": "7.12.5", + "@babel/template": "7.12.7", + "@discoveryjs/json-ext": "0.5.2", "@jsdevtools/coverage-istanbul-loader": "3.0.5", - "@ngtools/webpack": "10.2.0", - "autoprefixer": "9.8.6", - "babel-loader": "8.1.0", + "@ngtools/webpack": "11.2.14", + "ansi-colors": "4.1.1", + "autoprefixer": "10.2.4", + "babel-loader": "8.2.2", "browserslist": "^4.9.1", "cacache": "15.0.5", "caniuse-lite": "^1.0.30001032", - "circular-dependency-plugin": "5.2.0", - "copy-webpack-plugin": "6.0.3", - "core-js": "3.6.4", - "css-loader": "4.2.2", - "cssnano": "4.1.10", - "file-loader": "6.0.0", + "circular-dependency-plugin": "5.2.2", + "copy-webpack-plugin": "6.3.2", + "core-js": "3.8.3", + "critters": "0.0.7", + "css-loader": "5.0.1", + "cssnano": "5.0.2", + "file-loader": "6.2.0", "find-cache-dir": "3.3.1", "glob": "7.1.6", - "jest-worker": "26.3.0", + "https-proxy-agent": "5.0.0", + "inquirer": "7.3.3", + "jest-worker": "26.6.2", "karma-source-map-support": "1.4.0", - "less-loader": "6.2.0", - "license-webpack-plugin": "2.3.0", + "less": "4.1.1", + "less-loader": "7.3.0", + "license-webpack-plugin": "2.3.11", "loader-utils": "2.0.0", - "mini-css-extract-plugin": "0.10.0", + "mini-css-extract-plugin": "1.3.5", "minimatch": "3.0.4", - "open": "7.2.0", - "parse5": "6.0.1", - "parse5-htmlparser2-tree-adapter": "6.0.1", + "open": "7.4.0", + "ora": "5.3.0", + "parse5-html-rewriting-stream": "6.0.1", "pnp-webpack-plugin": "1.6.4", - "postcss": "7.0.32", - "postcss-import": "12.0.1", - "postcss-loader": "3.0.0", - "raw-loader": "4.0.1", + "postcss": "8.2.15", + "postcss-import": "14.0.0", + "postcss-loader": "4.2.0", + "raw-loader": "4.0.2", "regenerator-runtime": "0.13.7", - "resolve-url-loader": "3.1.2", + "resolve-url-loader": "4.0.0", "rimraf": "3.0.2", - "rollup": "2.26.5", - "rxjs": "6.6.2", - "sass": "1.26.10", - "sass-loader": "10.0.1", - "semver": "7.3.2", + "rollup": "2.38.4", + "rxjs": "6.6.3", + "sass": "1.32.6", + "sass-loader": "10.1.1", + "semver": "7.3.4", "source-map": "0.7.3", - "source-map-loader": "1.0.2", + "source-map-loader": "1.1.3", "source-map-support": "0.5.19", - "speed-measure-webpack-plugin": "1.3.3", - "style-loader": "1.2.1", + "speed-measure-webpack-plugin": "1.4.2", + "style-loader": "2.0.0", "stylus": "0.54.8", - "stylus-loader": "3.0.2", - "terser": "5.3.0", - "terser-webpack-plugin": "4.1.0", + "stylus-loader": "4.3.3", + "terser": "5.5.1", + "terser-webpack-plugin": "4.2.3", + "text-table": "0.2.0", "tree-kill": "1.2.2", - "webpack": "4.44.1", + "webpack": "4.44.2", "webpack-dev-middleware": "3.7.2", - "webpack-dev-server": "3.11.0", - "webpack-merge": "4.2.2", - "webpack-sources": "1.4.3", - "webpack-subresource-integrity": "1.4.1", + "webpack-dev-server": "3.11.2", + "webpack-merge": "5.7.3", + "webpack-sources": "2.2.0", + "webpack-subresource-integrity": "1.5.2", "worker-plugin": "5.0.0" }, - "dependencies": { - "@angular-devkit/architect": { - "version": "0.1002.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1002.0.tgz", - "integrity": "sha512-twM8V03ujBIGVpgV1PBlSDodUdxtUb7WakutfWafAvEHUsgwzfvQz2VtKWvjNZ9AiYjnCuwkQaclqVv0VHNo9w==", - "dev": true, - "requires": { - "@angular-devkit/core": "10.2.0", - "rxjs": "6.6.2" - } + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + }, + "peerDependencies": { + "@angular/compiler-cli": "^11.0.0 || ^11.2.0-next", + "@angular/localize": "^11.0.0 || ^11.2.0-next", + "@angular/service-worker": "^11.0.0 || ^11.2.0-next", + "karma": "^5.2.0 || ^6.0.0", + "ng-packagr": "^11.0.0 || ^11.2.0-next", + "protractor": "^7.0.0", + "tailwindcss": "^2.0.0", + "tslint": "^6.1.0", + "typescript": "~4.0.0 || ~4.1.0" + }, + "peerDependenciesMeta": { + "@angular/localize": { + "optional": true }, - "@angular-devkit/core": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-10.2.0.tgz", - "integrity": "sha512-XAszFhSF3mZw1VjoOsYGbArr5NJLcStjOvcCGjBPl1UBM2AKpuCQXHxI9XJGYKL3B93Vp5G58d8qkHvamT53OA==", - "dev": true, - "requires": { - "ajv": "6.12.4", - "fast-json-stable-stringify": "2.1.0", - "magic-string": "0.25.7", - "rxjs": "6.6.2", - "source-map": "0.7.3" - } + "@angular/service-worker": { + "optional": true }, - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } + "karma": { + "optional": true }, - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true + "ng-packagr": { + "optional": true + }, + "protractor": { + "optional": true + }, + "tailwindcss": { + "optional": true + }, + "tslint": { + "optional": true } } }, - "@angular-devkit/build-optimizer": { - "version": "0.1002.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-optimizer/-/build-optimizer-0.1002.0.tgz", - "integrity": "sha512-ACnm9doPMbRtSy1UZN5ir7smeLMx0g0oW7jX3jyPepeQKZ+9U1Bn09t10NLZQH+Z509jWZgvNJH/aOh85P6euw==", + "node_modules/@angular-devkit/build-angular/node_modules/@babel/template": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.12.7.tgz", + "integrity": "sha512-GkDzmHS6GV7ZeXfJZ0tLRBhZcMcY0/Lnb+eEbXDBfCAcZCjrZKe6p3J4we/D24O9Y8enxWAg1cWwof59yLh2ow==", "dev": true, - "requires": { + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/parser": "^7.12.7", + "@babel/types": "^7.12.7" + } + }, + "node_modules/@angular-devkit/build-angular/node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/@angular-devkit/build-angular/node_modules/https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "dev": true, + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@angular-devkit/build-angular/node_modules/semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular-devkit/build-optimizer": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-optimizer/-/build-optimizer-0.1102.14.tgz", + "integrity": "sha512-1j69rFqE6tPMO0lQvOH8ogF7vE+p+Ws1/OtdZKUkZPOerIbQ8A3n5wzCx6/ZzMVhBQ3sXNhaShb4b9/1YuwU/g==", + "dev": true, + "dependencies": { "loader-utils": "2.0.0", "source-map": "0.7.3", - "tslib": "2.0.1", - "typescript": "4.0.2", - "webpack-sources": "1.4.3" + "tslib": "2.1.0", + "typescript": "4.1.5", + "webpack-sources": "2.2.0" }, - "dependencies": { - "tslib": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.1.tgz", - "integrity": "sha512-SgIkNheinmEBgx1IUNirK0TUD4X9yjjBRTqqjggWCU3pUEqIk3/Uwl3yRixYKT6WjQuGiwDv4NomL3wqRCj+CQ==", - "dev": true - }, - "typescript": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.0.2.tgz", - "integrity": "sha512-e4ERvRV2wb+rRZ/IQeb3jm2VxBsirQLpQhdxplZ2MEzGvDkkMmPglecnNDfSUBivMjP93vRbngYYDQqQ/78bcQ==", - "dev": true - } + "bin": { + "build-optimizer": "src/build-optimizer/cli.js" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" } }, - "@angular-devkit/build-webpack": { - "version": "0.1002.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.1002.0.tgz", - "integrity": "sha512-TLBBQ6ANOLKXOPxpCOnxAtoknwHA7XhsLuueN06w5qqF+QNNbWUMPoieKFGs2TnotfCgbiq6x57IDEZTyT6V0w==", + "node_modules/@angular-devkit/build-optimizer/node_modules/tslib": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.1.0.tgz", + "integrity": "sha512-hcVC3wYEziELGGmEEXue7D75zbwIIVUMWAVbHItGPx0ziyXxrOMQx4rQEVEV45Ut/1IotuEvwqPopzIOkDMf0A==", + "dev": true + }, + "node_modules/@angular-devkit/build-webpack": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.1102.14.tgz", + "integrity": "sha512-+dJvzrwjbHY0bNr8fUDVbn4D4pAT/h1YVpGVyaoX7q66LN0x61zRC3e10gJ/Mr54l3yfc26M0OPD9KG8iZRbCA==", "dev": true, - "requires": { - "@angular-devkit/architect": "0.1002.0", - "@angular-devkit/core": "10.2.0", - "rxjs": "6.6.2" - }, "dependencies": { - "@angular-devkit/architect": { - "version": "0.1002.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1002.0.tgz", - "integrity": "sha512-twM8V03ujBIGVpgV1PBlSDodUdxtUb7WakutfWafAvEHUsgwzfvQz2VtKWvjNZ9AiYjnCuwkQaclqVv0VHNo9w==", - "dev": true, - "requires": { - "@angular-devkit/core": "10.2.0", - "rxjs": "6.6.2" - } - }, - "@angular-devkit/core": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-10.2.0.tgz", - "integrity": "sha512-XAszFhSF3mZw1VjoOsYGbArr5NJLcStjOvcCGjBPl1UBM2AKpuCQXHxI9XJGYKL3B93Vp5G58d8qkHvamT53OA==", - "dev": true, - "requires": { - "ajv": "6.12.4", - "fast-json-stable-stringify": "2.1.0", - "magic-string": "0.25.7", - "rxjs": "6.6.2", - "source-map": "0.7.3" - } - }, - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true - } + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "rxjs": "6.6.3" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + }, + "peerDependencies": { + "webpack": "^4.6.0", + "webpack-dev-server": "^3.1.4" } }, - "@angular-devkit/core": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-10.1.5.tgz", - "integrity": "sha512-Ly97h90Z6ZLhSnTkk2baUDNLeOrKgj/bUPkcBEKWranx6IRx8FMzin/+ysIQasBlEXWPIc8QbBmCz7xXkO4p7g==", + "node_modules/@angular-devkit/core": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-11.2.14.tgz", + "integrity": "sha512-Ad1fHqLxDwhkQgLPqq9i+G65NSOoIHXQx7ILcSPACKurV3XLS1RO9BgP/BDaqHAG+WslUAPbMStaTzzPm+9dNw==", "dev": true, - "requires": { - "ajv": "6.12.4", + "dependencies": { + "ajv": "6.12.6", "fast-json-stable-stringify": "2.1.0", "magic-string": "0.25.7", - "rxjs": "6.6.2", + "rxjs": "6.6.3", "source-map": "0.7.3" }, - "dependencies": { - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "tslib": { - "version": "1.14.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", - "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", - "dev": true - } + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" } }, - "@angular-devkit/schematics": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-10.1.5.tgz", - "integrity": "sha512-5bhQX/PC548wIPcgCx9Q0Oewe8/i8+0eZvD9qLVWzJvUEKqgbjgoA7r7KJIJx2WINbESJGTIjwbXSZ6JmAJNhA==", + "node_modules/@angular-devkit/schematics": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-11.2.14.tgz", + "integrity": "sha512-Ol6+0qdGKzuVJm5gCtQr47X0OCihTfAxI4h047cHYhPFIGGPSvkG/QeJMZugflgoobi2k/xcYokOu/VAkRtWbQ==", "dev": true, - "requires": { - "@angular-devkit/core": "10.1.5", - "ora": "5.0.0", - "rxjs": "6.6.2" + "dependencies": { + "@angular-devkit/core": "11.2.14", + "ora": "5.3.0", + "rxjs": "6.6.3" }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@angular/animations": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/animations/-/animations-11.2.14.tgz", + "integrity": "sha512-Heq/nNrCmb3jbkusu+BQszOecfFI/31Oxxj+CDQkqqYpBcswk6bOJLoEE472o+vmgxaXbgeflU9qbIiCQhpMFA==", "dependencies": { - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "tslib": { - "version": "1.14.0", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", - "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", - "dev": true - } - } - }, - "@angular/animations": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/animations/-/animations-10.1.5.tgz", - "integrity": "sha512-RbUIluxgE5pSWWdODlcEAQuRqc/D1A2v275zBsMFjwJg3/cZl/z+RWcFJedHpJHEtbz7Aay1UWHu9jhXfA8elg==", - "requires": { "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/core": "11.2.14" } }, - "@angular/cli": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-10.1.5.tgz", - "integrity": "sha512-HlJVDxuTfrmxp8CvABV1pn7Ffeo0q0PuAR7gNCDcVi2vN7EDmBRRnyxBvATO4KzE5DHiSIqF0xLIsokSS7JC6w==", + "node_modules/@angular/cli": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-11.2.14.tgz", + "integrity": "sha512-8Ud7vcUK7CKjzT2Ks1glLhleAPIC5ChcrA15XtOb7k+/uMHBkMscP/UKymbVQiBjCJlglbzJoyj8cpVYTZY5KA==", "dev": true, - "requires": { - "@angular-devkit/architect": "0.1001.5", - "@angular-devkit/core": "10.1.5", - "@angular-devkit/schematics": "10.1.5", - "@schematics/angular": "10.1.5", - "@schematics/update": "0.1001.5", + "hasInstallScript": true, + "dependencies": { + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", + "@schematics/angular": "11.2.14", + "@schematics/update": "0.1102.14", "@yarnpkg/lockfile": "1.1.0", "ansi-colors": "4.1.1", - "debug": "4.1.1", - "ini": "1.3.5", + "debug": "4.3.1", + "ini": "2.0.0", "inquirer": "7.3.3", - "npm-package-arg": "8.0.1", + "jsonc-parser": "3.0.0", + "npm-package-arg": "8.1.0", "npm-pick-manifest": "6.1.0", - "open": "7.2.0", - "pacote": "9.5.12", - "read-package-tree": "5.3.1", + "open": "7.4.0", + "ora": "5.3.0", + "pacote": "11.2.4", + "resolve": "1.19.0", "rimraf": "3.0.2", - "semver": "7.3.2", - "symbol-observable": "1.2.0", + "semver": "7.3.4", + "symbol-observable": "3.0.0", "universal-analytics": "0.4.23", - "uuid": "8.3.0" + "uuid": "8.3.2" }, - "dependencies": { - "ansi-colors": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", - "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", - "dev": true - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "uuid": { - "version": "8.3.0", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.0.tgz", - "integrity": "sha512-fX6Z5o4m6XsXBdli9g7DtWgAx+osMsRRZFKma1mIUsLCz6vRvv+pz5VNbyu9UEDzpMWulZfvpgb/cmDXVulYFQ==", - "dev": true - } + "bin": { + "ng": "bin/ng" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" } }, - "@angular/common": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/common/-/common-10.1.5.tgz", - "integrity": "sha512-xo10mSQYuf6x1XrnTfwt3Rs7JtSMkSyrJtAS/vNQKdBP/8zmn6pP9zRpp7vhQ5qF+W3HN8rPLb+YI2F6uaGjBg==", - "requires": { - "tslib": "^2.0.0" - } - }, - "@angular/compiler": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-10.1.5.tgz", - "integrity": "sha512-3LyFkEzs6P6YYKkE/6E4PasMd58EBddOt9kR9kPmj9Atv/BLY3nc5RSWkOe4rK4GnBVP+ByzQiT9Fn5CiQnG/g==", - "requires": { - "tslib": "^2.0.0" - } - }, - "@angular/compiler-cli": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-10.1.5.tgz", - "integrity": "sha512-AJ4eOHUxgDdfq/EagUlhJ6HaNlHajtmPkhXp2HmNMNN1nPN55VZSvN43Co2gdAHiFENqsTNlnQH630aXaDyVbQ==", + "node_modules/@angular/cli/node_modules/ini": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz", + "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==", "dev": true, - "requires": { + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/cli/node_modules/semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/common": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/common/-/common-11.2.14.tgz", + "integrity": "sha512-ZSLV/3j7eCTyLf/8g4yBFLWySjiLz3vLJAGWscYoUpnJWMnug1VRu6zoF/COxCbtORgE+Wz6K0uhfS6MziBGVw==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/core": "11.2.14", + "rxjs": "^6.5.3" + } + }, + "node_modules/@angular/compiler": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-11.2.14.tgz", + "integrity": "sha512-XBOK3HgA+/y6Cz7kOX4zcJYmgJ264XnfcbXUMU2cD7Ac+mbNhLPKohWrEiSWalfcjnpf5gRfufQrQP7lpAGu0A==", + "dependencies": { + "tslib": "^2.0.0" + } + }, + "node_modules/@angular/compiler-cli": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-11.2.14.tgz", + "integrity": "sha512-A7ltnCp03/EVqK/Q3tVUDsokgz5GHW3dSPGl0Csk7Ys5uBB9ibHTmVt4eiXA4jt0+6Bk+mKxwe5BEDqLvwYFAg==", + "dependencies": { + "@babel/core": "^7.8.6", + "@babel/types": "^7.8.6", "canonical-path": "1.0.0", "chokidar": "^3.0.0", "convert-source-map": "^1.5.1", @@ -373,46 +433,17015 @@ "source-map": "^0.6.1", "sourcemap-codec": "^1.4.8", "tslib": "^2.0.0", - "yargs": "15.3.0" + "yargs": "^16.2.0" + }, + "bin": { + "ivy-ngcc": "ngcc/main-ivy-ngcc.js", + "ng-xi18n": "src/extract_i18n.js", + "ngc": "src/main.js", + "ngcc": "ngcc/main-ngcc.js" + }, + "engines": { + "node": ">=10.0" + }, + "peerDependencies": { + "@angular/compiler": "11.2.14", + "typescript": ">=4.0 <4.2" + } + }, + "node_modules/@angular/compiler-cli/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/compiler-cli/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@angular/compiler-cli/node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/@angular/compiler-cli/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@angular/compiler-cli/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "node_modules/@angular/compiler-cli/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/@angular/compiler-cli/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/compiler-cli/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@angular/compiler-cli/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==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@angular/compiler-cli/node_modules/string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/compiler-cli/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/compiler-cli/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/@angular/compiler-cli/node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/compiler-cli/node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/compiler-cli/node_modules/yargs-parser": { + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/core": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/core/-/core-11.2.14.tgz", + "integrity": "sha512-vpR4XqBGitk1Faph37CSpemwIYTmJ3pdIVNoHKP6jLonpWu+0azkchf0f7oD8/2ivj2F81opcIw0tcsy/D/5Vg==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "rxjs": "^6.5.3", + "zone.js": "^0.10.2 || ^0.11.3" + } + }, + "node_modules/@angular/forms": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-11.2.14.tgz", + "integrity": "sha512-4LWqY6KEIk1AZQFnk+4PJSOCamlD4tumuVN06gO4D0dZo9Cx+GcvW6pM6N0CPubRvPs3sScCnu20WT11HNWC1w==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": "11.2.14", + "@angular/core": "11.2.14", + "@angular/platform-browser": "11.2.14", + "rxjs": "^6.5.3" + } + }, + "node_modules/@angular/localize": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-11.2.14.tgz", + "integrity": "sha512-ssMuquxxqxA98LgEICEO/3JdmSflWxu5rlm/HPo28bnGiZ4IzDamZjJ1cu4S6RgsonJ1drB3Z8wkidXfEYZiWA==", + "dependencies": { + "@babel/core": "7.8.3", + "glob": "7.1.2", + "yargs": "^16.2.0" + }, + "bin": { + "localize-extract": "src/tools/src/extract/main.js", + "localize-translate": "src/tools/src/translate/main.js" + }, + "engines": { + "node": ">=8.0" + }, + "peerDependencies": { + "@angular/compiler": "11.2.14", + "@angular/compiler-cli": "11.2.14" + } + }, + "node_modules/@angular/localize/node_modules/@babel/core": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.8.3.tgz", + "integrity": "sha512-4XFkf8AwyrEG7Ziu3L2L0Cv+WyY47Tcsp70JFmpftbAA1K7YL/sgE9jh9HyNj08Y/U50ItUchpN0w6HxAoX1rA==", + "dependencies": { + "@babel/code-frame": "^7.8.3", + "@babel/generator": "^7.8.3", + "@babel/helpers": "^7.8.3", + "@babel/parser": "^7.8.3", + "@babel/template": "^7.8.3", + "@babel/traverse": "^7.8.3", + "@babel/types": "^7.8.3", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.1", + "json5": "^2.1.0", + "lodash": "^4.17.13", + "resolve": "^1.3.2", + "semver": "^5.4.1", + "source-map": "^0.5.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/babel" + } + }, + "node_modules/@angular/localize/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/localize/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/@angular/localize/node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/@angular/localize/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/@angular/localize/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" + }, + "node_modules/@angular/localize/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" + }, + "node_modules/@angular/localize/node_modules/glob": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", + "integrity": "sha512-MJTUg1kjuLeQCJ+ccE4Vpa6kKVXkPYJ2mOCQyUuKLcLQsdrMCpBPUi8qVE6+YuaJkozeA9NusTAw3hLr8Xe5EQ==", + "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": "*" + } + }, + "node_modules/@angular/localize/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/localize/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/@angular/localize/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@angular/localize/node_modules/string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/localize/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/@angular/localize/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/@angular/localize/node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/localize/node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/localize/node_modules/yargs-parser": { + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@angular/platform-browser": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-11.2.14.tgz", + "integrity": "sha512-fb7b7ss/gRoP8wLAN17W62leMgjynuyjEPU2eUoAAazsG9f2cgM+z3rK29GYncDVyYQxZUZYnjSqvL6GSXx86A==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/animations": "11.2.14", + "@angular/common": "11.2.14", + "@angular/core": "11.2.14" + }, + "peerDependenciesMeta": { + "@angular/animations": { + "optional": true + } + } + }, + "node_modules/@angular/platform-browser-dynamic": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-11.2.14.tgz", + "integrity": "sha512-TWTPdFs6iBBcp+/YMsgCRQwdHpWGq8KjeJDJ2tfatGgBD3Gqt2YaHOMST1zPW6RkrmupytTejuVqXzeaKWFxuw==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": "11.2.14", + "@angular/compiler": "11.2.14", + "@angular/core": "11.2.14", + "@angular/platform-browser": "11.2.14" + } + }, + "node_modules/@angular/router": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/router/-/router-11.2.14.tgz", + "integrity": "sha512-3aYBmj+zrEL9yf/ntIQxHIYaWShZOBKP3U07X2mX+TPMpGlvHDnR7L6bWhQVZwewzMMz7YVR16ldg50IFuAlfA==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": "11.2.14", + "@angular/core": "11.2.14", + "@angular/platform-browser": "11.2.14", + "rxjs": "^6.5.3" + } + }, + "node_modules/@babel/code-frame": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.14.5.tgz", + "integrity": "sha512-9pzDqyc6OLDaqe+zbACgFkb6fKMNG6CObKpnYXChRsvYGyEdc7CA2BaqeOM+vOtCS5ndmJicPJhKAwYRI6UfFw==", + "dependencies": { + "@babel/highlight": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/compat-data": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.14.5.tgz", + "integrity": "sha512-kixrYn4JwfAVPa0f2yfzc2AWti6WRRyO3XjWW5PJAvtE11qhSayrrcrEnee05KAtNaPC+EwehE8Qt1UedEVB8w==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/core": { + "version": "7.12.10", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.12.10.tgz", + "integrity": "sha512-eTAlQKq65zHfkHZV0sIVODCPGVgoo1HdBlbSLi9CqOzuZanMv2ihzY+4paiKr1mH+XmYESMAmJ/dpZ68eN6d8w==", + "dependencies": { + "@babel/code-frame": "^7.10.4", + "@babel/generator": "^7.12.10", + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helpers": "^7.12.5", + "@babel/parser": "^7.12.10", + "@babel/template": "^7.12.7", + "@babel/traverse": "^7.12.10", + "@babel/types": "^7.12.10", + "convert-source-map": "^1.7.0", + "debug": "^4.1.0", + "gensync": "^1.0.0-beta.1", + "json5": "^2.1.2", + "lodash": "^4.17.19", + "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/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "bin": { + "semver": "bin/semver" + } + }, + "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=", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@babel/generator": { + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.11.tgz", + "integrity": "sha512-Ggg6WPOJtSi8yYQvLVjG8F/TlpWDlKx0OpS4Kt+xMQPs5OaGYWy+v1A+1TvxI6sAMGZpKWWoAQ1DaeQbImlItA==", + "dependencies": { + "@babel/types": "^7.12.11", + "jsesc": "^2.5.1", + "source-map": "^0.5.0" + } + }, + "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=", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@babel/helper-annotate-as-pure": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.14.5.tgz", + "integrity": "sha512-EivH9EgBIb+G8ij1B2jAwSH36WnGvkQSEC6CkX/6v6ZFlw5fVOHvsgGF4uiEHO2GzMvunZb6tDLQEQSdrdocrA==", + "dev": true, + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-builder-binary-assignment-operator-visitor": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.14.5.tgz", + "integrity": "sha512-YTA/Twn0vBXDVGJuAX6PwW7x5zQei1luDDo2Pl6q1qZ7hVNl0RZrhHCQG/ArGpR29Vl7ETiB8eJyrvpuRp300w==", + "dev": true, + "dependencies": { + "@babel/helper-explode-assignable-expression": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-compilation-targets": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.14.5.tgz", + "integrity": "sha512-v+QtZqXEiOnpO6EYvlImB6zCD2Lel06RzOPzmkz/D/XgQiUu3C/Jb1LOqSt/AIA34TYi/Q+KlT8vTQrgdxkbLw==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.14.5", + "@babel/helper-validator-option": "^7.14.5", + "browserslist": "^4.16.6", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-compilation-targets/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/@babel/helper-create-class-features-plugin": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.14.5.tgz", + "integrity": "sha512-Uq9z2e7ZtcnDMirRqAGLRaLwJn+Lrh388v5ETrR3pALJnElVh2zqQmdbz4W2RUJYohAPh2mtyPUgyMHMzXMncQ==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-member-expression-to-functions": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-create-regexp-features-plugin": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.14.5.tgz", + "integrity": "sha512-TLawwqpOErY2HhWbGJ2nZT5wSkR192QpN+nBg1THfBfftrlvOh+WbhrxXCH4q4xJ9Gl16BGPR/48JA+Ryiho/A==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.14.5", + "regexpu-core": "^4.7.1" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/helper-explode-assignable-expression": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.14.5.tgz", + "integrity": "sha512-Htb24gnGJdIGT4vnRKMdoXiOIlqOLmdiUYpAQ0mYfgVT/GDm8GOYhgi4GL+hMKrkiPRohO4ts34ELFsGAPQLDQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-function-name": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.14.5.tgz", + "integrity": "sha512-Gjna0AsXWfFvrAuX+VKcN/aNNWonizBj39yGwUzVDVTlMYJMK2Wp6xdpy72mfArFq5uK+NOuexfzZlzI1z9+AQ==", + "dependencies": { + "@babel/helper-get-function-arity": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-get-function-arity": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.14.5.tgz", + "integrity": "sha512-I1Db4Shst5lewOM4V+ZKJzQ0JGGaZ6VY1jYvMghRjqs6DWgxLCIyFt30GlnKkfUeFLpJt2vzbMVEXVSXlIFYUg==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-hoist-variables": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.14.5.tgz", + "integrity": "sha512-R1PXiz31Uc0Vxy4OEOm07x0oSjKAdPPCh3tPivn/Eo8cvz6gveAeuyUUPB21Hoiif0uoPQSSdhIPS3352nvdyQ==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-member-expression-to-functions": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.14.5.tgz", + "integrity": "sha512-UxUeEYPrqH1Q/k0yRku1JE7dyfyehNwT6SVkMHvYvPDv4+uu627VXBckVj891BO8ruKBkiDoGnZf4qPDD8abDQ==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-imports": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.14.5.tgz", + "integrity": "sha512-SwrNHu5QWS84XlHwGYPDtCxcA0hrSlL2yhWYLgeOc0w7ccOl2qv4s/nARI0aYZW+bSwAL5CukeXA47B/1NKcnQ==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-module-transforms": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.14.5.tgz", + "integrity": "sha512-iXpX4KW8LVODuAieD7MzhNjmM6dzYY5tfRqT+R9HDXWl0jPn/djKmA+G9s/2C2T9zggw5tK1QNqZ70USfedOwA==", + "dependencies": { + "@babel/helper-module-imports": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-simple-access": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", + "@babel/helper-validator-identifier": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-optimise-call-expression": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.14.5.tgz", + "integrity": "sha512-IqiLIrODUOdnPU9/F8ib1Fx2ohlgDhxnIDU7OEVi+kAbEZcyiF7BLU8W6PfvPi9LzztjS7kcbzbmL7oG8kD6VA==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-plugin-utils": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.14.5.tgz", + "integrity": "sha512-/37qQCE3K0vvZKwoK4XU/irIJQdIfCJuhU5eKnNxpFDsOkgFaUAwbv+RYw6eYgsC0E4hS7r5KqGULUogqui0fQ==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-remap-async-to-generator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.14.5.tgz", + "integrity": "sha512-rLQKdQU+HYlxBwQIj8dk4/0ENOUEhA/Z0l4hN8BexpvmSMN9oA9EagjnhnDpNsRdWCfjwa4mn/HyBXO9yhQP6A==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-wrap-function": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-replace-supers": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.14.5.tgz", + "integrity": "sha512-3i1Qe9/8x/hCHINujn+iuHy+mMRLoc77b2nI9TB0zjH1hvn9qGlXjWlggdwUcju36PkPCy/lpM7LLUdcTyH4Ow==", + "dependencies": { + "@babel/helper-member-expression-to-functions": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-simple-access": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.14.5.tgz", + "integrity": "sha512-nfBN9xvmCt6nrMZjfhkl7i0oTV3yxR4/FztsbOASyTvVcoYd0TRHh7eMLdlEcCqobydC0LAF3LtC92Iwxo0wyw==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-skip-transparent-expression-wrappers": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.14.5.tgz", + "integrity": "sha512-dmqZB7mrb94PZSAOYtr+ZN5qt5owZIAgqtoTuqiFbHFtxgEcmQlRJVI+bO++fciBunXtB6MK7HrzrfcAzIz2NQ==", + "dev": true, + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-split-export-declaration": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.14.5.tgz", + "integrity": "sha512-hprxVPu6e5Kdp2puZUmvOGjaLv9TCe58E/Fl6hRq4YiVQxIcNvuq6uTM2r1mT/oPskuS9CgR+I94sqAYv0NGKA==", + "dependencies": { + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-identifier": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.14.5.tgz", + "integrity": "sha512-5lsetuxCLilmVGyiLEfoHBRX8UCFD+1m2x3Rj97WrW3V7H3u4RWRXA4evMjImCsin2J2YT0QaVDGf+z8ondbAg==", + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-validator-option": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.14.5.tgz", + "integrity": "sha512-OX8D5eeX4XwcroVW45NMvoYaIuFI+GQpA2a8Gi+X/U/cDUIRsV37qQfF905F0htTRCREQIB4KqPeaveRJUl3Ow==", + "dev": true, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helper-wrap-function": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.14.5.tgz", + "integrity": "sha512-YEdjTCq+LNuNS1WfxsDCNpgXkJaIyqco6DAelTUjT4f2KIWC1nBcaCaSdHTBqQVLnTBexBcVcFhLSU1KnYuePQ==", + "dev": true, + "dependencies": { + "@babel/helper-function-name": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/helpers": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.14.5.tgz", + "integrity": "sha512-xtcWOuN9VL6nApgVHtq3PPcQv5qFBJzoSZzJ/2c0QK/IP/gxVcoWSNQwFEGvmbQsuS9rhYqjILDGGXcTkA705Q==", + "dependencies": { + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/highlight": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", + "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", + "dependencies": { + "@babel/helper-validator-identifier": "^7.14.5", + "chalk": "^2.0.0", + "js-tokens": "^4.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/parser": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.14.5.tgz", + "integrity": "sha512-TM8C+xtH/9n1qzX+JNHi7AN2zHMTiPUtspO0ZdHflW8KaskkALhMmuMHb4bCmNdv9VAPzJX3/bXqkVLnAvsPfg==", + "bin": { + "parser": "bin/babel-parser.js" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/@babel/plugin-proposal-async-generator-functions": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.14.5.tgz", + "integrity": "sha512-tbD/CG3l43FIXxmu4a7RBe4zH7MLJ+S/lFowPFO7HetS2hyOZ/0nnnznegDuzFzfkyQYTxqdTH/hKmuBngaDAA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-remap-async-to-generator": "^7.14.5", + "@babel/plugin-syntax-async-generators": "^7.8.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-class-properties": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.14.5.tgz", + "integrity": "sha512-q/PLpv5Ko4dVc1LYMpCY7RVAAO4uk55qPwrIuJ5QJ8c6cVuAmhu7I/49JOppXL6gXf7ZHzpRVEUZdYoPLM04Gg==", + "dev": true, + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-dynamic-import": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.14.5.tgz", + "integrity": "sha512-ExjiNYc3HDN5PXJx+bwC50GIx/KKanX2HiggnIUAYedbARdImiCU4RhhHfdf0Kd7JNXGpsBBBCOm+bBVy3Gb0g==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-dynamic-import": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-export-namespace-from": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.14.5.tgz", + "integrity": "sha512-g5POA32bXPMmSBu5Dx/iZGLGnKmKPc5AiY7qfZgurzrCYgIztDlHFbznSNCoQuv57YQLnQfaDi7dxCtLDIdXdA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-json-strings": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.14.5.tgz", + "integrity": "sha512-NSq2fczJYKVRIsUJyNxrVUMhB27zb7N7pOFGQOhBKJrChbGcgEAqyZrmZswkPk18VMurEeJAaICbfm57vUeTbQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-json-strings": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-logical-assignment-operators": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.14.5.tgz", + "integrity": "sha512-YGn2AvZAo9TwyhlLvCCWxD90Xq8xJ4aSgaX3G5D/8DW94L8aaT+dS5cSP+Z06+rCJERGSr9GxMBZ601xoc2taw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-nullish-coalescing-operator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.14.5.tgz", + "integrity": "sha512-gun/SOnMqjSb98Nkaq2rTKMwervfdAoz6NphdY0vTfuzMfryj+tDGb2n6UkDKwez+Y8PZDhE3D143v6Gepp4Hg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-numeric-separator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.14.5.tgz", + "integrity": "sha512-yiclALKe0vyZRZE0pS6RXgjUOt87GWv6FYa5zqj15PvhOGFO69R5DusPlgK/1K5dVnCtegTiWu9UaBSrLLJJBg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-numeric-separator": "^7.10.4" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-object-rest-spread": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.14.5.tgz", + "integrity": "sha512-VzMyY6PWNPPT3pxc5hi9LloKNr4SSrVCg7Yr6aZpW4Ym07r7KqSU/QXYwjXLVxqwSv0t/XSXkFoKBPUkZ8vb2A==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.14.5", + "@babel/helper-compilation-targets": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-transform-parameters": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-optional-catch-binding": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.14.5.tgz", + "integrity": "sha512-3Oyiixm0ur7bzO5ybNcZFlmVsygSIQgdOa7cTfOYCMY+wEPAYhZAJxi3mixKFCTCKUhQXuCTtQ1MzrpL3WT8ZQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-optional-chaining": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.14.5.tgz", + "integrity": "sha512-ycz+VOzo2UbWNI1rQXxIuMOzrDdHGrI23fRiz/Si2R4kv2XZQ1BK8ccdHwehMKBlcH/joGW/tzrUmo67gbJHlQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.14.5", + "@babel/plugin-syntax-optional-chaining": "^7.8.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-private-methods": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.14.5.tgz", + "integrity": "sha512-838DkdUA1u+QTCplatfq4B7+1lnDa/+QMI89x5WZHBcnNv+47N8QEj2k9I2MUU9xIv8XJ4XvPCviM/Dj7Uwt9g==", + "dev": true, + "dependencies": { + "@babel/helper-create-class-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-proposal-unicode-property-regex": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.14.5.tgz", + "integrity": "sha512-6axIeOU5LnY471KenAB9vI8I5j7NQ2d652hIYwVyRfgaZT5UpiqFKCuVXCDMSrU+3VFafnu2c5m3lrWIlr6A5Q==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-class-properties": { + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.12.13" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-export-namespace-from": { + "version": "7.8.3", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-export-namespace-from/-/plugin-syntax-export-namespace-from-7.8.3.tgz", + "integrity": "sha512-MXf5laXo6c1IbEbegDmzGPwGNTsHZmEy6QGznu5Sh2UCWvueywb2ee+CCE4zQiZstxU9BMoQO9i6zUFSY0Kj0Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.3" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-logical-assignment-operators": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-logical-assignment-operators/-/plugin-syntax-logical-assignment-operators-7.10.4.tgz", + "integrity": "sha512-d8waShlpFDinQ5MtvGU9xDAOzKH47+FFoney2baFIoMr952hKOLp1HR7VszoZvOsV/4+RRszNY7D17ba0te0ig==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-numeric-separator": { + "version": "7.10.4", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-numeric-separator/-/plugin-syntax-numeric-separator-7.10.4.tgz", + "integrity": "sha512-9H6YdfkcK/uOnY/K7/aA2xpzaAgkQn37yzWUMRK7OaPOqOpGS1+n0H5hxT9AUw9EsSjPW8SVyMJwYRtWs3X3ug==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.10.4" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "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, + "dependencies": { + "@babel/helper-plugin-utils": "^7.8.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-syntax-top-level-await": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-arrow-functions": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.14.5.tgz", + "integrity": "sha512-KOnO0l4+tD5IfOdi4x8C1XmEIRWUjNRV8wc6K2vz/3e8yAOoZZvsRXRRIF/yo/MAOFb4QjtAw9xSxMXbSMRy8A==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-async-to-generator": { + "version": "7.12.1", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.12.1.tgz", + "integrity": "sha512-SDtqoEcarK1DFlRJ1hHRY5HvJUj5kX4qmtpMAm2QnhOlyuMC4TMdCRgW6WXpv93rZeYNeLP22y8Aq2dbcDRM1A==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.12.1", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-remap-async-to-generator": "^7.12.1" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoped-functions": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.14.5.tgz", + "integrity": "sha512-dtqWqdWZ5NqBX3KzsVCWfQI3A53Ft5pWFCT2eCVUftWZgjc5DpDponbIF1+c+7cSGk2wN0YK7HGL/ezfRbpKBQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-block-scoping": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.14.5.tgz", + "integrity": "sha512-LBYm4ZocNgoCqyxMLoOnwpsmQ18HWTQvql64t3GvMUzLQrNoV1BDG0lNftC8QKYERkZgCCT/7J5xWGObGAyHDw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-classes": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.14.5.tgz", + "integrity": "sha512-J4VxKAMykM06K/64z9rwiL6xnBHgB1+FVspqvlgCdwD1KUbQNfszeKVVOMh59w3sztHYIZDgnhOC4WbdEfHFDA==", + "dev": true, + "dependencies": { + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-computed-properties": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.14.5.tgz", + "integrity": "sha512-pWM+E4283UxaVzLb8UBXv4EIxMovU4zxT1OPnpHJcmnvyY9QbPPTKZfEj31EUvG3/EQRbYAGaYEUZ4yWOBC2xg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-destructuring": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.14.5.tgz", + "integrity": "sha512-wU9tYisEbRMxqDezKUqC9GleLycCRoUsai9ddlsq54r8QRLaeEhc+d+9DqCG+kV9W2GgQjTZESPTpn5bAFMDww==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-dotall-regex": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.14.5.tgz", + "integrity": "sha512-loGlnBdj02MDsFaHhAIJzh7euK89lBrGIdM9EAtHFo6xKygCUGuuWe07o1oZVk287amtW1n0808sQM99aZt3gw==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-duplicate-keys": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.14.5.tgz", + "integrity": "sha512-iJjbI53huKbPDAsJ8EmVmvCKeeq21bAze4fu9GBQtSLqfvzj2oRuHVx4ZkDwEhg1htQ+5OBZh/Ab0XDf5iBZ7A==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-exponentiation-operator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.14.5.tgz", + "integrity": "sha512-jFazJhMBc9D27o9jDnIE5ZErI0R0m7PbKXVq77FFvqFbzvTMuv8jaAwLZ5PviOLSFttqKIW0/wxNSDbjLk0tYA==", + "dev": true, + "dependencies": { + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-for-of": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.14.5.tgz", + "integrity": "sha512-CfmqxSUZzBl0rSjpoQSFoR9UEj3HzbGuGNL21/iFTmjb5gFggJp3ph0xR1YBhexmLoKRHzgxuFvty2xdSt6gTA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-function-name": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.14.5.tgz", + "integrity": "sha512-vbO6kv0fIzZ1GpmGQuvbwwm+O4Cbm2NrPzwlup9+/3fdkuzo1YqOZcXw26+YUJB84Ja7j9yURWposEHLYwxUfQ==", + "dev": true, + "dependencies": { + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-literals": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.14.5.tgz", + "integrity": "sha512-ql33+epql2F49bi8aHXxvLURHkxJbSmMKl9J5yHqg4PLtdE6Uc48CH1GS6TQvZ86eoB/ApZXwm7jlA+B3kra7A==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-member-expression-literals": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.14.5.tgz", + "integrity": "sha512-WkNXxH1VXVTKarWFqmso83xl+2V3Eo28YY5utIkbsmXoItO8Q3aZxN4BTS2k0hz9dGUloHK26mJMyQEYfkn/+Q==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-amd": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.14.5.tgz", + "integrity": "sha512-3lpOU8Vxmp3roC4vzFpSdEpGUWSMsHFreTWOMMLzel2gNGfHE5UWIh/LN6ghHs2xurUp4jRFYMUIZhuFbody1g==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-commonjs": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.14.5.tgz", + "integrity": "sha512-en8GfBtgnydoao2PS+87mKyw62k02k7kJ9ltbKe0fXTHrQmG6QZZflYuGI1VVG7sVpx4E1n7KBpNlPb8m78J+A==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-simple-access": "^7.14.5", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-systemjs": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.14.5.tgz", + "integrity": "sha512-mNMQdvBEE5DcMQaL5LbzXFMANrQjd2W7FPzg34Y4yEz7dBgdaC+9B84dSO+/1Wba98zoDbInctCDo4JGxz1VYA==", + "dev": true, + "dependencies": { + "@babel/helper-hoist-variables": "^7.14.5", + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-validator-identifier": "^7.14.5", + "babel-plugin-dynamic-import-node": "^2.3.3" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-modules-umd": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.14.5.tgz", + "integrity": "sha512-RfPGoagSngC06LsGUYyM9QWSXZ8MysEjDJTAea1lqRjNECE3y0qIJF/qbvJxc4oA4s99HumIMdXOrd+TdKaAAA==", + "dev": true, + "dependencies": { + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-named-capturing-groups-regex": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.14.5.tgz", + "integrity": "sha512-+Xe5+6MWFo311U8SchgeX5c1+lJM+eZDBZgD+tvXu9VVQPXwwVzeManMMjYX6xw2HczngfOSZjoFYKwdeB/Jvw==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0" + } + }, + "node_modules/@babel/plugin-transform-new-target": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.14.5.tgz", + "integrity": "sha512-Nx054zovz6IIRWEB49RDRuXGI4Gy0GMgqG0cII9L3MxqgXz/+rgII+RU58qpo4g7tNEx1jG7rRVH4ihZoP4esQ==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-object-super": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.14.5.tgz", + "integrity": "sha512-MKfOBWzK0pZIrav9z/hkRqIk/2bTv9qvxHzPQc12RcVkMOzpIKnFCNYJip00ssKWYkd8Sf5g0Wr7pqJ+cmtuFg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-parameters": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.14.5.tgz", + "integrity": "sha512-Tl7LWdr6HUxTmzQtzuU14SqbgrSKmaR77M0OKyq4njZLQTPfOvzblNKyNkGwOfEFCEx7KeYHQHDI0P3F02IVkA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-property-literals": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.14.5.tgz", + "integrity": "sha512-r1uilDthkgXW8Z1vJz2dKYLV1tuw2xsbrp3MrZmD99Wh9vsfKoob+JTgri5VUb/JqyKRXotlOtwgu4stIYCmnw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-regenerator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.14.5.tgz", + "integrity": "sha512-NVIY1W3ITDP5xQl50NgTKlZ0GrotKtLna08/uGY6ErQt6VEQZXla86x/CTddm5gZdcr+5GSsvMeTmWA5Ii6pkg==", + "dev": true, + "dependencies": { + "regenerator-transform": "^0.14.2" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-reserved-words": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.14.5.tgz", + "integrity": "sha512-cv4F2rv1nD4qdexOGsRQXJrOcyb5CrgjUH9PKrrtyhSDBNWGxd0UIitjyJiWagS+EbUGjG++22mGH1Pub8D6Vg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime": { + "version": "7.12.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.12.10.tgz", + "integrity": "sha512-xOrUfzPxw7+WDm9igMgQCbO3cJKymX7dFdsgRr1eu9n3KjjyU4pptIXbXPseQDquw+W+RuJEJMHKHNsPNNm3CA==", + "dev": true, + "dependencies": { + "@babel/helper-module-imports": "^7.12.5", + "@babel/helper-plugin-utils": "^7.10.4", + "semver": "^5.5.1" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-runtime/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/@babel/plugin-transform-shorthand-properties": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.14.5.tgz", + "integrity": "sha512-xLucks6T1VmGsTB+GWK5Pl9Jl5+nRXD1uoFdA5TSO6xtiNjtXTjKkmPdFXVLGlK5A2/or/wQMKfmQ2Y0XJfn5g==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-spread": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.14.5.tgz", + "integrity": "sha512-/3iqoQdiWergnShZYl0xACb4ADeYCJ7X/RgmwtXshn6cIvautRPAFzhd58frQlokLO6Jb4/3JXvmm6WNTPtiTw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-sticky-regex": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.14.5.tgz", + "integrity": "sha512-Z7F7GyvEMzIIbwnziAZmnSNpdijdr4dWt+FJNBnBLz5mwDFkqIXU9wmBcWWad3QeJF5hMTkRe4dAq2sUZiG+8A==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-template-literals": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.14.5.tgz", + "integrity": "sha512-22btZeURqiepOfuy/VkFr+zStqlujWaarpMErvay7goJS6BWwdd6BY9zQyDLDa4x2S3VugxFb162IZ4m/S/+Gg==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-typeof-symbol": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.14.5.tgz", + "integrity": "sha512-lXzLD30ffCWseTbMQzrvDWqljvZlHkXU+CnseMhkMNqU1sASnCsz3tSzAaH3vCUXb9PHeUb90ZT1BdFTm1xxJw==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-escapes": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.14.5.tgz", + "integrity": "sha512-crTo4jATEOjxj7bt9lbYXcBAM3LZaUrbP2uUdxb6WIorLmjNKSpHfIybgY4B8SRpbf8tEVIWH3Vtm7ayCrKocA==", + "dev": true, + "dependencies": { + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/plugin-transform-unicode-regex": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.14.5.tgz", + "integrity": "sha512-UygduJpC5kHeCiRw/xDVzC+wj8VaYSoKl5JNVmbP7MadpNinAm3SvZCxZ42H37KZBKztz46YC73i9yV34d0Tzw==", + "dev": true, + "dependencies": { + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-env": { + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.12.11.tgz", + "integrity": "sha512-j8Tb+KKIXKYlDBQyIOy4BLxzv1NUOwlHfZ74rvW+Z0Gp4/cI2IMDPBWAgWceGcE7aep9oL/0K9mlzlMGxA8yNw==", + "dev": true, + "dependencies": { + "@babel/compat-data": "^7.12.7", + "@babel/helper-compilation-targets": "^7.12.5", + "@babel/helper-module-imports": "^7.12.5", + "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-validator-option": "^7.12.11", + "@babel/plugin-proposal-async-generator-functions": "^7.12.1", + "@babel/plugin-proposal-class-properties": "^7.12.1", + "@babel/plugin-proposal-dynamic-import": "^7.12.1", + "@babel/plugin-proposal-export-namespace-from": "^7.12.1", + "@babel/plugin-proposal-json-strings": "^7.12.1", + "@babel/plugin-proposal-logical-assignment-operators": "^7.12.1", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.12.1", + "@babel/plugin-proposal-numeric-separator": "^7.12.7", + "@babel/plugin-proposal-object-rest-spread": "^7.12.1", + "@babel/plugin-proposal-optional-catch-binding": "^7.12.1", + "@babel/plugin-proposal-optional-chaining": "^7.12.7", + "@babel/plugin-proposal-private-methods": "^7.12.1", + "@babel/plugin-proposal-unicode-property-regex": "^7.12.1", + "@babel/plugin-syntax-async-generators": "^7.8.0", + "@babel/plugin-syntax-class-properties": "^7.12.1", + "@babel/plugin-syntax-dynamic-import": "^7.8.0", + "@babel/plugin-syntax-export-namespace-from": "^7.8.3", + "@babel/plugin-syntax-json-strings": "^7.8.0", + "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0", + "@babel/plugin-syntax-numeric-separator": "^7.10.4", + "@babel/plugin-syntax-object-rest-spread": "^7.8.0", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.0", + "@babel/plugin-syntax-optional-chaining": "^7.8.0", + "@babel/plugin-syntax-top-level-await": "^7.12.1", + "@babel/plugin-transform-arrow-functions": "^7.12.1", + "@babel/plugin-transform-async-to-generator": "^7.12.1", + "@babel/plugin-transform-block-scoped-functions": "^7.12.1", + "@babel/plugin-transform-block-scoping": "^7.12.11", + "@babel/plugin-transform-classes": "^7.12.1", + "@babel/plugin-transform-computed-properties": "^7.12.1", + "@babel/plugin-transform-destructuring": "^7.12.1", + "@babel/plugin-transform-dotall-regex": "^7.12.1", + "@babel/plugin-transform-duplicate-keys": "^7.12.1", + "@babel/plugin-transform-exponentiation-operator": "^7.12.1", + "@babel/plugin-transform-for-of": "^7.12.1", + "@babel/plugin-transform-function-name": "^7.12.1", + "@babel/plugin-transform-literals": "^7.12.1", + "@babel/plugin-transform-member-expression-literals": "^7.12.1", + "@babel/plugin-transform-modules-amd": "^7.12.1", + "@babel/plugin-transform-modules-commonjs": "^7.12.1", + "@babel/plugin-transform-modules-systemjs": "^7.12.1", + "@babel/plugin-transform-modules-umd": "^7.12.1", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.12.1", + "@babel/plugin-transform-new-target": "^7.12.1", + "@babel/plugin-transform-object-super": "^7.12.1", + "@babel/plugin-transform-parameters": "^7.12.1", + "@babel/plugin-transform-property-literals": "^7.12.1", + "@babel/plugin-transform-regenerator": "^7.12.1", + "@babel/plugin-transform-reserved-words": "^7.12.1", + "@babel/plugin-transform-shorthand-properties": "^7.12.1", + "@babel/plugin-transform-spread": "^7.12.1", + "@babel/plugin-transform-sticky-regex": "^7.12.7", + "@babel/plugin-transform-template-literals": "^7.12.1", + "@babel/plugin-transform-typeof-symbol": "^7.12.10", + "@babel/plugin-transform-unicode-escapes": "^7.12.1", + "@babel/plugin-transform-unicode-regex": "^7.12.1", + "@babel/preset-modules": "^0.1.3", + "@babel/types": "^7.12.11", + "core-js-compat": "^3.8.0", + "semver": "^5.5.0" + }, + "peerDependencies": { + "@babel/core": "^7.0.0-0" + } + }, + "node_modules/@babel/preset-env/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/@babel/preset-modules": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/@babel/preset-modules/-/preset-modules-0.1.4.tgz", + "integrity": "sha512-J36NhwnfdzpmH41M1DrnkkgAqhZaqr/NBdPfQ677mLzlaXo+oDiv1deyCDtgAhz8p328otdob0Du7+xgHGZbKg==", + "dev": true, + "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" + } + }, + "node_modules/@babel/runtime": { + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.12.5.tgz", + "integrity": "sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==", + "dev": true, + "dependencies": { + "regenerator-runtime": "^0.13.4" + } + }, + "node_modules/@babel/template": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.14.5.tgz", + "integrity": "sha512-6Z3Po85sfxRGachLULUhOmvAaOo7xCvqGQtxINai2mEGPFm6pQ4z5QInFnUrRpfoSV60BnjyF5F3c+15fxFV1g==", + "dependencies": { + "@babel/code-frame": "^7.14.5", + "@babel/parser": "^7.14.5", + "@babel/types": "^7.14.5" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.14.5.tgz", + "integrity": "sha512-G3BiS15vevepdmFqmUc9X+64y0viZYygubAMO8SvBmKARuF6CPSZtH4Ng9vi/lrWlZFGe3FWdXNy835akH8Glg==", + "dependencies": { + "@babel/code-frame": "^7.14.5", + "@babel/generator": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-hoist-variables": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", + "@babel/parser": "^7.14.5", + "@babel/types": "^7.14.5", + "debug": "^4.1.0", + "globals": "^11.1.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse/node_modules/@babel/generator": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.14.5.tgz", + "integrity": "sha512-y3rlP+/G25OIX3mYKKIOlQRcqj7YgrvHxOLbVmyLJ9bPmi5ttvUmpydVjcFjZphOktWuA7ovbx91ECloWTfjIA==", + "dependencies": { + "@babel/types": "^7.14.5", + "jsesc": "^2.5.1", + "source-map": "^0.5.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@babel/traverse/node_modules/source-map": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@babel/types": { + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.14.5.tgz", + "integrity": "sha512-M/NzBpEL95I5Hh4dwhin5JlE7EzO5PHMAuzjxss3tiOBD46KfQvVedN/3jEPZvdRvtsK2222XfdHogNIttFgcg==", + "dependencies": { + "@babel/helper-validator-identifier": "^7.14.5", + "to-fast-properties": "^2.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/@ctrl/tinycolor": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/@ctrl/tinycolor/-/tinycolor-3.4.0.tgz", + "integrity": "sha512-JZButFdZ1+/xAfpguQHoabIXkcqRRKpMrWKBkpEZZyxfY9C1DpADFB8PEqGSTeFr135SaTRfKqGKx5xSCLI7ZQ==", + "engines": { + "node": ">=10" + } + }, + "node_modules/@discoveryjs/json-ext": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.2.tgz", + "integrity": "sha512-HyYEUDeIj5rRQU2Hk5HTB2uHsbRQpF70nvMhVzi+VJR0X+xNEhjPui4/kBf3VeH/wqD28PT4sVOm8qqLjBrSZg==", + "dev": true, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/@istanbuljs/schema": { + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/@jsdevtools/coverage-istanbul-loader": { + "version": "3.0.5", + "resolved": "https://registry.npmjs.org/@jsdevtools/coverage-istanbul-loader/-/coverage-istanbul-loader-3.0.5.tgz", + "integrity": "sha512-EUCPEkaRPvmHjWAAZkWMT7JDzpw7FKB00WTISaiXsbNOd5hCHg77XLA8sLYLFDo1zepYLo2w7GstN8YBqRXZfA==", + "dev": true, + "dependencies": { + "convert-source-map": "^1.7.0", + "istanbul-lib-instrument": "^4.0.3", + "loader-utils": "^2.0.0", + "merge-source-map": "^1.1.0", + "schema-utils": "^2.7.0" + } + }, + "node_modules/@ng-bootstrap/ng-bootstrap": { + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/@ng-bootstrap/ng-bootstrap/-/ng-bootstrap-9.1.2.tgz", + "integrity": "sha512-oZt05AjwIY/oagS3QtST2GIlW9LQgTM7gX+wsuLQvHQt8bh9THOHAws8rDobeZi9fTUR3OvuxK56Mvj1jvyljw==", + "dependencies": { + "tslib": "^2.0.0" + }, + "peerDependencies": { + "@angular/common": "^11.0.0", + "@angular/core": "^11.0.0", + "@angular/forms": "^11.0.0", + "@angular/localize": "^11.0.0", + "rxjs": "^6.5.5" + } + }, + "node_modules/@ng-select/ng-select": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@ng-select/ng-select/-/ng-select-7.0.0.tgz", + "integrity": "sha512-YKQcohssQzLZgItCsEb0vbJR6xGRTIVq4FRp+5M7fkaX7wWM+o2o1R4LxsiWhz0CKajslkoR0VDytyiumf8QiQ==", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">= 10.0.0", + "npm": ">= 3.0.0" + }, + "peerDependencies": { + "@angular/common": ">=11.0.0 <12.0.0", + "@angular/core": ">=11.0.0 <12.0.0", + "@angular/forms": ">=11.0.0 <12.0.0" + } + }, + "node_modules/@ngtools/webpack": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-11.2.14.tgz", + "integrity": "sha512-6q57tEWtUJRsxfTKE19L20iXvNesfVy8hrVdyzVk64DZQh0lIl4/xZT4d5bJCWOuQQDaAeZK4YbEFcYJn7k1yw==", + "dev": true, + "dependencies": { + "@angular-devkit/core": "11.2.14", + "enhanced-resolve": "5.7.0", + "webpack-sources": "2.2.0" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + }, + "peerDependencies": { + "@angular/compiler-cli": "^11.0.0 || ^11.2.0-next", + "typescript": "~4.0.0 || ~4.1.0", + "webpack": "^4.0.0" + } + }, + "node_modules/@nodelib/fs.scandir": { + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "2.0.5", + "run-parallel": "^1.1.9" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.stat": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@nodelib/fs.walk": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.7.tgz", + "integrity": "sha512-BTIhocbPBSrRmHxOAJFtR18oLhxTtAFDAvL8hY1S3iU8k+E60W/YFs4jrixGzQjMpF4qPXxIQHcjVD9dz1C2QA==", + "dev": true, + "dependencies": { + "@nodelib/fs.scandir": "2.1.5", + "fastq": "^1.6.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/ci-detect": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@npmcli/ci-detect/-/ci-detect-1.3.0.tgz", + "integrity": "sha512-oN3y7FAROHhrAt7Rr7PnTSwrHrZVRTS2ZbyxeQwSSYD0ifwM3YNgQqbaRmjcWoPyq77MjchusjJDspbzMmip1Q==", + "dev": true + }, + "node_modules/@npmcli/git": { + "version": "2.0.9", + "resolved": "https://registry.npmjs.org/@npmcli/git/-/git-2.0.9.tgz", + "integrity": "sha512-hTMbMryvOqGLwnmMBKs5usbPsJtyEsMsgXwJbmNrsEuQQh1LAIMDU77IoOrwkCg+NgQWl+ySlarJASwM3SutCA==", + "dev": true, + "dependencies": { + "@npmcli/promise-spawn": "^1.3.2", + "lru-cache": "^6.0.0", + "mkdirp": "^1.0.4", + "npm-pick-manifest": "^6.1.1", + "promise-inflight": "^1.0.1", + "promise-retry": "^2.0.1", + "semver": "^7.3.5", + "which": "^2.0.2" + } + }, + "node_modules/@npmcli/git/node_modules/err-code": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", + "dev": true + }, + "node_modules/@npmcli/git/node_modules/hosted-git-info": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.0.2.tgz", + "integrity": "sha512-c9OGXbZ3guC/xOlCg1Ci/VgWlwsqDv1yMQL1CWqXDL0hDjXuNcq0zuR4xqPSuasI3kqFDhqSyTjREz5gzq0fXg==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/npm-package-arg": { + "version": "8.1.4", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.1.4.tgz", + "integrity": "sha512-xLokoCFqj/rPdr3LvcdDL6Kj6ipXGEDHD/QGpzwU6/pibYUOXmp5DBmg76yukFyx4ZDbrXNOTn+BPyd8TD4Jlw==", + "dev": true, + "dependencies": { + "hosted-git-info": "^4.0.1", + "semver": "^7.3.4", + "validate-npm-package-name": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/npm-pick-manifest": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-6.1.1.tgz", + "integrity": "sha512-dBsdBtORT84S8V8UTad1WlUyKIY9iMsAmqxHbLdeEeBNMLQDlDWWra3wYUx9EBEIiG/YwAy0XyNHDd2goAsfuA==", + "dev": true, + "dependencies": { + "npm-install-checks": "^4.0.0", + "npm-normalize-package-bin": "^1.0.1", + "npm-package-arg": "^8.1.2", + "semver": "^7.3.4" + } + }, + "node_modules/@npmcli/git/node_modules/promise-retry": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", + "dev": true, + "dependencies": { + "err-code": "^2.0.2", + "retry": "^0.12.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/git/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/@npmcli/installed-package-contents": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/@npmcli/installed-package-contents/-/installed-package-contents-1.0.7.tgz", + "integrity": "sha512-9rufe0wnJusCQoLpV9ZPKIVP55itrM5BxOXs10DmdbRfgWtHy1LDyskbwRnBghuB0PrF7pNPOqREVtpz4HqzKw==", + "dev": true, + "dependencies": { + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + }, + "bin": { + "installed-package-contents": "index.js" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/@npmcli/move-file": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@npmcli/move-file/-/move-file-1.1.2.tgz", + "integrity": "sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg==", + "dev": true, + "dependencies": { + "mkdirp": "^1.0.4", + "rimraf": "^3.0.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/move-file/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@npmcli/node-gyp": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@npmcli/node-gyp/-/node-gyp-1.0.2.tgz", + "integrity": "sha512-yrJUe6reVMpktcvagumoqD9r08fH1iRo01gn1u0zoCApa9lnZGEigVKUd2hzsCId4gdtkZZIVscLhNxMECKgRg==", + "dev": true + }, + "node_modules/@npmcli/promise-spawn": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@npmcli/promise-spawn/-/promise-spawn-1.3.2.tgz", + "integrity": "sha512-QyAGYo/Fbj4MXeGdJcFzZ+FkDkomfRBrPM+9QYJSg+PxgAUL+LU3FneQk37rKR2/zjqkCV1BLHccX98wRXG3Sg==", + "dev": true, + "dependencies": { + "infer-owner": "^1.0.4" + } + }, + "node_modules/@npmcli/run-script": { + "version": "1.8.5", + "resolved": "https://registry.npmjs.org/@npmcli/run-script/-/run-script-1.8.5.tgz", + "integrity": "sha512-NQspusBCpTjNwNRFMtz2C5MxoxyzlbuJ4YEhxAKrIonTiirKDtatsZictx9RgamQIx6+QuHMNmPl0wQdoESs9A==", + "dev": true, + "dependencies": { + "@npmcli/node-gyp": "^1.0.2", + "@npmcli/promise-spawn": "^1.3.2", + "infer-owner": "^1.0.4", + "node-gyp": "^7.1.0", + "read-package-json-fast": "^2.0.1" + } + }, + "node_modules/@npmcli/run-script/node_modules/read-package-json-fast": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-2.0.2.tgz", + "integrity": "sha512-5fyFUyO9B799foVk4n6ylcoAktG/FbE3jwRKxvwaeSrIunaoMc0u81dzXxjeAFKOce7O5KncdfwpGvvs6r5PsQ==", + "dev": true, + "dependencies": { + "json-parse-even-better-errors": "^2.3.0", + "npm-normalize-package-bin": "^1.0.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@scarf/scarf": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-1.1.0.tgz", + "integrity": "sha512-b2iE8kjjzzUo2WZ0xuE2N77kfnTds7ClrDxcz3Atz7h2XrNVoAPUoT75i7CY0st5x++70V91Y+c6RpBX9MX7Jg==", + "hasInstallScript": true + }, + "node_modules/@schematics/angular": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-11.2.14.tgz", + "integrity": "sha512-nErn5BFYp4HB7mOkt23kF+dyM6zPxolejM8eXQ5vd/rdhcc6ROaMZ0EmeEAWkfqB3+vqaSDz/D2Nm/IjJlyW/Q==", + "dev": true, + "dependencies": { + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", + "jsonc-parser": "3.0.0" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@schematics/update": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@schematics/update/-/update-0.1102.14.tgz", + "integrity": "sha512-OsWuC0iyNjpST1+hVUUZAegXAFpEFpS5uKYSQF3jsbyw8XHx7oA5/HbEwyr2WkX2EdV1tKrDLz6BrD5b8W6EYw==", + "dev": true, + "dependencies": { + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", + "@yarnpkg/lockfile": "1.1.0", + "ini": "2.0.0", + "npm-package-arg": "^8.0.0", + "pacote": "11.2.4", + "semver": "7.3.4", + "semver-intersect": "1.4.0" + }, + "engines": { + "node": ">= 10.13.0", + "npm": "^6.11.0 || ^7.5.6", + "yarn": ">= 1.13.0" + } + }, + "node_modules/@schematics/update/node_modules/ini": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz", + "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/@schematics/update/node_modules/semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/@trysound/sax": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.1.1.tgz", + "integrity": "sha512-Z6DoceYb/1xSg5+e+ZlPZ9v0N16ZvZ+wYMraFue4HYrE4ttONKtsvruIRf6t9TBR0YvSOfi1hUU0fJfBLCDYow==", + "dev": true, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/@types/component-emitter": { + "version": "1.2.10", + "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.10.tgz", + "integrity": "sha512-bsjleuRKWmGqajMerkzox19aGbscQX5rmmvvXl3wlIp5gMG1HgkiwPxsN5p070fBDKTNSPgojVbuY1+HWMbFhg==", + "dev": true + }, + "node_modules/@types/cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-y7mImlc/rNkvCRmg8gC3/lj87S7pTUIJ6QGjwHR9WQJcFs+ZMTOaoPrkdFA/YdbuqVEmEbb5RdhVxMkAcgOnpg==", + "dev": true + }, + "node_modules/@types/cors": { + "version": "2.8.10", + "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.10.tgz", + "integrity": "sha512-C7srjHiVG3Ey1nR6d511dtDkCEjxuN9W1HWAEjGq8kpcwmNM6JJkpC0xvabM7BXTG2wDq8Eu33iH9aQKa7IvLQ==", + "dev": true + }, + "node_modules/@types/glob": { + "version": "7.1.3", + "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.3.tgz", + "integrity": "sha512-SEYeGAIQIQX8NN6LDKprLjbrd5dARM5EXsd8GI/A5l0apYI1fGMWgPHSe4ZKL4eozlAyI+doUE9XbYS4xCkQ1w==", + "dev": true, + "dependencies": { + "@types/minimatch": "*", + "@types/node": "*" + } + }, + "node_modules/@types/jasmine": { + "version": "3.6.11", + "resolved": "https://registry.npmjs.org/@types/jasmine/-/jasmine-3.6.11.tgz", + "integrity": "sha512-S6pvzQDvMZHrkBz2Mcn/8Du7cpr76PlRJBAoHnSDNbulULsH5dp0Gns+WRyNX5LHejz/ljxK4/vIHK/caHt6SQ==", + "dev": true + }, + "node_modules/@types/jasminewd2": { + "version": "2.0.8", + "resolved": "https://registry.npmjs.org/@types/jasminewd2/-/jasminewd2-2.0.8.tgz", + "integrity": "sha512-d9p31r7Nxk0ZH0U39PTH0hiDlJ+qNVGjlt1ucOoTUptxb2v+Y5VMnsxfwN+i3hK4yQnqBi3FMmoMFcd1JHDxdg==", + "dev": true, + "dependencies": { + "@types/jasmine": "*" + } + }, + "node_modules/@types/json-schema": { + "version": "7.0.7", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.7.tgz", + "integrity": "sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA==", + "dev": true + }, + "node_modules/@types/minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-1z8k4wzFnNjVK/tlxvrWuK5WMt6mydWWP7+zvH5eFep4oj+UkrfiJTRtjCeBXNpwaA/FYqqtb4/QS4ianFpIRA==", + "dev": true + }, + "node_modules/@types/node": { + "version": "12.12.64", + "resolved": "https://registry.npmjs.org/@types/node/-/node-12.12.64.tgz", + "integrity": "sha512-UV1/ZJMC+HcP902wWdpC43cAcGu0IQk/I5bXjP2aSuCjsk3cE74mDvFrLKga7oDC170ugOAYBwfT4DSQW3akDA==", + "dev": true + }, + "node_modules/@types/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==", + "dev": true + }, + "node_modules/@types/pdfjs-dist": { + "version": "2.1.7", + "resolved": "https://registry.npmjs.org/@types/pdfjs-dist/-/pdfjs-dist-2.1.7.tgz", + "integrity": "sha512-nQIwcPUhkAIyn7x9NS0lR/qxYfd5unRtfGkMjvpgF4Sh28IXftRymaNmFKTTdejDNY25NDGSIyjwj/BRwAPexg==" + }, + "node_modules/@types/selenium-webdriver": { + "version": "3.0.17", + "resolved": "https://registry.npmjs.org/@types/selenium-webdriver/-/selenium-webdriver-3.0.17.tgz", + "integrity": "sha512-tGomyEuzSC1H28y2zlW6XPCaDaXFaD6soTdb4GNdmte2qfHtrKqhy0ZFs4r/1hpazCfEZqeTSRLvSasmEx89uw==", + "dev": true + }, + "node_modules/@types/source-list-map": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@types/source-list-map/-/source-list-map-0.1.2.tgz", + "integrity": "sha512-K5K+yml8LTo9bWJI/rECfIPrGgxdpeNbj+d53lwN4QjW1MCwlkhUms+gtdzigTeUyBr09+u8BwOIY3MXvHdcsA==", + "dev": true + }, + "node_modules/@types/webpack-sources": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/@types/webpack-sources/-/webpack-sources-0.1.8.tgz", + "integrity": "sha512-JHB2/xZlXOjzjBB6fMOpH1eQAfsrpqVVIbneE0Rok16WXwFaznaI5vfg75U5WgGJm7V9W1c4xeRQDjX/zwvghA==", + "dev": true, + "dependencies": { + "@types/node": "*", + "@types/source-list-map": "*", + "source-map": "^0.6.1" + } + }, + "node_modules/@types/webpack-sources/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/@webassemblyjs/ast": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ast/-/ast-1.9.0.tgz", + "integrity": "sha512-C6wW5L+b7ogSDVqymbkkvuW9kruN//YisMED04xzeBBqjHa2FYnmvOlS6Xj68xWQRgWvI9cIglsjFowH/RJyEA==", + "dev": true, + "dependencies": { + "@webassemblyjs/helper-module-context": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0" + } + }, + "node_modules/@webassemblyjs/floating-point-hex-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/floating-point-hex-parser/-/floating-point-hex-parser-1.9.0.tgz", + "integrity": "sha512-TG5qcFsS8QB4g4MhrxK5TqfdNe7Ey/7YL/xN+36rRjl/BlGE/NcBvJcqsRgCP6Z92mRE+7N50pRIi8SmKUbcQA==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-api-error": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-api-error/-/helper-api-error-1.9.0.tgz", + "integrity": "sha512-NcMLjoFMXpsASZFxJ5h2HZRcEhDkvnNFOAKneP5RbKRzaWJN36NC4jqQHKwStIhGXu5mUWlUUk7ygdtrO8lbmw==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-buffer": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-buffer/-/helper-buffer-1.9.0.tgz", + "integrity": "sha512-qZol43oqhq6yBPx7YM3m9Bv7WMV9Eevj6kMi6InKOuZxhw+q9hOkvq5e/PpKSiLfyetpaBnogSbNCfBwyB00CA==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-code-frame": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-code-frame/-/helper-code-frame-1.9.0.tgz", + "integrity": "sha512-ERCYdJBkD9Vu4vtjUYe8LZruWuNIToYq/ME22igL+2vj2dQ2OOujIZr3MEFvfEaqKoVqpsFKAGsRdBSBjrIvZA==", + "dev": true, + "dependencies": { + "@webassemblyjs/wast-printer": "1.9.0" + } + }, + "node_modules/@webassemblyjs/helper-fsm": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-fsm/-/helper-fsm-1.9.0.tgz", + "integrity": "sha512-OPRowhGbshCb5PxJ8LocpdX9Kl0uB4XsAjl6jH/dWKlk/mzsANvhwbiULsaiqT5GZGT9qinTICdj6PLuM5gslw==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-module-context": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-module-context/-/helper-module-context-1.9.0.tgz", + "integrity": "sha512-MJCW8iGC08tMk2enck1aPW+BE5Cw8/7ph/VGZxwyvGbJwjktKkDK7vy7gAmMDx88D7mhDTCNKAW5tED+gZ0W8g==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0" + } + }, + "node_modules/@webassemblyjs/helper-wasm-bytecode": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-bytecode/-/helper-wasm-bytecode-1.9.0.tgz", + "integrity": "sha512-R7FStIzyNcd7xKxCZH5lE0Bqy+hGTwS3LJjuv1ZVxd9O7eHCedSdrId/hMOd20I+v8wDXEn+bjfKDLzTepoaUw==", + "dev": true + }, + "node_modules/@webassemblyjs/helper-wasm-section": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/helper-wasm-section/-/helper-wasm-section-1.9.0.tgz", + "integrity": "sha512-XnMB8l3ek4tvrKUUku+IVaXNHz2YsJyOOmz+MMkZvh8h1uSJpSen6vYnw3IoQ7WwEuAhL8Efjms1ZWjqh2agvw==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0" + } + }, + "node_modules/@webassemblyjs/ieee754": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/ieee754/-/ieee754-1.9.0.tgz", + "integrity": "sha512-dcX8JuYU/gvymzIHc9DgxTzUUTLexWwt8uCTWP3otys596io0L5aW02Gb1RjYpx2+0Jus1h4ZFqjla7umFniTg==", + "dev": true, + "dependencies": { + "@xtuc/ieee754": "^1.2.0" + } + }, + "node_modules/@webassemblyjs/leb128": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/leb128/-/leb128-1.9.0.tgz", + "integrity": "sha512-ENVzM5VwV1ojs9jam6vPys97B/S65YQtv/aanqnU7D8aSoHFX8GyhGg0CMfyKNIHBuAVjy3tlzd5QMMINa7wpw==", + "dev": true, + "dependencies": { + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/utf8": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/utf8/-/utf8-1.9.0.tgz", + "integrity": "sha512-GZbQlWtopBTP0u7cHrEx+73yZKrQoBMpwkGEIqlacljhXCkVM1kMQge/Mf+csMJAjEdSwhOyLAS0AoR3AG5P8w==", + "dev": true + }, + "node_modules/@webassemblyjs/wasm-edit": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-edit/-/wasm-edit-1.9.0.tgz", + "integrity": "sha512-FgHzBm80uwz5M8WKnMTn6j/sVbqilPdQXTWraSjBwFXSYGirpkSWE2R9Qvz9tNiTKQvoKILpCuTjBKzOIm0nxw==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/helper-wasm-section": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0", + "@webassemblyjs/wasm-opt": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0", + "@webassemblyjs/wast-printer": "1.9.0" + } + }, + "node_modules/@webassemblyjs/wasm-gen": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-gen/-/wasm-gen-1.9.0.tgz", + "integrity": "sha512-cPE3o44YzOOHvlsb4+E9qSqjc9Qf9Na1OO/BHFy4OI91XDE14MjFN4lTMezzaIWdPqHnsTodGGNP+iRSYfGkjA==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/ieee754": "1.9.0", + "@webassemblyjs/leb128": "1.9.0", + "@webassemblyjs/utf8": "1.9.0" + } + }, + "node_modules/@webassemblyjs/wasm-opt": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-opt/-/wasm-opt-1.9.0.tgz", + "integrity": "sha512-Qkjgm6Anhm+OMbIL0iokO7meajkzQD71ioelnfPEj6r4eOFuqm4YC3VBPqXjFyyNwowzbMD+hizmprP/Fwkl2A==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-buffer": "1.9.0", + "@webassemblyjs/wasm-gen": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0" + } + }, + "node_modules/@webassemblyjs/wasm-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wasm-parser/-/wasm-parser-1.9.0.tgz", + "integrity": "sha512-9+wkMowR2AmdSWQzsPEjFU7njh8HTO5MqO8vjwEHuM+AMHioNqSBONRdr0NQQ3dVQrzp0s8lTcYqzUdb7YgELA==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-api-error": "1.9.0", + "@webassemblyjs/helper-wasm-bytecode": "1.9.0", + "@webassemblyjs/ieee754": "1.9.0", + "@webassemblyjs/leb128": "1.9.0", + "@webassemblyjs/utf8": "1.9.0" + } + }, + "node_modules/@webassemblyjs/wast-parser": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-parser/-/wast-parser-1.9.0.tgz", + "integrity": "sha512-qsqSAP3QQ3LyZjNC/0jBJ/ToSxfYJ8kYyuiGvtn/8MK89VrNEfwj7BPQzJVHi0jGTRK2dGdJ5PRqhtjzoww+bw==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/floating-point-hex-parser": "1.9.0", + "@webassemblyjs/helper-api-error": "1.9.0", + "@webassemblyjs/helper-code-frame": "1.9.0", + "@webassemblyjs/helper-fsm": "1.9.0", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@webassemblyjs/wast-printer": { + "version": "1.9.0", + "resolved": "https://registry.npmjs.org/@webassemblyjs/wast-printer/-/wast-printer-1.9.0.tgz", + "integrity": "sha512-2J0nE95rHXHyQ24cWjMKJ1tqB/ds8z/cyeOZxJhcb+rW+SQASVjuznUSmdz5GpVJTzU8JkhYut0D3siFDD6wsA==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/wast-parser": "1.9.0", + "@xtuc/long": "4.2.2" + } + }, + "node_modules/@xtuc/ieee754": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/@xtuc/ieee754/-/ieee754-1.2.0.tgz", + "integrity": "sha512-DX8nKgqcGwsc0eJSqYt5lwP4DH5FlHnmuWWBRy7X0NcaGR0ZtuyeESgMwTYVEtxmsNGY+qit4QYT/MIYTOTPeA==", + "dev": true + }, + "node_modules/@xtuc/long": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/@xtuc/long/-/long-4.2.2.tgz", + "integrity": "sha512-NuHqBY1PB/D8xU6s/thBgOAiAP7HOYDQ32+BFZILJ8ivkUkAHQnWfn6WhL79Owj1qmUnoN/YPhktdIoucipkAQ==", + "dev": true + }, + "node_modules/@yarnpkg/lockfile": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/@yarnpkg/lockfile/-/lockfile-1.1.0.tgz", + "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", + "dev": true + }, + "node_modules/abab": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", + "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", + "dev": true + }, + "node_modules/abbrev": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", + "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", + "dev": true + }, + "node_modules/accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "dev": true, + "dependencies": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/acorn": { + "version": "6.4.2", + "resolved": "https://registry.npmjs.org/acorn/-/acorn-6.4.2.tgz", + "integrity": "sha512-XtGIhXwF8YM8bJhGxG5kXgjkEuNGLTkoYqVE+KMR+aspr4KGYmKYg7yUe3KghyQ9yheNwLnjmzh/7+gfDBmHCQ==", + "dev": true, + "bin": { + "acorn": "bin/acorn" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/adjust-sourcemap-loader": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/adjust-sourcemap-loader/-/adjust-sourcemap-loader-4.0.0.tgz", + "integrity": "sha512-OXwN5b9pCUXNQHJpwwD2qP40byEmSgzj8B4ydSN0uMNYWiFmJ6x6KwUllMmfk8Rwu/HJDFR7U8ubsWBoN0Xp0A==", + "dev": true, + "dependencies": { + "loader-utils": "^2.0.0", + "regex-parser": "^2.2.11" + }, + "engines": { + "node": ">=8.9" + } + }, + "node_modules/adm-zip": { + "version": "0.4.16", + "resolved": "https://registry.npmjs.org/adm-zip/-/adm-zip-0.4.16.tgz", + "integrity": "sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg==", + "dev": true, + "engines": { + "node": ">=0.3.0" + } + }, + "node_modules/agent-base": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", + "integrity": "sha512-salcGninV0nPrwpGNn4VTXBb1SOuXQBiqbrNXoeizJsHrsL6ERFM2Ne3JUSBWRE6aeNJI2ROP/WEEIDUiDe3cg==", + "dev": true, + "dependencies": { + "es6-promisify": "^5.0.0" + }, + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/agentkeepalive": { + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.1.4.tgz", + "integrity": "sha512-+V/rGa3EuU74H6wR04plBb7Ks10FbtUQgRj/FQOG7uUIEuaINI+AiqJR1k6t3SVNs7o7ZjIdus6706qqzVq8jQ==", + "dev": true, + "dependencies": { + "debug": "^4.1.0", + "depd": "^1.1.2", + "humanize-ms": "^1.2.1" + }, + "engines": { + "node": ">= 8.0.0" + } + }, + "node_modules/aggregate-error": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/aggregate-error/-/aggregate-error-3.1.0.tgz", + "integrity": "sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA==", + "dev": true, + "dependencies": { + "clean-stack": "^2.0.0", + "indent-string": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/ajv": { + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", + "dev": true, + "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" + }, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/epoberezkin" + } + }, + "node_modules/ajv-errors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", + "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", + "dev": true, + "peerDependencies": { + "ajv": ">=5.0.0" + } + }, + "node_modules/ajv-keywords": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", + "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", + "dev": true, + "peerDependencies": { + "ajv": "^6.9.1" + } + }, + "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 + }, + "node_modules/ansi-colors": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", + "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/ansi-escapes": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/ansi-escapes/-/ansi-escapes-4.3.1.tgz", + "integrity": "sha512-JWF7ocqNrp8u9oqpgV+wH5ftbt+cfvv+PTjOvKLT3AdYly/LmORARfEVT1iyjwN+4MqE5UmVKoAdIBqeoCHgLA==", + "dev": true, + "dependencies": { + "type-fest": "^0.11.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/ansi-html": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/ansi-html/-/ansi-html-0.0.7.tgz", + "integrity": "sha1-gTWEAhliqenm/QOflA0S9WynhZ4=", + "dev": true, + "engines": [ + "node >= 0.8.0" + ], + "bin": { + "ansi-html": "bin/ansi-html" + } + }, + "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/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==", + "dependencies": { + "color-convert": "^1.9.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/anymatch": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", + "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", + "dependencies": { + "normalize-path": "^3.0.0", + "picomatch": "^2.0.4" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/app-root-path": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/app-root-path/-/app-root-path-3.0.0.tgz", + "integrity": "sha512-qMcx+Gy2UZynHjOHOIXPNvpf+9cjvk3cWrBBK7zg4gH9+clobJRb9NGzcT7mQTcV/6Gm/1WelUtqxVXnNlrwcw==", + "dev": true, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/aproba": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/aproba/-/aproba-1.2.0.tgz", + "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", + "dev": true + }, + "node_modules/are-we-there-yet": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", + "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", + "dev": true, + "dependencies": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, + "node_modules/arg": { + "version": "4.1.3", + "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", + "integrity": "sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==", + "dev": true + }, + "node_modules/argparse": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/argparse/-/argparse-1.0.10.tgz", + "integrity": "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg==", + "dev": true, + "dependencies": { + "sprintf-js": "~1.0.2" + } + }, + "node_modules/aria-query": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/aria-query/-/aria-query-3.0.0.tgz", + "integrity": "sha1-ZbP8wcoRVajJrmTW7uKX8V1RM8w=", + "dev": true, + "dependencies": { + "ast-types-flow": "0.0.7", + "commander": "^2.11.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/array-flatten": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-2.1.2.tgz", + "integrity": "sha512-hNfzcOV8W4NdualtqBFPyVO+54DSJuZGY9qT4pRroB6S9e3iiido2ISIC5h9R2sPJ8H3FHCIiEnsv1lPXO3KtQ==", + "dev": true + }, + "node_modules/array-union": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-2.1.0.tgz", + "integrity": "sha512-HGyxoOTYUyCM6stUe6EJgnd4EoewAI7zMdfqO+kGjnlZmBDz/cR5pf8r/cR4Wq60sL/p0IkcjUEEPwS3GFrIyw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/array-uniq": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/array-uniq/-/array-uniq-1.0.3.tgz", + "integrity": "sha1-r2rId6Jcx/dOBYiUdThY39sk/bY=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/arrify": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", + "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "safer-buffer": "~2.1.0" + } + }, + "node_modules/asn1.js": { + "version": "5.4.1", + "resolved": "https://registry.npmjs.org/asn1.js/-/asn1.js-5.4.1.tgz", + "integrity": "sha512-+I//4cYPccV8LdmBLiX8CYvf9Sp3vQsrqu2QNXRcrbiWvcx/UdlFiqUJJzxRQxgsZmvhXhn4cSKeSmoFjVdupA==", + "dev": true, + "dependencies": { + "bn.js": "^4.0.0", + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0", + "safer-buffer": "^2.1.0" + } + }, + "node_modules/asn1.js/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "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, + "dependencies": { + "object-assign": "^4.1.1", + "util": "0.10.3" + } + }, + "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 + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ast-types-flow": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/ast-types-flow/-/ast-types-flow-0.0.7.tgz", + "integrity": "sha1-9wtzXGvKGlycItmCw+Oef+ujva0=", + "dev": true + }, + "node_modules/async": { + "version": "2.6.3", + "resolved": "https://registry.npmjs.org/async/-/async-2.6.3.tgz", + "integrity": "sha512-zflvls11DCy+dQWzTW2dzuilv8Z5X/pjfmZOWba6TNIVDm+2UDaJmXSOXlasHKfNBs8oo3M0aT50fDEWfKZjXg==", + "dev": true, + "dependencies": { + "lodash": "^4.17.14" + } + }, + "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 + }, + "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 + }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha1-x57Zf380y48robyXkLzDZkdLS3k=", + "dev": true + }, + "node_modules/atob": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/atob/-/atob-2.1.2.tgz", + "integrity": "sha512-Wm6ukoaOGJi/73p/cl2GvLjTI5JM1k/O14isD73YML8StrH/7/lRFgmg8nICZgD3bZZvjwCGxtMOD3wWNAu8cg==", + "dev": true, + "bin": { + "atob": "bin/atob.js" + }, + "engines": { + "node": ">= 4.5.0" + } + }, + "node_modules/autoprefixer": { + "version": "10.2.4", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.2.4.tgz", + "integrity": "sha512-DCCdUQiMD+P/as8m3XkeTUkUKuuRqLGcwD0nll7wevhqoJfMRpJlkFd1+MQh1pvupjiQuip42lc/VFvfUTMSKw==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.1", + "caniuse-lite": "^1.0.30001181", + "colorette": "^1.2.1", + "fraction.js": "^4.0.13", + "normalize-range": "^0.1.2", + "postcss-value-parser": "^4.1.0" + }, + "bin": { + "autoprefixer": "bin/autoprefixer" + }, + "engines": { + "node": "^10 || ^12 || >=14" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "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, + "engines": { + "node": "*" + } + }, + "node_modules/aws4": { + "version": "1.10.1", + "resolved": "https://registry.npmjs.org/aws4/-/aws4-1.10.1.tgz", + "integrity": "sha512-zg7Hz2k5lI8kb7U32998pRRFin7zJlkfezGJjUc2heaD4Pw2wObakCDVzkKztTm/Ln7eiVvYsjqak0Ed4LkMDA==", + "dev": true + }, + "node_modules/axobject-query": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-2.0.2.tgz", + "integrity": "sha512-MCeek8ZH7hKyO1rWUbKNQBbl4l2eY0ntk7OGi+q0RlafrCnfPxC06WZA+uebCfmYp4mNU9jRBP1AhGyf8+W3ww==", + "dev": true, + "dependencies": { + "ast-types-flow": "0.0.7" + } + }, + "node_modules/babel-loader": { + "version": "8.2.2", + "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.2.2.tgz", + "integrity": "sha512-JvTd0/D889PQBtUXJ2PXaKU/pjZDMtHA9V2ecm+eNRmmBCMR09a+fmpGTNwnJtFmFl5Ei7Vy47LjBb+L0wQ99g==", + "dev": true, + "dependencies": { + "find-cache-dir": "^3.3.1", + "loader-utils": "^1.4.0", + "make-dir": "^3.1.0", + "schema-utils": "^2.6.5" + }, + "engines": { + "node": ">= 8.9" + }, + "peerDependencies": { + "@babel/core": "^7.0.0", + "webpack": ">=2" + } + }, + "node_modules/babel-loader/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, + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/babel-loader/node_modules/loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/babel-loader/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/babel-loader/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/babel-plugin-dynamic-import-node": { + "version": "2.3.3", + "resolved": "https://registry.npmjs.org/babel-plugin-dynamic-import-node/-/babel-plugin-dynamic-import-node-2.3.3.tgz", + "integrity": "sha512-jZVI+s9Zg3IqA/kdi0i6UDCybUI3aSBLnglhYbSSjKlV7yF1F/5LWv8MakQmvYpnbJDS6fcBL2KzHSxNCMtWSQ==", + "dev": true, + "dependencies": { + "object.assign": "^4.1.0" + } + }, + "node_modules/balanced-match": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" + }, + "node_modules/base": { + "version": "0.11.2", + "resolved": "https://registry.npmjs.org/base/-/base-0.11.2.tgz", + "integrity": "sha512-5T6P4xPgpp0YDFvSWwEZ4NoE3aM4QBQXDzmVbraCkFj8zHM+mba8SyqB5DbZWyR7mYHo6Y7BdQo3MoA4m0TeQg==", + "dev": true, + "dependencies": { + "cache-base": "^1.0.1", + "class-utils": "^0.3.5", + "component-emitter": "^1.2.1", + "define-property": "^1.0.0", + "isobject": "^3.0.1", + "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/base64-arraybuffer": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.4.tgz", + "integrity": "sha1-mBjHngWbE1X5fgQooBfIOOkLqBI=", + "dev": true, + "engines": { + "node": ">= 0.6.0" + } + }, + "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 + }, + "node_modules/base64id": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/base64id/-/base64id-2.0.0.tgz", + "integrity": "sha512-lGe34o6EHj9y3Kts9R4ZYs/Gr+6N7MCaMlIFA3F1R2O5/m7K06AxfSeO5530PEERE6/WyEg3lsuyw4GHlPZHog==", + "dev": true, + "engines": { + "node": "^4.5.0 || >= 5.9" + } + }, + "node_modules/batch": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/batch/-/batch-0.6.1.tgz", + "integrity": "sha1-3DQxT05nkxgJP8dgJyUl+UvyXBY=", + "dev": true + }, + "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, + "dependencies": { + "tweetnacl": "^0.14.3" + } + }, + "node_modules/big.js": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", + "integrity": "sha512-vyL2OymJxmarO8gxMr0mhChsO9QGwhynfuu4+MHTAW6czfq9humCB7rKpUjDd9YUiDPU4mzpyupFSvOClAwbmQ==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/binary-extensions": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", + "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", + "engines": { + "node": ">=8" + } + }, + "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, + "optional": true, + "dependencies": { + "file-uri-to-path": "1.0.0" + } + }, + "node_modules/bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "dependencies": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + } + }, + "node_modules/bl/node_modules/buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "node_modules/bl/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": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/blocking-proxy": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/blocking-proxy/-/blocking-proxy-1.0.1.tgz", + "integrity": "sha512-KE8NFMZr3mN2E0HcvCgRtX7DjhiIQrwle+nSVJVC/yqFb9+xznHl2ZcoBp2L9qzkI4t4cBFJ1efXF8Dwi132RA==", + "dev": true, + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "blocking-proxy": "built/lib/bin.js" + }, + "engines": { + "node": ">=6.9.x" + } + }, + "node_modules/bluebird": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/bluebird/-/bluebird-3.7.2.tgz", + "integrity": "sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg==", + "dev": true + }, + "node_modules/bn.js": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.2.0.tgz", + "integrity": "sha512-D7iWRBvnZE8ecXiLj/9wbxH7Tk79fAh8IHaTNq1RWRixsS02W+5qS+iE9yq6RYl0asXx5tw0bLhmT5pIfbSquw==", + "dev": true + }, + "node_modules/body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "dev": true, + "dependencies": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/body-parser/node_modules/bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/body-parser/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/body-parser/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "node_modules/bonjour": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/bonjour/-/bonjour-3.5.0.tgz", + "integrity": "sha1-jokKGD2O6aI5OzhExpGkK897yfU=", + "dev": true, + "dependencies": { + "array-flatten": "^2.1.0", + "deep-equal": "^1.0.1", + "dns-equal": "^1.0.0", + "dns-txt": "^2.0.2", + "multicast-dns": "^6.0.1", + "multicast-dns-service-types": "^1.1.0" + } + }, + "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 + }, + "node_modules/bootstrap": { + "version": "4.5.2", + "resolved": "https://registry.npmjs.org/bootstrap/-/bootstrap-4.5.2.tgz", + "integrity": "sha512-vlGn0bcySYl/iV+BGA544JkkZP5LB3jsmkeKLFQakCOwCM3AOk7VkldBz4jrzSe+Z0Ezn99NVXa1o45cQY4R6A==" + }, + "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==", + "dependencies": { + "balanced-match": "^1.0.0", + "concat-map": "0.0.1" + } + }, + "node_modules/braces": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", + "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", + "dependencies": { + "fill-range": "^7.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/brorand": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/brorand/-/brorand-1.1.0.tgz", + "integrity": "sha1-EsJe/kCkXjwyPrhnWgoM5XsiNx8=", + "dev": true + }, + "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, + "dependencies": { + "buffer-xor": "^1.0.3", + "cipher-base": "^1.0.0", + "create-hash": "^1.1.0", + "evp_bytestokey": "^1.0.3", + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "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, + "dependencies": { + "browserify-aes": "^1.0.4", + "browserify-des": "^1.0.0", + "evp_bytestokey": "^1.0.0" + } + }, + "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, + "dependencies": { + "cipher-base": "^1.0.1", + "des.js": "^1.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/browserify-rsa": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.1.0.tgz", + "integrity": "sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog==", + "dev": true, + "dependencies": { + "bn.js": "^5.0.0", + "randombytes": "^2.0.1" + } + }, + "node_modules/browserify-sign": { + "version": "4.2.1", + "resolved": "https://registry.npmjs.org/browserify-sign/-/browserify-sign-4.2.1.tgz", + "integrity": "sha512-/vrA5fguVAKKAVTNJjgSm1tRQDHUU6DbwO9IROu/0WAzC8PKhucDSh18J0RMvVeHAn5puMd+QHC2erPRNf8lmg==", + "dev": true, + "dependencies": { + "bn.js": "^5.1.1", + "browserify-rsa": "^4.0.1", + "create-hash": "^1.2.0", + "create-hmac": "^1.1.7", + "elliptic": "^6.5.3", + "inherits": "^2.0.4", + "parse-asn1": "^5.1.5", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + } + }, + "node_modules/browserify-sign/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": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/browserify-sign/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "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, + "dependencies": { + "pako": "~1.0.5" + } + }, + "node_modules/browserslist": { + "version": "4.16.6", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.16.6.tgz", + "integrity": "sha512-Wspk/PqO+4W9qp5iUTJsa1B/QrYn1keNCcEP5OvP7WBwT4KaDly0uONYmC6Xa3Z5IqnUgS0KcgLYu1l74x0ZXQ==", + "dev": true, + "dependencies": { + "caniuse-lite": "^1.0.30001219", + "colorette": "^1.2.2", + "electron-to-chromium": "^1.3.723", + "escalade": "^3.1.1", + "node-releases": "^1.1.71" + }, + "bin": { + "browserslist": "cli.js" + }, + "engines": { + "node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + } + }, + "node_modules/browserstack": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/browserstack/-/browserstack-1.6.0.tgz", + "integrity": "sha512-HJDJ0TSlmkwnt9RZ+v5gFpa1XZTBYTj0ywvLwJ3241J7vMw2jAsGNVhKHtmCOyg+VxeLZyaibO9UL71AsUeDIw==", + "dev": true, + "dependencies": { + "https-proxy-agent": "^2.2.1" + } + }, + "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, + "dependencies": { + "base64-js": "^1.0.2", + "ieee754": "^1.1.4", + "isarray": "^1.0.0" + } + }, + "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 + }, + "node_modules/buffer-indexof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/buffer-indexof/-/buffer-indexof-1.1.1.tgz", + "integrity": "sha512-4/rOEg86jivtPTeOUUT61jJO1Ya1TrR/OkqCSZDyq84WJh3LuuiphBYJN+fm5xufIk4XAFcEwte/8WzC8If/1g==", + "dev": true + }, + "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 + }, + "node_modules/builtin-modules": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-1.1.1.tgz", + "integrity": "sha1-Jw8HbFpywC9bZaR9+Uxf46J4iS8=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/builtins": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/builtins/-/builtins-1.0.3.tgz", + "integrity": "sha1-y5T662HIaWRR2zZTThQi+U8K7og=", + "dev": true + }, + "node_modules/bytes": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz", + "integrity": "sha1-0ygVQE1olpn4Wk6k+odV3ROpYEg=", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/cacache": { + "version": "15.0.5", + "resolved": "https://registry.npmjs.org/cacache/-/cacache-15.0.5.tgz", + "integrity": "sha512-lloiL22n7sOjEEXdL8NAjTgv9a1u43xICE9/203qonkZUCj5X1UEWIdf2/Y0d6QcCtMzbKQyhrcDbdvlZTs/+A==", + "dev": true, + "dependencies": { + "@npmcli/move-file": "^1.0.1", + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "glob": "^7.1.4", + "infer-owner": "^1.0.4", + "lru-cache": "^6.0.0", + "minipass": "^3.1.1", + "minipass-collect": "^1.0.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.2", + "mkdirp": "^1.0.3", + "p-map": "^4.0.0", + "promise-inflight": "^1.0.1", + "rimraf": "^3.0.2", + "ssri": "^8.0.0", + "tar": "^6.0.2", + "unique-filename": "^1.1.1" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/cacache/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/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, + "dependencies": { + "collection-visit": "^1.0.0", + "component-emitter": "^1.2.1", + "get-value": "^2.0.6", + "has-value": "^1.0.0", + "isobject": "^3.0.1", + "set-value": "^2.0.0", + "to-object-path": "^0.3.0", + "union-value": "^1.0.0", + "unset-value": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/callsites": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "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, + "engines": { + "node": ">=6" + } + }, + "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, + "dependencies": { + "browserslist": "^4.0.0", + "caniuse-lite": "^1.0.0", + "lodash.memoize": "^4.1.2", + "lodash.uniq": "^4.5.0" + } + }, + "node_modules/caniuse-lite": { + "version": "1.0.30001237", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001237.tgz", + "integrity": "sha512-pDHgRndit6p1NR2GhzMbQ6CkRrp4VKuSsqbcLeOQppYPKOYkKT/6ZvZDvKJUqcmtyWIAHuZq3SVS2vc1egCZzw==", + "dev": true, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/browserslist" + } + }, + "node_modules/canonical-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/canonical-path/-/canonical-path-1.0.0.tgz", + "integrity": "sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg==" + }, + "node_modules/caseless": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/caseless/-/caseless-0.12.0.tgz", + "integrity": "sha1-G2gcIf+EAzyCZUMJBolCDRhxUdw=", + "dev": true + }, + "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==", + "dependencies": { + "ansi-styles": "^3.2.1", + "escape-string-regexp": "^1.0.5", + "supports-color": "^5.3.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/chardet": { + "version": "0.7.0", + "resolved": "https://registry.npmjs.org/chardet/-/chardet-0.7.0.tgz", + "integrity": "sha512-mT8iDcrh03qDGRRmoA2hmBJnxpllMR+0/0qlzjqZES6NdiWDcZkCNAk4rPFZ9Q85r27unkiNNg8ZOiwZXBHwcA==", + "dev": true + }, + "node_modules/chokidar": { + "version": "3.4.2", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.2.tgz", + "integrity": "sha512-IZHaDeBeI+sZJRX7lGcXsdzgvZqKv6sECqsbErJA4mHWfpRrD8B97kSFN4cQz6nGBGiuFia1MKR4d6c1o8Cv7A==", + "dependencies": { + "anymatch": "~3.1.1", + "braces": "~3.0.2", + "glob-parent": "~5.1.0", + "is-binary-path": "~2.1.0", + "is-glob": "~4.0.1", + "normalize-path": "~3.0.0", + "readdirp": "~3.4.0" + }, + "engines": { + "node": ">= 8.10.0" + }, + "optionalDependencies": { + "fsevents": "~2.1.2" + } + }, + "node_modules/chownr": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-2.0.0.tgz", + "integrity": "sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/chrome-trace-event": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", + "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", + "dev": true, + "engines": { + "node": ">=6.0" + } + }, + "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, + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/circular-dependency-plugin": { + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/circular-dependency-plugin/-/circular-dependency-plugin-5.2.2.tgz", + "integrity": "sha512-g38K9Cm5WRwlaH6g03B9OEz/0qRizI+2I7n+Gz+L5DxXJAPAiWQvwlYNm1V1jkdpUv95bOe/ASm2vfi/G560jQ==", + "dev": true, + "engines": { + "node": ">=6.0.0" + }, + "peerDependencies": { + "webpack": ">=4.0.1" + } + }, + "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, + "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": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/class-utils/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" + } + }, + "node_modules/clean-stack": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/clean-stack/-/clean-stack-2.2.0.tgz", + "integrity": "sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/cli-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/cli-cursor/-/cli-cursor-3.1.0.tgz", + "integrity": "sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw==", + "dev": true, + "dependencies": { + "restore-cursor": "^3.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/cli-spinners": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.6.0.tgz", + "integrity": "sha512-t+4/y50K/+4xcCRosKkA7W4gTr1MySvLV0q+PxmG7FJ5g+66ChKurYjxBCjHggHH3HA5Hh9cy+lcUGWDqVH+4Q==", + "dev": true, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/cli-width": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/cli-width/-/cli-width-3.0.0.tgz", + "integrity": "sha512-FxqpkPPwu1HjuN93Omfm4h8uIanXofW0RxVEW3k5RKx+mJJYSthzNhp32Kzxxy3YAEZ/Dc/EWN1vZRY0+kOhbw==", + "dev": true, + "engines": { + "node": ">= 10" + } + }, + "node_modules/cliui": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-5.0.0.tgz", + "integrity": "sha512-PYeGSEmmHM6zvoef2w8TPzlrnNpXIjTipYK780YswmIP9vjxmd6Y2a3CB2Ks6/AU8NHjZugXvo8w3oWM2qnwXA==", + "dev": true, + "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": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "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" + } + }, + "node_modules/clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", + "dev": true, + "dependencies": { + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/codelyzer": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/codelyzer/-/codelyzer-6.0.1.tgz", + "integrity": "sha512-cOyGQgMdhnRYtW2xrJUNrNYDjEgwQ+BrE2y93Bwz3h4DJ6vJRLfupemU5N3pbYsUlBHJf0u1j1UGk+NLW4d97g==", + "dev": true, + "dependencies": { + "@angular/compiler": "9.0.0", + "@angular/core": "9.0.0", + "app-root-path": "^3.0.0", + "aria-query": "^3.0.0", + "axobject-query": "2.0.2", + "css-selector-tokenizer": "^0.7.1", + "cssauron": "^1.4.0", + "damerau-levenshtein": "^1.0.4", + "rxjs": "^6.5.3", + "semver-dsl": "^1.0.1", + "source-map": "^0.5.7", + "sprintf-js": "^1.1.2", + "tslib": "^1.10.0", + "zone.js": "~0.10.3" + } + }, + "node_modules/codelyzer/node_modules/@angular/compiler": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-9.0.0.tgz", + "integrity": "sha512-ctjwuntPfZZT2mNj2NDIVu51t9cvbhl/16epc5xEwyzyDt76pX9UgwvY+MbXrf/C/FWwdtmNtfP698BKI+9leQ==", + "dev": true + }, + "node_modules/codelyzer/node_modules/@angular/core": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/@angular/core/-/core-9.0.0.tgz", + "integrity": "sha512-6Pxgsrf0qF9iFFqmIcWmjJGkkCaCm6V5QNnxMy2KloO3SDq6QuMVRbN9RtC8Urmo25LP+eZ6ZgYqFYpdD8Hd9w==", + "dev": true + }, + "node_modules/codelyzer/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/codelyzer/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==", + "dev": true + }, + "node_modules/codelyzer/node_modules/tslib": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", + "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", + "dev": true + }, + "node_modules/codelyzer/node_modules/zone.js": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.10.3.tgz", + "integrity": "sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg==", + "dev": true + }, + "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, + "dependencies": { + "map-visit": "^1.0.0", + "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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==", + "dependencies": { + "color-name": "1.1.3" + } + }, + "node_modules/color-name": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", + "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" + }, + "node_modules/colord": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/colord/-/colord-2.0.1.tgz", + "integrity": "sha512-vm5YpaWamD0Ov6TSG0GGmUIwstrWcfKQV/h2CmbR7PbNu41+qdB5PW9lpzhjedrpm08uuYvcXi0Oel1RLZIJuA==", + "dev": true + }, + "node_modules/colorette": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.2.tgz", + "integrity": "sha512-MKGMzyfeuutC/ZJ1cba9NqcNpfeqMUcYmyF1ZFY6/Cn7CNSAKx6a+s48sqLqyAiZuaP2TcqMhoo+dlwFnVxT9w==", + "dev": true + }, + "node_modules/colors": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/colors/-/colors-1.4.0.tgz", + "integrity": "sha512-a+UqTh4kgZg/SlGvfbzDHpgRu7AAQOmmqRHJnxhRZICKFUT91brVhNNt58CMWU9PsBbv3PDCZUHbVxuDiH2mtA==", + "dev": true, + "engines": { + "node": ">=0.1.90" + } + }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/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 + }, + "node_modules/commondir": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/commondir/-/commondir-1.0.1.tgz", + "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", + "dev": true + }, + "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 + }, + "node_modules/compressible": { + "version": "2.0.18", + "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", + "integrity": "sha512-AF3r7P5dWxL8MxyITRMlORQNaOA2IkAFaTr4k7BUumjPtRpGDTZpl0Pb1XCO6JeDCBdp126Cgs9sMxqSjgYyRg==", + "dev": true, + "dependencies": { + "mime-db": ">= 1.43.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/compression": { + "version": "1.7.4", + "resolved": "https://registry.npmjs.org/compression/-/compression-1.7.4.tgz", + "integrity": "sha512-jaSIDzP9pZVS4ZfQ+TzvtiWhdpFhE2RDHz8QJkpX9SIpLq88VueF5jJw6t+6CUQcAoA6t+x89MLrWAqpfDE8iQ==", + "dev": true, + "dependencies": { + "accepts": "~1.3.5", + "bytes": "3.0.0", + "compressible": "~2.0.16", + "debug": "2.6.9", + "on-headers": "~1.0.2", + "safe-buffer": "5.1.2", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/compression/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/compression/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/concat-map": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" + }, + "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, + "engines": [ + "node >= 0.8" + ], + "dependencies": { + "buffer-from": "^1.0.0", + "inherits": "^2.0.3", + "readable-stream": "^2.2.2", + "typedarray": "^0.0.6" + } + }, + "node_modules/connect": { + "version": "3.7.0", + "resolved": "https://registry.npmjs.org/connect/-/connect-3.7.0.tgz", + "integrity": "sha512-ZqRXc+tZukToSNmh5C2iWMSoV3X1YUcPbqEM4DkEG5tNQXrQUZCNVGGv3IuicnkMtPfGf3Xtp8WCXs295iQ1pQ==", + "dev": true, + "dependencies": { + "debug": "2.6.9", + "finalhandler": "1.1.2", + "parseurl": "~1.3.3", + "utils-merge": "1.0.1" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/connect-history-api-fallback": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/connect-history-api-fallback/-/connect-history-api-fallback-1.6.0.tgz", + "integrity": "sha512-e54B99q/OUoH64zYYRf3HBP5z24G38h5D3qXu23JGRoigpX5Ss4r9ZnDk3g0Z8uQC2x2lPaJ+UlWBc1ZWBWdLg==", + "dev": true, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/connect/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/connect/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/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 + }, + "node_modules/console-control-strings": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", + "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=", + "dev": true + }, + "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 + }, + "node_modules/content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "dev": true, + "dependencies": { + "safe-buffer": "5.1.2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "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==", + "dependencies": { + "safe-buffer": "~5.1.1" + } + }, + "node_modules/cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=", + "dev": true + }, + "node_modules/copy-anything": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/copy-anything/-/copy-anything-2.0.3.tgz", + "integrity": "sha512-GK6QUtisv4fNS+XcI7shX0Gx9ORg7QqIznyfho79JTnX1XhLiyZHfftvGiziqzRiEi/Bjhgpi+D2o7HxJFPnDQ==", + "dev": true, + "dependencies": { + "is-what": "^3.12.0" + } + }, + "node_modules/copy-concurrently": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/copy-concurrently/-/copy-concurrently-1.0.5.tgz", + "integrity": "sha512-f2domd9fsVDFtaFcbaRZuYXwtdmnzqbADSwhSWYxYB/Q8zsdUUFMXVRwXGDMWmbEzAn1kdRrtI1T/KTFOL4X2A==", + "dev": true, + "dependencies": { + "aproba": "^1.1.1", + "fs-write-stream-atomic": "^1.0.8", + "iferr": "^0.1.5", + "mkdirp": "^0.5.1", + "rimraf": "^2.5.4", + "run-queue": "^1.0.0" + } + }, + "node_modules/copy-concurrently/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/copy-descriptor": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/copy-descriptor/-/copy-descriptor-0.1.1.tgz", + "integrity": "sha1-Z29us8OZl8LuGsOpJP1hJHSPV40=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/copy-webpack-plugin": { + "version": "6.3.2", + "resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-6.3.2.tgz", + "integrity": "sha512-MgJ1uouLIbDg4ST1GzqrGQyKoXY5iPqi6fghFqarijam7FQcBa/r6Rg0VkoIuzx75Xq8iAMghyOueMkWUQ5OaA==", + "dev": true, + "dependencies": { + "cacache": "^15.0.5", + "fast-glob": "^3.2.4", + "find-cache-dir": "^3.3.1", + "glob-parent": "^5.1.1", + "globby": "^11.0.1", + "loader-utils": "^2.0.0", + "normalize-path": "^3.0.0", + "p-limit": "^3.0.2", + "schema-utils": "^3.0.0", + "serialize-javascript": "^5.0.1", + "webpack-sources": "^1.4.3" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.37.0 || ^5.0.0" + } + }, + "node_modules/copy-webpack-plugin/node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/copy-webpack-plugin/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/copy-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/copy-webpack-plugin/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/core-js": { + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.8.3.tgz", + "integrity": "sha512-KPYXeVZYemC2TkNEkX/01I+7yd+nX3KddKwZ1Ww7SKWdI2wQprSgLmrTddT8nw92AjEklTsPBoSdQBhbI1bQ6Q==", + "dev": true, + "hasInstallScript": true, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "node_modules/core-js-compat": { + "version": "3.14.0", + "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.14.0.tgz", + "integrity": "sha512-R4NS2eupxtiJU+VwgkF9WTpnSfZW4pogwKHd8bclWU2sp93Pr5S1uYJI84cMOubJRou7bcfL0vmwtLslWN5p3A==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.6", + "semver": "7.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/core-js" + } + }, + "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 + }, + "node_modules/cors": { + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", + "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", + "dev": true, + "dependencies": { + "object-assign": "^4", + "vary": "^1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/cosmiconfig": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.0.0.tgz", + "integrity": "sha512-pondGvTuVYDk++upghXJabWzL6Kxu6f26ljFw64Swq9v6sQPUL3EUlVDV56diOjpCayKihL6hVe8exIACU4XcA==", + "dev": true, + "dependencies": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/create-ecdh": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/create-ecdh/-/create-ecdh-4.0.4.tgz", + "integrity": "sha512-mf+TCx8wWc9VpuxfP2ht0iSISLZnt0JgWlrOKZiNqyUZWnjIaCIVNQArMHnCZKfEYRg6IM7A+NeJoN8gf/Ws0A==", + "dev": true, + "dependencies": { + "bn.js": "^4.1.0", + "elliptic": "^6.5.3" + } + }, + "node_modules/create-ecdh/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "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, + "dependencies": { + "cipher-base": "^1.0.1", + "inherits": "^2.0.1", + "md5.js": "^1.3.4", + "ripemd160": "^2.0.1", + "sha.js": "^2.4.0" + } + }, + "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, + "dependencies": { + "cipher-base": "^1.0.3", + "create-hash": "^1.1.0", + "inherits": "^2.0.1", + "ripemd160": "^2.0.0", + "safe-buffer": "^5.0.1", + "sha.js": "^2.4.8" + } + }, + "node_modules/critters": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/critters/-/critters-0.0.7.tgz", + "integrity": "sha512-qUF2SaAWFYjNPdCcPpu68p2DnHiosia84yx5mPTlUMQjkjChR+n6sO1/I7yn2U2qNDgSPTd2SoaTIDQcUL+EwQ==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0", + "css": "^3.0.0", + "parse5": "^6.0.1", + "parse5-htmlparser2-tree-adapter": "^6.0.1", + "pretty-bytes": "^5.3.0" + } + }, + "node_modules/critters/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/critters/node_modules/chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/critters/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/critters/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/critters/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/critters/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/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, + "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" + } + }, + "node_modules/cross-spawn/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "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, + "dependencies": { + "browserify-cipher": "^1.0.0", + "browserify-sign": "^4.0.0", + "create-ecdh": "^4.0.0", + "create-hash": "^1.1.0", + "create-hmac": "^1.1.0", + "diffie-hellman": "^5.0.0", + "inherits": "^2.0.1", + "pbkdf2": "^3.0.3", + "public-encrypt": "^4.0.0", + "randombytes": "^2.0.0", + "randomfill": "^1.0.3" + }, + "engines": { + "node": "*" + } + }, + "node_modules/css": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/css/-/css-3.0.0.tgz", + "integrity": "sha512-DG9pFfwOrzc+hawpmqX/dHYHJG+Bsdb0klhyi1sDneOgGOXy9wQIC8hzyVp1e4NRYDBdxcylvywPkkXCHAzTyQ==", + "dev": true, + "dependencies": { + "inherits": "^2.0.4", + "source-map": "^0.6.1", + "source-map-resolve": "^0.6.0" + } + }, + "node_modules/css-color-names": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/css-color-names/-/css-color-names-1.0.1.tgz", + "integrity": "sha512-/loXYOch1qU1biStIFsHH8SxTmOseh1IJqFvy8IujXOm1h+QjUdDhkzOrR5HG8K8mlxREj0yfi8ewCHx0eMxzA==", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/css-declaration-sorter": { + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-6.0.3.tgz", + "integrity": "sha512-52P95mvW1SMzuRZegvpluT6yEv0FqQusydKQPZsNN5Q7hh8EwQvN8E2nwuJ16BBvNN6LcoIZXu/Bk58DAhrrxw==", + "dev": true, + "dependencies": { + "timsort": "^0.3.0" + }, + "engines": { + "node": ">= 10" + }, + "peerDependencies": { + "postcss": "^8.0.9" + } + }, + "node_modules/css-loader": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-5.0.1.tgz", + "integrity": "sha512-cXc2ti9V234cq7rJzFKhirb2L2iPy8ZjALeVJAozXYz9te3r4eqLSixNAbMDJSgJEQywqXzs8gonxaboeKqwiw==", + "dev": true, + "dependencies": { + "camelcase": "^6.2.0", + "cssesc": "^3.0.0", + "icss-utils": "^5.0.0", + "loader-utils": "^2.0.0", + "postcss": "^8.1.4", + "postcss-modules-extract-imports": "^3.0.0", + "postcss-modules-local-by-default": "^4.0.0", + "postcss-modules-scope": "^3.0.0", + "postcss-modules-values": "^4.0.0", + "postcss-value-parser": "^4.1.0", + "schema-utils": "^3.0.0", + "semver": "^7.3.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.27.0 || ^5.0.0" + } + }, + "node_modules/css-loader/node_modules/camelcase": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", + "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/css-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/css-parse": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/css-parse/-/css-parse-2.0.0.tgz", + "integrity": "sha1-pGjuZnwW2BzPBcWMONKpfHgNv9Q=", + "dev": true, + "dependencies": { + "css": "^2.0.0" + } + }, + "node_modules/css-parse/node_modules/css": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/css/-/css-2.2.4.tgz", + "integrity": "sha512-oUnjmWpy0niI3x/mPL8dVEI1l7MnG3+HHyRPHf+YFSbK+svOhXpmSOcDURUh2aOCgl2grzrOPt1nHLuCVFULLw==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "source-map": "^0.6.1", + "source-map-resolve": "^0.5.2", + "urix": "^0.1.0" + } + }, + "node_modules/css-parse/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-parse/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==", + "dev": true, + "dependencies": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, + "node_modules/css-select": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-3.1.2.tgz", + "integrity": "sha512-qmss1EihSuBNWNNhHjxzxSfJoFBM/lERB/Q4EnsJQQC62R2evJDW481091oAdOr9uh46/0n4nrg0It5cAnj1RA==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0", + "css-what": "^4.0.0", + "domhandler": "^4.0.0", + "domutils": "^2.4.3", + "nth-check": "^2.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/css-selector-tokenizer": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.3.tgz", + "integrity": "sha512-jWQv3oCEL5kMErj4wRnK/OPoBi0D+P1FR2cDCKYPaMeD2eW3/mttav8HT4hT1CKopiJI/psEULjkClhvJo4Lvg==", + "dev": true, + "dependencies": { + "cssesc": "^3.0.0", + "fastparse": "^1.1.2" + } + }, + "node_modules/css-tree": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.3.tgz", + "integrity": "sha512-tRpdppF7TRazZrjJ6v3stzv93qxRcSsFmW6cX0Zm2NVKpxE1WV1HblnghVv9TreireHkqI/VDEsfolRF1p6y7Q==", + "dev": true, + "dependencies": { + "mdn-data": "2.0.14", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/css-tree/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/css-what": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-4.0.0.tgz", + "integrity": "sha512-teijzG7kwYfNVsUh2H/YN62xW3KK9YhXEgSlbxMlcyjPNvdKJqFx5lrwlJgoFP1ZHlB89iGDlo/JyshKeRhv5A==", + "dev": true, + "engines": { + "node": ">= 6" + }, + "funding": { + "url": "https://github.com/sponsors/fb55" + } + }, + "node_modules/css/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/cssauron": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/cssauron/-/cssauron-1.4.0.tgz", + "integrity": "sha1-pmAt/34EqDBtwNuaVR6S6LVmKtg=", + "dev": true, + "dependencies": { + "through": "X.X.X" + } + }, + "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, + "bin": { + "cssesc": "bin/cssesc" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/cssnano": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-5.0.2.tgz", + "integrity": "sha512-8JK3EnPsjQsULme9/e5M2hF564f/480hwsdcHvQ7ZtAIMfQ1O3SCfs+b8Mjf5KJxhYApyRshR2QSovEJi2K72Q==", + "dev": true, + "dependencies": { + "cosmiconfig": "^7.0.0", + "cssnano-preset-default": "^5.0.1", + "is-resolvable": "^1.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/cssnano" + }, + "peerDependencies": { + "postcss": "^8.2.1" + } + }, + "node_modules/cssnano-preset-default": { + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-5.1.3.tgz", + "integrity": "sha512-qo9tX+t4yAAZ/yagVV3b+QBKeLklQbmgR3wI7mccrDcR+bEk9iHgZN1E7doX68y9ThznLya3RDmR+nc7l6/2WQ==", + "dev": true, + "dependencies": { + "css-declaration-sorter": "^6.0.3", + "cssnano-utils": "^2.0.1", + "postcss-calc": "^8.0.0", + "postcss-colormin": "^5.2.0", + "postcss-convert-values": "^5.0.1", + "postcss-discard-comments": "^5.0.1", + "postcss-discard-duplicates": "^5.0.1", + "postcss-discard-empty": "^5.0.1", + "postcss-discard-overridden": "^5.0.1", + "postcss-merge-longhand": "^5.0.2", + "postcss-merge-rules": "^5.0.2", + "postcss-minify-font-values": "^5.0.1", + "postcss-minify-gradients": "^5.0.1", + "postcss-minify-params": "^5.0.1", + "postcss-minify-selectors": "^5.1.0", + "postcss-normalize-charset": "^5.0.1", + "postcss-normalize-display-values": "^5.0.1", + "postcss-normalize-positions": "^5.0.1", + "postcss-normalize-repeat-style": "^5.0.1", + "postcss-normalize-string": "^5.0.1", + "postcss-normalize-timing-functions": "^5.0.1", + "postcss-normalize-unicode": "^5.0.1", + "postcss-normalize-url": "^5.0.2", + "postcss-normalize-whitespace": "^5.0.1", + "postcss-ordered-values": "^5.0.2", + "postcss-reduce-initial": "^5.0.1", + "postcss-reduce-transforms": "^5.0.1", + "postcss-svgo": "^5.0.2", + "postcss-unique-selectors": "^5.0.1" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/cssnano-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/cssnano-utils/-/cssnano-utils-2.0.1.tgz", + "integrity": "sha512-i8vLRZTnEH9ubIyfdZCAdIdgnHAUeQeByEeQ2I7oTilvP9oHO6RScpeq3GsFUVqeB8uZgOQ9pw8utofNn32hhQ==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/csso": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/csso/-/csso-4.2.0.tgz", + "integrity": "sha512-wvlcdIbf6pwKEk7vHj8/Bkc0B4ylXZruLvOgs9doS5eOsOpuodOV2zJChSpkp+pRpYQLQMeF04nr3Z68Sta9jA==", + "dev": true, + "dependencies": { + "css-tree": "^1.1.2" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/custom-event": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/custom-event/-/custom-event-1.0.1.tgz", + "integrity": "sha1-XQKkaFCt8bSjF5RqOSj8y1v9BCU=", + "dev": true + }, + "node_modules/cyclist": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/cyclist/-/cyclist-1.0.1.tgz", + "integrity": "sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=", + "dev": true + }, + "node_modules/damerau-levenshtein": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.6.tgz", + "integrity": "sha512-JVrozIeElnj3QzfUIt8tB8YMluBJom4Vw9qTPpjGYQ9fYlB3D/rb6OordUxf3xeFB35LKWs0xqcO5U6ySvBtug==", + "dev": true + }, + "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, + "dependencies": { + "assert-plus": "^1.0.0" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/date-format": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/date-format/-/date-format-3.0.0.tgz", + "integrity": "sha512-eyTcpKOcamdhWJXj56DpQMo1ylSQpcGtGKXcU0Tb97+K56/CF5amAqqqNj0+KvA0iw2ynxtHWFsPDSClCxe48w==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/debug": { + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", + "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", + "dependencies": { + "ms": "2.1.2" + }, + "engines": { + "node": ">=6.0" + }, + "peerDependenciesMeta": { + "supports-color": { + "optional": true + } + } + }, + "node_modules/decamelize": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/deep-equal": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/deep-equal/-/deep-equal-1.1.1.tgz", + "integrity": "sha512-yd9c5AdiqVcR+JjcwUQb9DkhJc8ngNr0MahEBGvDiJw8puWab2yZlh+nkasOnZP+EGTAP6rRp2JzJhJZzvNF8g==", + "dev": true, + "dependencies": { + "is-arguments": "^1.0.4", + "is-date-object": "^1.0.1", + "is-regex": "^1.0.4", + "object-is": "^1.0.1", + "object-keys": "^1.1.1", + "regexp.prototype.flags": "^1.2.0" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/default-gateway": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/default-gateway/-/default-gateway-4.2.0.tgz", + "integrity": "sha512-h6sMrVB1VMWVrW13mSc6ia/DwYYw5MN6+exNu1OaJeFac5aSAvwM7lZ0NVfTABuSkQelr4h5oebg3KB1XPdjgA==", + "dev": true, + "dependencies": { + "execa": "^1.0.0", + "ip-regex": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "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, + "dependencies": { + "clone": "^1.0.2" + } + }, + "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, + "dependencies": { + "object-keys": "^1.0.12" + }, + "engines": { + "node": ">= 0.4" + } + }, + "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, + "dependencies": { + "is-descriptor": "^1.0.2", + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/del/-/del-4.1.1.tgz", + "integrity": "sha512-QwGuEUouP2kVwQenAsOof5Fv8K9t3D8Ca8NxcXKrIpEHjTXK5J2nXLdP+ALI1cgv8wj7KuwBhTwBkOZSJKM5XQ==", + "dev": true, + "dependencies": { + "@types/glob": "^7.1.1", + "globby": "^6.1.0", + "is-path-cwd": "^2.0.0", + "is-path-in-cwd": "^2.0.0", + "p-map": "^2.0.0", + "pify": "^4.0.1", + "rimraf": "^2.6.3" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/del/node_modules/array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "dependencies": { + "array-uniq": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del/node_modules/globby": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-6.1.0.tgz", + "integrity": "sha1-9abXDoOV4hyFj7BInWTfAkJNUGw=", + "dev": true, + "dependencies": { + "array-union": "^1.0.1", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del/node_modules/globby/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/del/node_modules/p-map": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-2.1.0.tgz", + "integrity": "sha512-y3b8Kpd8OAN444hxfBbFfj1FY/RjtTd8tzYwhUqNYXx0fXx2iX4maP4Qr6qhIKbQXI02wTLAda4fYUbDagTUFw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/del/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", + "dev": true, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/delegates": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", + "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=", + "dev": true + }, + "node_modules/depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/dependency-graph": { + "version": "0.7.2", + "resolved": "https://registry.npmjs.org/dependency-graph/-/dependency-graph-0.7.2.tgz", + "integrity": "sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ==", + "engines": { + "node": ">= 0.6.0" + } + }, + "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, + "dependencies": { + "inherits": "^2.0.1", + "minimalistic-assert": "^1.0.0" + } + }, + "node_modules/destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=", + "dev": true + }, + "node_modules/detect-node": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.1.0.tgz", + "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==", + "dev": true + }, + "node_modules/di": { + "version": "0.0.1", + "resolved": "https://registry.npmjs.org/di/-/di-0.0.1.tgz", + "integrity": "sha1-gGZJMmzqp8qjMG112YXqJ0i6kTw=", + "dev": true + }, + "node_modules/diff": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/diff/-/diff-4.0.2.tgz", + "integrity": "sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A==", + "dev": true, + "engines": { + "node": ">=0.3.1" + } + }, + "node_modules/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, + "dependencies": { + "bn.js": "^4.1.0", + "miller-rabin": "^4.0.0", + "randombytes": "^2.0.0" + } + }, + "node_modules/diffie-hellman/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "node_modules/dir-glob": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/dir-glob/-/dir-glob-3.0.1.tgz", + "integrity": "sha512-WkrWp9GR4KXfKGYzOLmTuGVi1UWFfws377n9cc55/tb6DuqyF6pcQ5AbiHEshaDpY9v6oaSr2XCDidGmMwdzIA==", + "dev": true, + "dependencies": { + "path-type": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/dns-equal": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/dns-equal/-/dns-equal-1.0.0.tgz", + "integrity": "sha1-s55/HabrCnW6nBcySzR1PEfgZU0=", + "dev": true + }, + "node_modules/dns-packet": { + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.4.tgz", + "integrity": "sha512-BQ6F4vycLXBvdrJZ6S3gZewt6rcrks9KBgM9vrhW+knGRqc8uEdT7fuCwloc7nny5xNoMJ17HGH0R/6fpo8ECA==", + "dev": true, + "dependencies": { + "ip": "^1.1.0", + "safe-buffer": "^5.0.1" + } + }, + "node_modules/dns-txt": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/dns-txt/-/dns-txt-2.0.2.tgz", + "integrity": "sha1-uR2Ab10nGI5Ks+fRB9iBocxGQrY=", + "dev": true, + "dependencies": { + "buffer-indexof": "^1.0.0" + } + }, + "node_modules/dom-serialize": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/dom-serialize/-/dom-serialize-2.2.1.tgz", + "integrity": "sha1-ViromZ9Evl6jB29UGdzVnrQ6yVs=", + "dev": true, + "dependencies": { + "custom-event": "~1.0.0", + "ent": "~2.2.0", + "extend": "^3.0.0", + "void-elements": "^2.0.0" + } + }, + "node_modules/dom-serializer": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.3.2.tgz", + "integrity": "sha512-5c54Bk5Dw4qAxNOI1pFEizPSjVsx5+bpJKmL2kPn8JhBUq2q09tTCa3mjijun2NfK78NMouDYNMBkOrPZiS+ig==", + "dev": true, + "dependencies": { + "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", + "entities": "^2.0.0" + }, + "funding": { + "url": "https://github.com/cheeriojs/dom-serializer?sponsor=1" + } + }, + "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, + "engines": { + "node": ">=0.4", + "npm": ">=1.2" + } + }, + "node_modules/domelementtype": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.2.0.tgz", + "integrity": "sha512-DtBMo82pv1dFtUmHyr48beiuq792Sxohr+8Hm9zoxklYPfa6n0Z3Byjj2IV7bmr2IyqClnqEQhfgHJJ5QF0R5A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/fb55" + } + ] + }, + "node_modules/domhandler": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.2.0.tgz", + "integrity": "sha512-zk7sgt970kzPks2Bf+dwT/PLzghLnsivb9CcxkvR8Mzr66Olr0Ofd8neSbglHJHaHa2MadfoSdNlKYAaafmWfA==", + "dev": true, + "dependencies": { + "domelementtype": "^2.2.0" + }, + "engines": { + "node": ">= 4" + }, + "funding": { + "url": "https://github.com/fb55/domhandler?sponsor=1" + } + }, + "node_modules/domutils": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.7.0.tgz", + "integrity": "sha512-8eaHa17IwJUPAiB+SoTYBo5mCdeMgdcAoXJ59m6DT1vw+5iLS3gNoqYaRowaBKtGVrOF1Jz4yDTgYKLK2kvfJg==", + "dev": true, + "dependencies": { + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" + }, + "funding": { + "url": "https://github.com/fb55/domutils?sponsor=1" + } + }, + "node_modules/duplexify": { + "version": "3.7.1", + "resolved": "https://registry.npmjs.org/duplexify/-/duplexify-3.7.1.tgz", + "integrity": "sha512-07z8uv2wMyS51kKhD1KsdXJg5WQ6t93RneqRxUHnskXVtlYYkLqM0gqStQZ3pj073g687jPCHrqNfCzawLYh5g==", + "dev": true, + "dependencies": { + "end-of-stream": "^1.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.0.0", + "stream-shift": "^1.0.0" + } + }, + "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, + "dependencies": { + "jsbn": "~0.1.0", + "safer-buffer": "^2.1.0" + } + }, + "node_modules/ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=", + "dev": true + }, + "node_modules/electron-to-chromium": { + "version": "1.3.752", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.752.tgz", + "integrity": "sha512-2Tg+7jSl3oPxgsBsWKh5H83QazTkmWG/cnNwJplmyZc7KcN61+I10oUgaXSVk/NwfvN3BdkKDR4FYuRBQQ2v0A==", + "dev": true + }, + "node_modules/elliptic": { + "version": "6.5.4", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz", + "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==", + "dev": true, + "dependencies": { + "bn.js": "^4.11.9", + "brorand": "^1.1.0", + "hash.js": "^1.0.0", + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/elliptic/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "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 + }, + "node_modules/emojis-list": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-3.0.0.tgz", + "integrity": "sha512-/kyM18EfinwXZbno9FyUGeFh87KC8HRQBQGildHZbEuRyWFOmv1U10o9BBp8XVZDVNNuQKyIGIu5ZYAAXJ0V2Q==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/encoding": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.13.tgz", + "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", + "dev": true, + "optional": true, + "dependencies": { + "iconv-lite": "^0.6.2" + } + }, + "node_modules/encoding/node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "optional": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/end-of-stream": { + "version": "1.4.4", + "resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz", + "integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==", + "dev": true, + "dependencies": { + "once": "^1.4.0" + } + }, + "node_modules/engine.io": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-4.1.1.tgz", + "integrity": "sha512-t2E9wLlssQjGw0nluF6aYyfX8LwYU8Jj0xct+pAhfWfv/YrBn6TSNtEYsgxHIfaMqfrLx07czcMg9bMN6di+3w==", + "dev": true, + "dependencies": { + "accepts": "~1.3.4", + "base64id": "2.0.0", + "cookie": "~0.4.1", + "cors": "~2.8.5", + "debug": "~4.3.1", + "engine.io-parser": "~4.0.0", + "ws": "~7.4.2" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/engine.io-parser": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-4.0.2.tgz", + "integrity": "sha512-sHfEQv6nmtJrq6TKuIz5kyEKH/qSdK56H/A+7DnAuUPWosnIZAS2NHNcPLmyjtY3cGS/MqJdZbUjW97JU72iYg==", + "dev": true, + "dependencies": { + "base64-arraybuffer": "0.1.4" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/engine.io/node_modules/cookie": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.1.tgz", + "integrity": "sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/engine.io/node_modules/ws": { + "version": "7.4.6", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.6.tgz", + "integrity": "sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==", + "dev": true, + "engines": { + "node": ">=8.3.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": "^5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, + "node_modules/enhanced-resolve": { + "version": "5.7.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.7.0.tgz", + "integrity": "sha512-6njwt/NsZFUKhM6j9U8hzVyD4E4r0x7NQzhTCbcWOJ0IQjNSAoalWmb0AE51Wn+fwan5qVESWi7t2ToBxs9vrw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/ent": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/ent/-/ent-2.2.0.tgz", + "integrity": "sha1-6WQhkyWiHQX0RGai9obtbOX13R0=", + "dev": true + }, + "node_modules/entities": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", + "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", + "dev": true, + "funding": { + "url": "https://github.com/fb55/entities?sponsor=1" + } + }, + "node_modules/env-paths": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", + "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/err-code": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-1.1.2.tgz", + "integrity": "sha1-BuARbTAo9q70gGhJ6w6mp0iuaWA=", + "dev": true + }, + "node_modules/errno": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.8.tgz", + "integrity": "sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A==", + "dev": true, + "dependencies": { + "prr": "~1.0.1" + }, + "bin": { + "errno": "cli.js" + } + }, + "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, + "dependencies": { + "is-arrayish": "^0.2.1" + } + }, + "node_modules/es6-promise": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", + "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", + "dev": true + }, + "node_modules/es6-promisify": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/es6-promisify/-/es6-promisify-5.0.0.tgz", + "integrity": "sha1-UQnWLz5W6pZ8S2NQWu8IKRyKUgM=", + "dev": true, + "dependencies": { + "es6-promise": "^4.0.3" + } + }, + "node_modules/escalade": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", + "engines": { + "node": ">=6" + } + }, + "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 + }, + "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=", + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/eslint-scope": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/eslint-scope/-/eslint-scope-4.0.3.tgz", + "integrity": "sha512-p7VutNr1O/QrxysMo3E45FjYDTeXBy0iTltPFNSqKAIfjDSXC+4dj+qfyuD8bfAXrW/y6lW3O76VaYNPKfpKrg==", + "dev": true, + "dependencies": { + "esrecurse": "^4.1.0", + "estraverse": "^4.1.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "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/esrecurse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/esrecurse/-/esrecurse-4.3.0.tgz", + "integrity": "sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag==", + "dev": true, + "dependencies": { + "estraverse": "^5.2.0" + }, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/esrecurse/node_modules/estraverse": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-5.2.0.tgz", + "integrity": "sha512-BxbNGGNm0RyRYvUdHpIwv9IWzeM9XClbOxwoATuFdOE7ZE6wHL+HQ5T8hoPM+zHvmKzzsEqhgy0GrQ5X13afiQ==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/estraverse": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/estraverse/-/estraverse-4.3.0.tgz", + "integrity": "sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/eventemitter3": { + "version": "4.0.7", + "resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz", + "integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==", + "dev": true + }, + "node_modules/events": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", + "dev": true, + "engines": { + "node": ">=0.8.x" + } + }, + "node_modules/eventsource": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-1.1.0.tgz", + "integrity": "sha512-VSJjT5oCNrFvCS6igjzPAt5hBzQ2qPBFIbJ03zLI9SE0mxwZpMw6BfJrbFHm1a141AavMEB8JHmBhWAd66PfCg==", + "dev": true, + "dependencies": { + "original": "^1.0.0" + }, + "engines": { + "node": ">=0.12.0" + } + }, + "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, + "dependencies": { + "md5.js": "^1.3.4", + "safe-buffer": "^5.1.1" + } + }, + "node_modules/execa": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/execa/-/execa-1.0.0.tgz", + "integrity": "sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA==", + "dev": true, + "dependencies": { + "cross-spawn": "^6.0.0", + "get-stream": "^4.0.0", + "is-stream": "^1.1.0", + "npm-run-path": "^2.0.0", + "p-finally": "^1.0.0", + "signal-exit": "^3.0.0", + "strip-eof": "^1.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/exit": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/exit/-/exit-0.1.2.tgz", + "integrity": "sha1-BjJjj42HfMghB9MKD/8aF8uhzQw=", + "dev": true, + "engines": { + "node": ">= 0.8.0" + } + }, + "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, + "dependencies": { + "debug": "^2.3.3", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "posix-character-classes": "^0.1.0", + "regex-not": "^1.0.0", + "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": { + "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" + } + }, + "node_modules/expand-brackets/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/expand-brackets/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" + } + }, + "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/express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "dev": true, + "dependencies": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + }, + "engines": { + "node": ">= 0.10.0" + } + }, + "node_modules/express/node_modules/array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=", + "dev": true + }, + "node_modules/express/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/express/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "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 + }, + "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, + "dependencies": { + "assign-symbols": "^1.0.0", + "is-extendable": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/external-editor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/external-editor/-/external-editor-3.1.0.tgz", + "integrity": "sha512-hMQ4CX1p1izmuLYyZqLMO/qGNw10wSv9QDCPfzXfyFrOaCSSoRfqE1Kf1s5an66J5JZC62NewG+mK49jOCtQew==", + "dev": true, + "dependencies": { + "chardet": "^0.7.0", + "iconv-lite": "^0.4.24", + "tmp": "^0.0.33" + }, + "engines": { + "node": ">=4" + } + }, + "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, + "dependencies": { + "array-unique": "^0.3.2", + "define-property": "^1.0.0", + "expand-brackets": "^2.1.4", + "extend-shallow": "^2.0.1", + "fragment-cache": "^0.2.1", + "regex-not": "^1.0.0", + "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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extglob/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/extsprintf": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/extsprintf/-/extsprintf-1.3.0.tgz", + "integrity": "sha1-lpGEQOMEGnpBT4xS48V06zw+HgU=", + "dev": true, + "engines": [ + "node >=0.6.0" + ] + }, + "node_modules/fast-deep-equal": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz", + "integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==", + "dev": true + }, + "node_modules/fast-glob": { + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.5.tgz", + "integrity": "sha512-2DtFcgT68wiTTiwZ2hNdJfcHNke9XOfnwmBRWXhmeKM8rF0TGwmC/Qto3S7RoZKp5cilZbxzO5iTNTQsJ+EeDg==", + "dev": true, + "dependencies": { + "@nodelib/fs.stat": "^2.0.2", + "@nodelib/fs.walk": "^1.2.3", + "glob-parent": "^5.1.0", + "merge2": "^1.3.0", + "micromatch": "^4.0.2", + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8" + } + }, + "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 + }, + "node_modules/fastparse": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/fastparse/-/fastparse-1.1.2.tgz", + "integrity": "sha512-483XLLxTVIwWK3QTrMGRqUfUpoOs/0hbQrl2oz4J0pAcm3A3bu84wxTFqGqkJzewCLdME38xJLJAxBABfQT8sQ==", + "dev": true + }, + "node_modules/fastq": { + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.11.0.tgz", + "integrity": "sha512-7Eczs8gIPDrVzT+EksYBcupqMyxSHXXrHOLRRxU2/DicV8789MRBRR8+Hc2uWzUupOs4YS4JzBmBxjjCVBxD/g==", + "dev": true, + "dependencies": { + "reusify": "^1.0.4" + } + }, + "node_modules/faye-websocket": { + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", + "dev": true, + "dependencies": { + "websocket-driver": ">=0.5.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/figgy-pudding": { + "version": "3.5.2", + "resolved": "https://registry.npmjs.org/figgy-pudding/-/figgy-pudding-3.5.2.tgz", + "integrity": "sha512-0btnI/H8f2pavGMN8w40mlSKOfTK2SVJmBfBeVIj3kNw0swwgzyRq0d5TJVOwodFmtvpPeWPN/MCcfuWF0Ezbw==", + "dev": true + }, + "node_modules/figures": { + "version": "3.2.0", + "resolved": "https://registry.npmjs.org/figures/-/figures-3.2.0.tgz", + "integrity": "sha512-yaduQFRKLXYOGgEn6AZau90j3ggSOyiqXU0F9JZfeXYhNa+Jk4X+s45A2zg5jns87GAFa34BBm2kXw4XpNcbdg==", + "dev": true, + "dependencies": { + "escape-string-regexp": "^1.0.5" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/file-loader": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.2.0.tgz", + "integrity": "sha512-qo3glqyTa61Ytg4u73GultjHGjdRyig3tG6lPtyX/jOEJvHif9uB0/OCI2Kif6ctF3caQTW2G5gym21oAsI4pw==", + "dev": true, + "dependencies": { + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/file-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/file-saver": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/file-saver/-/file-saver-2.0.5.tgz", + "integrity": "sha512-P9bmyZ3h/PRG+Nzga+rbdI4OEpNDzAVyy74uVO9ATgzLK6VtAsYybF/+TOCvrc0MO793d6+42lLyZTw7/ArVzA==" + }, + "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, + "optional": true + }, + "node_modules/fill-range": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", + "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", + "dependencies": { + "to-regex-range": "^5.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "dev": true, + "dependencies": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/finalhandler/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/finalhandler/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "dev": true + }, + "node_modules/find-cache-dir": { + "version": "3.3.1", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-3.3.1.tgz", + "integrity": "sha512-t2GDMt3oGC/v+BMwzmllWDuJF/xcDtE5j/fCGbqDD7OLuJkj0cfh1YSA5VKPvwMeLFLNDBkwOKZ2X85jGLVftQ==", + "dev": true, + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^3.0.2", + "pkg-dir": "^4.1.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/avajs/find-cache-dir?sponsor=1" + } + }, + "node_modules/find-cache-dir/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/find-cache-dir/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/find-up": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", + "dev": true, + "dependencies": { + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/flatted": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/flatted/-/flatted-2.0.2.tgz", + "integrity": "sha512-r5wGx7YeOwNWNlCA0wQ86zKyDLMQr+/RB8xy74M4hTphfmjlijTSSXGuH8rnvKZnfT9i+75zmd8jcKdMR4O6jA==", + "dev": true + }, + "node_modules/flush-write-stream": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/flush-write-stream/-/flush-write-stream-1.1.1.tgz", + "integrity": "sha512-3Z4XhFZ3992uIq0XOqb9AreonueSYphE6oYbpt5+3u06JWklbsPkNv3ZKkP9Bz/r+1MWCaMoSQ28P85+1Yc77w==", + "dev": true, + "dependencies": { + "inherits": "^2.0.3", + "readable-stream": "^2.3.6" + } + }, + "node_modules/follow-redirects": { + "version": "1.13.0", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.13.0.tgz", + "integrity": "sha512-aq6gF1BEKje4a9i9+5jimNFIpq4Q1WiwBToeRK5NvZBd/TRsmW8BsJfOEGkr76TbOyPVD3OVDN910EcUNtRYEA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": "*" + } + }, + "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, + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.6", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 0.12" + } + }, + "node_modules/forwarded": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/fraction.js": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.1.1.tgz", + "integrity": "sha512-MHOhvvxHTfRFpF1geTK9czMIZ6xclsEor2wkIGYYq+PxcQqT7vStJqjhe6S1TenZrMZzo+wlqOufBDVepUEgPg==", + "dev": true, + "engines": { + "node": "*" + }, + "funding": { + "type": "patreon", + "url": "https://www.patreon.com/infusion" + } + }, + "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, + "dependencies": { + "map-cache": "^0.2.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/from2": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/from2/-/from2-2.3.0.tgz", + "integrity": "sha1-i/tVAr3kpNNs/e6gB/zKIdfjgq8=", + "dev": true, + "dependencies": { + "inherits": "^2.0.1", + "readable-stream": "^2.0.0" + } + }, + "node_modules/fs-extra": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.2.tgz", + "integrity": "sha1-+RcExT0bRh+JNFKwwwfZmXZHq2s=", + "dependencies": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, + "node_modules/fs-minipass": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-2.1.0.tgz", + "integrity": "sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/fs-write-stream-atomic": { + "version": "1.0.10", + "resolved": "https://registry.npmjs.org/fs-write-stream-atomic/-/fs-write-stream-atomic-1.0.10.tgz", + "integrity": "sha1-tH31NJPvkR33VzHnCp3tAYnbQMk=", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "iferr": "^0.1.5", + "imurmurhash": "^0.1.4", + "readable-stream": "1 || 2" + } + }, + "node_modules/fs.realpath": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", + "integrity": "sha1-FQStJSMVjKpA20onh8sBQRmU6k8=" + }, + "node_modules/fsevents": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", + "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/function-bind": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" + }, + "node_modules/gauge": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", + "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", + "dev": 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/gauge/node_modules/is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "dependencies": { + "number-is-nan": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/gauge/node_modules/string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": 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/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==", + "engines": { + "node": ">=6.9.0" + } + }, + "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==", + "engines": { + "node": "6.* || 8.* || >= 10.*" + } + }, + "node_modules/get-intrinsic": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", + "dev": true, + "dependencies": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/get-stream": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", + "integrity": "sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w==", + "dev": true, + "dependencies": { + "pump": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/getpass": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/getpass/-/getpass-0.1.7.tgz", + "integrity": "sha1-Xv+OPmhNVprkyysSgmBOi6YhSfo=", + "dev": true, + "dependencies": { + "assert-plus": "^1.0.0" + } + }, + "node_modules/glob": { + "version": "7.1.6", + "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz", + "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==", + "dev": true, + "dependencies": { + "fs.realpath": "^1.0.0", + "inflight": "^1.0.4", + "inherits": "2", + "minimatch": "^3.0.4", + "once": "^1.3.0", + "path-is-absolute": "^1.0.0" + }, + "engines": { + "node": "*" + } + }, + "node_modules/glob-parent": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", + "dependencies": { + "is-glob": "^4.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "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==", + "engines": { + "node": ">=4" + } + }, + "node_modules/globby": { + "version": "11.0.3", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.3.tgz", + "integrity": "sha512-ffdmosjA807y7+lA1NM0jELARVmYul/715xiILEjo3hBLPTcirgQNnXECn5g3mtR8TOLCVbkfua1Hpen25/Xcg==", + "dev": true, + "dependencies": { + "array-union": "^2.1.0", + "dir-glob": "^3.0.1", + "fast-glob": "^3.1.1", + "ignore": "^5.1.4", + "merge2": "^1.3.0", + "slash": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/graceful-fs": { + "version": "4.2.4", + "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", + "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==" + }, + "node_modules/handle-thing": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/handle-thing/-/handle-thing-2.0.1.tgz", + "integrity": "sha512-9Qn4yBxelxoh2Ow62nP+Ka/kMnOXRi8BXnRaUwezLNhqelnN49xKz4F/dPP8OYLxLxq6JDtZb2i9XznUQbNPTg==", + "dev": true + }, + "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, + "engines": { + "node": ">=4" + } + }, + "node_modules/har-validator": { + "version": "5.1.5", + "resolved": "https://registry.npmjs.org/har-validator/-/har-validator-5.1.5.tgz", + "integrity": "sha512-nmT2T0lljbxdQZfspsno9hgrG3Uir6Ks5afism62poxqBM6sDnMEuPmzTq8XN0OEwqKLLdh1jQI3qyE66Nzb3w==", + "dev": true, + "dependencies": { + "ajv": "^6.12.3", + "har-schema": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/has": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/has/-/has-1.0.3.tgz", + "integrity": "sha512-f2dvO0VU6Oej7RkWJGrehjbzMAjFp5/VKPp5tTpWIV4JHHZK1/BxbFRtf/siA2SWTe09caDmVtYYzWEIbBS4zw==", + "dependencies": { + "function-bind": "^1.1.1" + }, + "engines": { + "node": ">= 0.4.0" + } + }, + "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, + "dependencies": { + "ansi-regex": "^2.0.0" + }, + "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=", + "engines": { + "node": ">=4" + } + }, + "node_modules/has-symbols": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", + "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/has-unicode": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", + "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=", + "dev": true + }, + "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, + "dependencies": { + "get-value": "^2.0.6", + "has-values": "^1.0.0", + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "is-number": "^3.0.0", + "kind-of": "^4.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-values/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, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/has-values/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": { + "is-buffer": "^1.1.5" + }, + "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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/hash-base": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/hash-base/-/hash-base-3.1.0.tgz", + "integrity": "sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==", + "dev": true, + "dependencies": { + "inherits": "^2.0.4", + "readable-stream": "^3.6.0", + "safe-buffer": "^5.2.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/hash-base/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": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/hash-base/node_modules/safe-buffer": { + "version": "5.2.1", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", + "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "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, + "dependencies": { + "inherits": "^2.0.3", + "minimalistic-assert": "^1.0.1" + } + }, + "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 + }, + "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, + "dependencies": { + "hash.js": "^1.0.3", + "minimalistic-assert": "^1.0.0", + "minimalistic-crypto-utils": "^1.0.1" + } + }, + "node_modules/hosted-git-info": { + "version": "3.0.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-3.0.8.tgz", + "integrity": "sha512-aXpmwoOhRBrw6X3j0h5RloK4x1OzsxMPyxqIHyNfSe2pypkVTZFpEiRoSipPEPlMrh0HW/XsjkJ5WgnCirpNUw==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/hpack.js": { + "version": "2.1.6", + "resolved": "https://registry.npmjs.org/hpack.js/-/hpack.js-2.1.6.tgz", + "integrity": "sha1-h3dMCUnlE/QuhFdbPEVoH63ioLI=", + "dev": true, + "dependencies": { + "inherits": "^2.0.1", + "obuf": "^1.0.0", + "readable-stream": "^2.0.1", + "wbuf": "^1.1.0" + } + }, + "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 + }, + "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 + }, + "node_modules/html-entities": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-1.4.0.tgz", + "integrity": "sha512-8nxjcBcd8wovbeKx7h3wTji4e6+rhaVuPNpMqwWgnHh+N9ToqsCs6XztWRBPQ+UtzsoMAdKZtUENoVzU/EMtZA==", + "dev": true + }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==", + "dev": true + }, + "node_modules/http-cache-semantics": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", + "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", + "dev": true + }, + "node_modules/http-deceiver": { + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/http-deceiver/-/http-deceiver-1.2.7.tgz", + "integrity": "sha1-+nFolEq5pRnTN8sL7HKE3D5yPYc=", + "dev": true + }, + "node_modules/http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "dev": true, + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/http-errors/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/http-parser-js": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.3.tgz", + "integrity": "sha512-t7hjvef/5HEK7RWTdUzVUhl8zkEu+LlaE0IYzdMuvbSDipxBRpOn4Uhw8ZyECEa808iVT8XCjzo6xmYt4CiLZg==", + "dev": true + }, + "node_modules/http-proxy": { + "version": "1.18.1", + "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", + "integrity": "sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ==", + "dev": true, + "dependencies": { + "eventemitter3": "^4.0.0", + "follow-redirects": "^1.0.0", + "requires-port": "^1.0.0" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/http-proxy-agent": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", + "dev": true, + "dependencies": { + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/http-proxy-agent/node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/http-proxy-middleware": { + "version": "0.19.1", + "resolved": "https://registry.npmjs.org/http-proxy-middleware/-/http-proxy-middleware-0.19.1.tgz", + "integrity": "sha512-yHYTgWMQO8VvwNS22eLLloAkvungsKdKTLO8AJlftYIKNfJr3GK3zK0ZCfzDDGUBttdGc8xFy1mCitvNKQtC3Q==", + "dev": true, + "dependencies": { + "http-proxy": "^1.17.0", + "is-glob": "^4.0.0", + "lodash": "^4.17.11", + "micromatch": "^3.1.10" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/http-proxy-middleware/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, + "dependencies": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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, + "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/http-proxy-middleware/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/http-proxy-middleware/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, + "dependencies": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "assert-plus": "^1.0.0", + "jsprim": "^1.2.2", + "sshpk": "^1.7.0" + }, + "engines": { + "node": ">=0.8", + "npm": ">=1.3.7" + } + }, + "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 + }, + "node_modules/https-proxy-agent": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-2.2.4.tgz", + "integrity": "sha512-OmvfoQ53WLjtA9HeYP9RNrWMJzzAz1JGaSFr1nijg0PVR1JaD/xbJq1mdEIIlxGpXp9eSe/O2LgU9DJmTPd0Eg==", + "dev": true, + "dependencies": { + "agent-base": "^4.3.0", + "debug": "^3.1.0" + }, + "engines": { + "node": ">= 4.5.0" + } + }, + "node_modules/https-proxy-agent/node_modules/debug": { + "version": "3.2.6", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", + "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/humanize-ms": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/humanize-ms/-/humanize-ms-1.2.1.tgz", + "integrity": "sha1-xG4xWaKT9riW2ikxbYtv6Lt5u+0=", + "dev": true, + "dependencies": { + "ms": "^2.0.0" + } + }, + "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, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/icss-utils": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", + "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "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 + }, + "node_modules/iferr": { + "version": "0.1.5", + "resolved": "https://registry.npmjs.org/iferr/-/iferr-0.1.5.tgz", + "integrity": "sha1-xg7taebY/bazEEofy8ocGS3FtQE=", + "dev": true + }, + "node_modules/ignore": { + "version": "5.1.8", + "resolved": "https://registry.npmjs.org/ignore/-/ignore-5.1.8.tgz", + "integrity": "sha512-BMpfD7PpiETpBl/A6S498BaIJ6Y/ABT93ETbby2fP00v4EbvPBXWEoaR1UBPKs3iR53pJY7EtZk5KACI57i1Uw==", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/ignore-walk": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.4.tgz", + "integrity": "sha512-PY6Ii8o1jMRA1z4F2hRkH/xN59ox43DavKvD3oDpfurRlOJyAHpifIwpbdv1n4jt4ov0jSpw3kQ4GhJnpBL6WQ==", + "dev": true, + "dependencies": { + "minimatch": "^3.0.4" + } + }, + "node_modules/image-size": { + "version": "0.5.5", + "resolved": "https://registry.npmjs.org/image-size/-/image-size-0.5.5.tgz", + "integrity": "sha1-Cd/Uq50g4p6xw+gLiZA3jfnjy5w=", + "dev": true, + "optional": true, + "bin": { + "image-size": "bin/image-size.js" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/immediate": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/immediate/-/immediate-3.0.6.tgz", + "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=", + "dev": true + }, + "node_modules/import-fresh": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", + "dev": true, + "dependencies": { + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" + }, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/import-local": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/import-local/-/import-local-2.0.0.tgz", + "integrity": "sha512-b6s04m3O+s3CGSbqDIyP4R6aAwAeYlVq9+WUWep6iHa8ETRf9yei1U48C5MmfJmV9AiLYYBKPMq/W+/WRpQmCQ==", + "dev": true, + "dependencies": { + "pkg-dir": "^3.0.0", + "resolve-cwd": "^2.0.0" + }, + "bin": { + "import-local-fixture": "fixtures/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/import-local/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/import-local/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/import-local/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/import-local/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/import-local/node_modules/pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "dependencies": { + "find-up": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/imurmurhash": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/imurmurhash/-/imurmurhash-0.1.4.tgz", + "integrity": "sha1-khi5srkoojixPcT7a21XbyMUU+o=", + "dev": true, + "engines": { + "node": ">=0.8.19" + } + }, + "node_modules/indent-string": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/indent-string/-/indent-string-4.0.0.tgz", + "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/infer-owner": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", + "integrity": "sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A==", + "dev": true + }, + "node_modules/inflight": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz", + "integrity": "sha1-Sb1jMdfQLQwJvJEKEHW6gWW1bfk=", + "dependencies": { + "once": "^1.3.0", + "wrappy": "1" + } + }, + "node_modules/inherits": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz", + "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" + }, + "node_modules/ini": { + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", + "dev": true + }, + "node_modules/inquirer": { + "version": "7.3.3", + "resolved": "https://registry.npmjs.org/inquirer/-/inquirer-7.3.3.tgz", + "integrity": "sha512-JG3eIAj5V9CwcGvuOmoo6LB9kbAYT8HXffUl6memuszlwDC/qvFAJw49XJ5NROSFNPxp3iQg1GqkFhaY/CR0IA==", + "dev": true, + "dependencies": { + "ansi-escapes": "^4.2.1", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-width": "^3.0.0", + "external-editor": "^3.0.3", + "figures": "^3.0.0", + "lodash": "^4.17.19", + "mute-stream": "0.0.8", + "run-async": "^2.4.0", + "rxjs": "^6.6.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0", + "through": "^2.3.6" + }, + "engines": { + "node": ">=8.0.0" + } + }, + "node_modules/inquirer/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/chalk": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", + "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/inquirer/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/inquirer/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/inquirer/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/inquirer/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/string-width": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", + "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/inquirer/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/internal-ip": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/internal-ip/-/internal-ip-4.3.0.tgz", + "integrity": "sha512-S1zBo1D6zcsyuC6PMmY5+55YMILQ9av8lotMx447Bq6SAgo/sDK6y6uUKmuYhW7eacnIhFfsPmCNYdDzsnnDCg==", + "dev": true, + "dependencies": { + "default-gateway": "^4.2.0", + "ipaddr.js": "^1.9.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/ip": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", + "integrity": "sha1-vd7XARQpCCjAoDnnLvJfWq7ENUo=", + "dev": true + }, + "node_modules/ip-regex": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/ip-regex/-/ip-regex-2.1.0.tgz", + "integrity": "sha1-+ni/XS5pE8kRzp+BnuUUa7bYROk=", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "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" + } + }, + "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==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-arguments": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.0.tgz", + "integrity": "sha512-1Ij4lOMPl/xB5kBDn7I+b2ttPMKa8szhEIrXDuXQD/oe3HJLTLhqhgGspwgyGd6MOywBUqVvYicF72lkgDnIHg==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.0" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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 + }, + "node_modules/is-binary-path": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", + "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", + "dependencies": { + "binary-extensions": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "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 + }, + "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, + "dependencies": { + "css-color-names": "^0.0.4", + "hex-color-regex": "^1.1.0", + "hsl-regex": "^1.0.0", + "hsla-regex": "^1.0.0", + "rgb-regex": "^1.0.1", + "rgba-regex": "^1.0.0" + } + }, + "node_modules/is-color-stop/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, + "engines": { + "node": "*" + } + }, + "node_modules/is-core-module": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.4.0.tgz", + "integrity": "sha512-6A2fkfq1rfeQZjxrZJGerpLCTHRNEBiSgnu0+obeJpEPZRUooHgsizvzv0ZjJwOz3iWIHdJtVWJ/tmPr3D21/A==", + "dependencies": { + "has": "^1.0.3" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-date-object": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.4.tgz", + "integrity": "sha512-/b4ZVsG7Z5XVtIxs/h9W8nvfLgSAyKYdtGWQLbqy6jA1icmgjf8WCoTKgeS4wy5tYaPePouzFMANbnj94c2Z+A==", + "dev": true, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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" + } + }, + "node_modules/is-docker": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", + "dev": true, + "bin": { + "is-docker": "cli.js" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "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-plain-object": "^2.0.4" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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=", + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=4" + } + }, + "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==", + "dependencies": { + "is-extglob": "^2.1.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-interactive": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-interactive/-/is-interactive-1.0.0.tgz", + "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/is-lambda": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-lambda/-/is-lambda-1.0.1.tgz", + "integrity": "sha1-PZh3iZ5qU+/AFgUEzeFfgubwYdU=", + "dev": true + }, + "node_modules/is-number": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/is-path-cwd": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-2.2.0.tgz", + "integrity": "sha512-w942bTcih8fdJPJmQHFzkS76NEP8Kzzvmw92cXsazb8intwLqPibPPdXf4ANdKV3rYMuuQYGIWtvz9JilB3NFQ==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/is-path-in-cwd": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-2.1.0.tgz", + "integrity": "sha512-rNocXHgipO+rvnP6dk3zI20RpOtrAM/kzbB258Uw5BWr3TpXi861yzjo16Dn4hUox07iw5AyeMLHWsujkjzvRQ==", + "dev": true, + "dependencies": { + "is-path-inside": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/is-path-inside": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-2.1.0.tgz", + "integrity": "sha512-wiyhTzfDWsvwAW53OBWF5zuvaOGlZ6PwYxAbPVDhpm+gM09xKQGjBq/8uYN12aDvMxnAnq3dxTyoSoRNmg5YFg==", + "dev": true, + "dependencies": { + "path-is-inside": "^1.0.2" + }, + "engines": { + "node": ">=6" + } + }, + "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, + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-regex": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.3.tgz", + "integrity": "sha512-qSVXFz28HM7y+IWX6vLCsexdlvzT1PJNFSBuaQLQ5o0IEw8UDYW6/2+eCMVyIsbM8CNLX2a/QWmSpyxYEHY7CQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "has-symbols": "^1.0.2" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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 + }, + "node_modules/is-stream": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-stream/-/is-stream-1.1.0.tgz", + "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/is-unicode-supported": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/is-what": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-3.14.1.tgz", + "integrity": "sha512-sNxgpk9793nzSs7bA6JQJGeIuRBQhAaNGG77kzYQgMkrID+lS6SlK07K5LaptscDlSaIgH+GPFzf+d75FVxozA==", + "dev": true + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/is-wsl": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/is-wsl/-/is-wsl-2.2.0.tgz", + "integrity": "sha512-fKzAra0rGJUUBwGBgNkHZuToZcn+TtXHpeCgmkMJMMYx1sQDYaCSyjJBSCa2nH1DGm7s3n1oBnohoVTBaN7Lww==", + "dev": true, + "dependencies": { + "is-docker": "^2.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/isarray": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/isarray/-/isarray-1.0.0.tgz", + "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE=", + "dev": true + }, + "node_modules/isbinaryfile": { + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.8.tgz", + "integrity": "sha512-53h6XFniq77YdW+spoRrebh0mnmTxRPTlcuIArO57lmMdq4uBKFKaeTjnb92oYWrSn/LVL+LT+Hap2tFQj8V+w==", + "dev": true, + "engines": { + "node": ">= 8.0.0" + }, + "funding": { + "url": "https://github.com/sponsors/gjtorikian/" + } + }, + "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 + }, + "node_modules/isobject": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/isobject/-/isobject-3.0.1.tgz", + "integrity": "sha1-TkMekrEalzFjaqH5yNHMvP2reN8=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/istanbul-lib-coverage": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-3.0.0.tgz", + "integrity": "sha512-UiUIqxMgRDET6eR+o5HbfRYP1l0hqkWOs7vNxC/mggutCMUIhWMm8gAHb8tHlyfD3/l6rlgNA5cKdDzEAf6hEg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/istanbul-lib-instrument/-/istanbul-lib-instrument-4.0.3.tgz", + "integrity": "sha512-BXgQl9kf4WTCPCCpmFGoJkz/+uhvm7h7PFKUYxh7qarQd3ER33vHG//qaE8eN25l07YqZPpHXU9I09l/RD5aGQ==", + "dev": true, + "dependencies": { + "@babel/core": "^7.7.5", + "@istanbuljs/schema": "^0.1.2", + "istanbul-lib-coverage": "^3.0.0", + "semver": "^6.3.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-instrument/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/istanbul-lib-report": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/istanbul-lib-report/-/istanbul-lib-report-3.0.0.tgz", + "integrity": "sha512-wcdi+uAKzfiGT2abPpKZ0hSU1rGQjUQnLvtY5MpQ7QCTahD3VODhcu4wcfY1YtkGaDD5yuydOLINXsfbus9ROw==", + "dev": true, + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "make-dir": "^3.0.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-report/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-report/node_modules/make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "dependencies": { + "semver": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-report/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/istanbul-lib-report/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/istanbul-lib-source-maps": { + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/istanbul-lib-source-maps/-/istanbul-lib-source-maps-3.0.6.tgz", + "integrity": "sha512-R47KzMtDJH6X4/YW9XTx+jrLnZnscW4VpNN+1PViSYTejLVPWv7oov+Duf8YQSPyVRUvueQqz1TcsC6mooZTXw==", + "dev": true, + "dependencies": { + "debug": "^4.1.1", + "istanbul-lib-coverage": "^2.0.5", + "make-dir": "^2.1.0", + "rimraf": "^2.6.3", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/istanbul-lib-coverage": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.5.tgz", + "integrity": "sha512-8aXznuEPCJvGnMSRft4udDRDtb1V3pkQkMMI5LI+6HuQz5oQ4J2UFn1H82raA3qJtyOLkkwVqICBQkjnGtn5mA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/istanbul-lib-source-maps/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/istanbul-reports": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/istanbul-reports/-/istanbul-reports-3.0.2.tgz", + "integrity": "sha512-9tZvz7AiR3PEDNGiV9vIouQ/EAcqMXFmkcA1CDFTwOB98OZVDL0PH9glHotf5Ugp6GCOTypfzGWI/OqjWNCRUw==", + "dev": true, + "dependencies": { + "html-escaper": "^2.0.0", + "istanbul-lib-report": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/jasmine": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/jasmine/-/jasmine-2.8.0.tgz", + "integrity": "sha1-awicChFXax8W3xG4AUbZHU6Lij4=", + "dev": true, + "dependencies": { + "exit": "^0.1.2", + "glob": "^7.0.6", + "jasmine-core": "~2.8.0" + }, + "bin": { + "jasmine": "bin/jasmine.js" + } + }, + "node_modules/jasmine-core": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/jasmine-core/-/jasmine-core-3.6.0.tgz", + "integrity": "sha512-8uQYa7zJN8hq9z+g8z1bqCfdC8eoDAeVnM5sfqs7KHv9/ifoJ500m018fpFc7RDaO6SWCLCXwo/wPSNcdYTgcw==", + "dev": true + }, + "node_modules/jasmine-spec-reporter": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/jasmine-spec-reporter/-/jasmine-spec-reporter-5.0.2.tgz", + "integrity": "sha512-6gP1LbVgJ+d7PKksQBc2H0oDGNRQI3gKUsWlswKaQ2fif9X5gzhQcgM5+kiJGCQVurOG09jqNhk7payggyp5+g==", + "dev": true, + "dependencies": { + "colors": "1.4.0" + } + }, + "node_modules/jasmine/node_modules/jasmine-core": { + "version": "2.8.0", + "resolved": "https://registry.npmjs.org/jasmine-core/-/jasmine-core-2.8.0.tgz", + "integrity": "sha1-vMl5rh+f0FcB5F5S5l06XWPxok4=", + "dev": true + }, + "node_modules/jasminewd2": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/jasminewd2/-/jasminewd2-2.2.0.tgz", + "integrity": "sha1-43zwsX8ZnM4jvqcbIDk5Uka07E4=", + "dev": true, + "engines": { + "node": ">= 6.9.x" + } + }, + "node_modules/jest-worker": { + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", + "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", + "dev": true, + "dependencies": { + "@types/node": "*", + "merge-stream": "^2.0.0", + "supports-color": "^7.0.0" + }, + "engines": { + "node": ">= 10.13.0" + } + }, + "node_modules/jest-worker/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/jest-worker/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/js-tokens": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz", + "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==" + }, + "node_modules/js-yaml": { + "version": "3.14.0", + "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.14.0.tgz", + "integrity": "sha512-/4IbIeHcD9VMHFqDR/gQ7EdZdLimOvW2DdcxFjdyyZ9NsbS+ccrXqVWDtab/lRl5AlUqmpBx8EhPaWR+OtY17A==", + "dev": true, + "dependencies": { + "argparse": "^1.0.7", + "esprima": "^4.0.0" + }, + "bin": { + "js-yaml": "bin/js-yaml.js" + } + }, + "node_modules/jsbn": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/jsbn/-/jsbn-0.1.1.tgz", + "integrity": "sha1-peZUwuWi3rXyAdls77yoDA7y9RM=", + "dev": true + }, + "node_modules/jsesc": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/jsesc/-/jsesc-2.5.2.tgz", + "integrity": "sha512-OYu7XEzjkCQ3C5Ps3QIZsQfNpqoJyZZA99wd9aWd05NCtC5pWOkShK2mkL6HXQR6/Cy2lbNdPlZBpuQHXE63gA==", + "bin": { + "jsesc": "bin/jsesc" + }, + "engines": { + "node": ">=4" + } + }, + "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 + }, + "node_modules/json-parse-even-better-errors": { + "version": "2.3.1", + "resolved": "https://registry.npmjs.org/json-parse-even-better-errors/-/json-parse-even-better-errors-2.3.1.tgz", + "integrity": "sha512-xyFwyhro/JEof6Ghe2iz2NcXoj2sloNsWr/XsERDK/oiPCfaNhl5ONfp+jQdAZRQQ0IJWNzH9zIZF7li91kh2w==", + "dev": true + }, + "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 + }, + "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 + }, + "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 + }, + "node_modules/json3": { + "version": "3.3.3", + "resolved": "https://registry.npmjs.org/json3/-/json3-3.3.3.tgz", + "integrity": "sha512-c7/8mbUsKigAbLkD5B010BK4D9LZm7A1pNItkEwiUZRpIN66exu/e7YQWysGun+TRKaJp8MhemM+VkfWv42aCA==", + "dev": true + }, + "node_modules/json5": { + "version": "2.1.3", + "resolved": "https://registry.npmjs.org/json5/-/json5-2.1.3.tgz", + "integrity": "sha512-KXPvOm8K9IJKFM0bmdn8QXh7udDh1g/giieX0NLCaMnb4hEiVFqnop2ImTXCc5e0/oHz3LTqmHGtExn5hfMkOA==", + "dependencies": { + "minimist": "^1.2.5" + }, + "bin": { + "json5": "lib/cli.js" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/jsonc-parser": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.0.0.tgz", + "integrity": "sha512-fQzRfAbIBnR0IQvftw9FJveWiHp72Fg20giDrHz6TdfB12UH/uue0D3hm57UB5KgAVuniLMCaS8P1IMj9NR7cA==", + "dev": true + }, + "node_modules/jsonfile": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", + "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", + "optionalDependencies": { + "graceful-fs": "^4.1.6" + } + }, + "node_modules/jsonparse": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/jsonparse/-/jsonparse-1.3.1.tgz", + "integrity": "sha1-P02uSpH6wxX3EGL4UhzCOfE2YoA=", + "dev": true, + "engines": [ + "node >= 0.2.0" + ] + }, + "node_modules/jsprim": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/jsprim/-/jsprim-1.4.1.tgz", + "integrity": "sha1-MT5mvB5cwG5Di8G3SZwuXFastqI=", + "dev": true, + "engines": [ + "node >=0.6.0" + ], + "dependencies": { + "assert-plus": "1.0.0", + "extsprintf": "1.3.0", + "json-schema": "0.2.3", + "verror": "1.10.0" + } + }, + "node_modules/jszip": { + "version": "3.5.0", + "resolved": "https://registry.npmjs.org/jszip/-/jszip-3.5.0.tgz", + "integrity": "sha512-WRtu7TPCmYePR1nazfrtuF216cIVon/3GWOvHS9QR5bIwSbnxtdpma6un3jyGGNhHsKCSzn5Ypk+EkDRvTGiFA==", + "dev": true, + "dependencies": { + "lie": "~3.3.0", + "pako": "~1.0.2", + "readable-stream": "~2.3.6", + "set-immediate-shim": "~1.0.1" + } + }, + "node_modules/karma": { + "version": "6.3.3", + "resolved": "https://registry.npmjs.org/karma/-/karma-6.3.3.tgz", + "integrity": "sha512-JRAujkKWaOtO2LmyPH7K2XXRhrxuFAn9loIL9+iiah6vjz+ZLkqdKsySV9clRITGhj10t9baIfbCl6CJ5hu9gQ==", + "dev": true, + "dependencies": { + "body-parser": "^1.19.0", + "braces": "^3.0.2", + "chokidar": "^3.4.2", + "colors": "^1.4.0", + "connect": "^3.7.0", + "di": "^0.0.1", + "dom-serialize": "^2.2.1", + "glob": "^7.1.6", + "graceful-fs": "^4.2.4", + "http-proxy": "^1.18.1", + "isbinaryfile": "^4.0.6", + "lodash": "^4.17.19", + "log4js": "^6.2.1", + "mime": "^2.4.5", + "minimatch": "^3.0.4", + "qjobs": "^1.2.0", + "range-parser": "^1.2.1", + "rimraf": "^3.0.2", + "socket.io": "^3.1.0", + "source-map": "^0.6.1", + "tmp": "0.2.1", + "ua-parser-js": "^0.7.23", + "yargs": "^16.1.1" + }, + "bin": { + "karma": "bin/karma" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/karma-chrome-launcher": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/karma-chrome-launcher/-/karma-chrome-launcher-3.1.0.tgz", + "integrity": "sha512-3dPs/n7vgz1rxxtynpzZTvb9y/GIaW8xjAwcIGttLbycqoFtI7yo1NGnQi6oFTherRE+GIhCAHZC4vEqWGhNvg==", + "dev": true, + "dependencies": { + "which": "^1.2.1" + } + }, + "node_modules/karma-coverage-istanbul-reporter": { + "version": "3.0.3", + "resolved": "https://registry.npmjs.org/karma-coverage-istanbul-reporter/-/karma-coverage-istanbul-reporter-3.0.3.tgz", + "integrity": "sha512-wE4VFhG/QZv2Y4CdAYWDbMmcAHeS926ZIji4z+FkB2aF/EposRb6DP6G5ncT/wXhqUfAb/d7kZrNKPonbvsATw==", + "dev": true, + "dependencies": { + "istanbul-lib-coverage": "^3.0.0", + "istanbul-lib-report": "^3.0.0", + "istanbul-lib-source-maps": "^3.0.6", + "istanbul-reports": "^3.0.2", + "minimatch": "^3.0.4" + } + }, + "node_modules/karma-jasmine": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/karma-jasmine/-/karma-jasmine-4.0.1.tgz", + "integrity": "sha512-h8XDAhTiZjJKzfkoO1laMH+zfNlra+dEQHUAjpn5JV1zCPtOIVWGQjLBrqhnzQa/hrU2XrZwSyBa6XjEBzfXzw==", + "dev": true, + "dependencies": { + "jasmine-core": "^3.6.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/karma-jasmine-html-reporter": { + "version": "1.5.4", + "resolved": "https://registry.npmjs.org/karma-jasmine-html-reporter/-/karma-jasmine-html-reporter-1.5.4.tgz", + "integrity": "sha512-PtilRLno5O6wH3lDihRnz0Ba8oSn0YUJqKjjux1peoYGwo0AQqrWRbdWk/RLzcGlb+onTyXAnHl6M+Hu3UxG/Q==", + "dev": true + }, + "node_modules/karma-source-map-support": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/karma-source-map-support/-/karma-source-map-support-1.4.0.tgz", + "integrity": "sha512-RsBECncGO17KAoJCYXjv+ckIz+Ii9NCi+9enk+rq6XC81ezYkb4/RHE6CTXdA7IOJqoF3wcaLfVG0CPmE5ca6A==", + "dev": true, + "dependencies": { + "source-map-support": "^0.5.5" + } + }, + "node_modules/karma/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/karma/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/karma/node_modules/cliui": { + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^7.0.0" + } + }, + "node_modules/karma/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/karma/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/karma/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/karma/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/karma/node_modules/mime": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.5.2.tgz", + "integrity": "sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg==", + "dev": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/karma/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/karma/node_modules/string-width": { + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/karma/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/karma/node_modules/tmp": { + "version": "0.2.1", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.2.1.tgz", + "integrity": "sha512-76SUhtfqR2Ijn+xllcI5P1oyannHNHByD80W1q447gU3mp9G9PSpGdWmjUOHRDPiHYacIk66W7ubDTuPF3BEtQ==", + "dev": true, + "dependencies": { + "rimraf": "^3.0.0" + }, + "engines": { + "node": ">=8.17.0" + } + }, + "node_modules/karma/node_modules/wrap-ansi": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/wrap-ansi?sponsor=1" + } + }, + "node_modules/karma/node_modules/y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/karma/node_modules/yargs": { + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", + "dev": true, + "dependencies": { + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", + "require-directory": "^2.1.1", + "string-width": "^4.2.0", + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/karma/node_modules/yargs-parser": { + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==", + "dev": true, + "engines": { + "node": ">=10" + } + }, + "node_modules/killable": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/killable/-/killable-1.0.1.tgz", + "integrity": "sha512-LzqtLKlUwirEUyl/nicirVmNiPvYs7l5n8wOPP7fyJVpUPkvCnW/vuiXGpylGUlnPDnB7311rARzAt3Mhswpjg==", + "dev": true + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/klona": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/klona/-/klona-2.0.4.tgz", + "integrity": "sha512-ZRbnvdg/NxqzC7L9Uyqzf4psi1OM4Cuc+sJAkQPjO6XkQIJTNbfK2Rsmbw8fx1p2mkZdp2FZYo2+LwXYY/uwIA==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/less": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/less/-/less-4.1.1.tgz", + "integrity": "sha512-w09o8tZFPThBscl5d0Ggp3RcrKIouBoQscnOMgFH3n5V3kN/CXGHNfCkRPtxJk6nKryDXaV9aHLK55RXuH4sAw==", + "dev": true, + "dependencies": { + "copy-anything": "^2.0.1", + "parse-node-version": "^1.0.1", + "tslib": "^1.10.0" + }, + "bin": { + "lessc": "bin/lessc" + }, + "engines": { + "node": ">=6" + }, + "optionalDependencies": { + "errno": "^0.1.1", + "graceful-fs": "^4.1.2", + "image-size": "~0.5.0", + "make-dir": "^2.1.0", + "mime": "^1.4.1", + "needle": "^2.5.2", + "source-map": "~0.6.0" + } + }, + "node_modules/less-loader": { + "version": "7.3.0", + "resolved": "https://registry.npmjs.org/less-loader/-/less-loader-7.3.0.tgz", + "integrity": "sha512-Mi8915g7NMaLlgi77mgTTQvK022xKRQBIVDSyfl3ErTuBhmZBQab0mjeJjNNqGbdR+qrfTleKXqbGI4uEFavxg==", + "dev": true, + "dependencies": { + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "less": "^3.5.0 || ^4.0.0", + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/less-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/less/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, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/less/node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", + "dev": true + }, + "node_modules/license-webpack-plugin": { + "version": "2.3.11", + "resolved": "https://registry.npmjs.org/license-webpack-plugin/-/license-webpack-plugin-2.3.11.tgz", + "integrity": "sha512-0iVGoX5vx0WDy8dmwTTpOOMYiGqILyUbDeVMFH52AjgBlS58lHwOlFMSoqg5nY8Kxl6+FRKyUZY/UdlQaOyqDw==", + "dev": true, + "dependencies": { + "@types/webpack-sources": "^0.1.5", + "webpack-sources": "^1.2.0" + } + }, + "node_modules/license-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/license-webpack-plugin/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/lie": { + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/lie/-/lie-3.3.0.tgz", + "integrity": "sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ==", + "dev": true, + "dependencies": { + "immediate": "~3.0.5" + } + }, + "node_modules/lines-and-columns": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz", + "integrity": "sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=", + "dev": true + }, + "node_modules/loader-runner": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-2.4.0.tgz", + "integrity": "sha512-Jsmr89RcXGIwivFY21FcRrisYZfvLMTWx5kOLc+JTxtpBOG6xML0vzbc6SEQG2FO9/4Fc3wW4LVcB5DmGflaRw==", + "dev": true, + "engines": { + "node": ">=4.3.0 <5.0.0 || >=5.10" + } + }, + "node_modules/loader-utils": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-2.0.0.tgz", + "integrity": "sha512-rP4F0h2RaWSvPEkD7BLDFQnvSf+nK+wr3ESUjNTyAGobqrijmW92zc+SO6d4p4B1wh7+B/Jg1mkQe5NYUEHtHQ==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^2.1.2" + }, + "engines": { + "node": ">=8.9.0" + } + }, + "node_modules/locate-path": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", + "dev": true, + "dependencies": { + "p-locate": "^4.1.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/lodash": { + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" + }, + "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 + }, + "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 + }, + "node_modules/log-symbols": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/log-symbols/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/log-symbols/node_modules/chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/log-symbols/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/log-symbols/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/log-symbols/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/log-symbols/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/log4js": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/log4js/-/log4js-6.3.0.tgz", + "integrity": "sha512-Mc8jNuSFImQUIateBFwdOQcmC6Q5maU0VVvdC2R6XMb66/VnT+7WS4D/0EeNMZu1YODmJe5NIn2XftCzEocUgw==", + "dev": true, + "dependencies": { + "date-format": "^3.0.0", + "debug": "^4.1.1", + "flatted": "^2.0.1", + "rfdc": "^1.1.4", + "streamroller": "^2.2.4" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/loglevel": { + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz", + "integrity": "sha512-Hesni4s5UkWkwCGJMQGAh71PaLUmKFM60dHvq0zi/vDhhrzuk+4GgNbTXJ12YYQJn6ZKBDNIjYcuQGKudvqrIw==", + "dev": true, + "engines": { + "node": ">= 0.6.0" + }, + "funding": { + "type": "tidelift", + "url": "https://tidelift.com/funding/github/npm/loglevel" + } + }, + "node_modules/lru-cache": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", + "integrity": "sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/magic-string": { + "version": "0.25.7", + "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz", + "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==", + "dependencies": { + "sourcemap-codec": "^1.4.4" + } + }, + "node_modules/make-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", + "integrity": "sha512-LS9X+dc8KLxXCb8dni79fLIIUA5VyZoyjSMCwTluaXA0o27cCK0bhXkpgw+sTXVpPy/lSO57ilRixqk0vDmtRA==", + "dev": true, + "dependencies": { + "pify": "^4.0.1", + "semver": "^5.6.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/make-dir/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/make-error": { + "version": "1.3.6", + "resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz", + "integrity": "sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==", + "dev": true + }, + "node_modules/make-fetch-happen": { + "version": "8.0.14", + "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-8.0.14.tgz", + "integrity": "sha512-EsS89h6l4vbfJEtBZnENTOFk8mCRpY5ru36Xe5bcX1KYIli2mkSHqoFsp5O1wMDvTJJzxe/4THpCTtygjeeGWQ==", + "dev": true, + "dependencies": { + "agentkeepalive": "^4.1.3", + "cacache": "^15.0.5", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^5.0.0", + "ssri": "^8.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/make-fetch-happen/node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/make-fetch-happen/node_modules/err-code": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", + "dev": true + }, + "node_modules/make-fetch-happen/node_modules/https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "dev": true, + "dependencies": { + "agent-base": "6", + "debug": "4" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/make-fetch-happen/node_modules/promise-retry": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", + "dev": true, + "dependencies": { + "err-code": "^2.0.2", + "retry": "^0.12.0" + }, + "engines": { + "node": ">=10" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "object-visit": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/material-colors": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/material-colors/-/material-colors-1.2.6.tgz", + "integrity": "sha512-6qE4B9deFBIa9YSpOc9O0Sgc43zTeVYbgDT5veRKSlB2+ZuHNoVVxA1L/ckMUayV9Ay9y7Z/SZCLcGteW9i7bg==" + }, + "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, + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/mdn-data": { + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", + "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==", + "dev": true + }, + "node_modules/media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/memory-fs": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", + "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", + "dev": true, + "dependencies": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + }, + "node_modules/merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=", + "dev": true + }, + "node_modules/merge-source-map": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/merge-source-map/-/merge-source-map-1.1.0.tgz", + "integrity": "sha512-Qkcp7P2ygktpMPh2mCQZaf3jhN6D3Z/qVZHSdWvQ+2Ef5HgRAPBO57A77+ENm0CPx2+1Ce/MYKi3ymqdfuqibw==", + "dev": true, + "dependencies": { + "source-map": "^0.6.1" + } + }, + "node_modules/merge-source-map/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/merge-stream": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz", + "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==", + "dev": true + }, + "node_modules/merge2": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz", + "integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/micromatch": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", + "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", + "dev": true, + "dependencies": { + "braces": "^3.0.1", + "picomatch": "^2.2.3" + }, + "engines": { + "node": ">=8.6" + } + }, + "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, + "dependencies": { + "bn.js": "^4.0.0", + "brorand": "^1.0.1" + }, + "bin": { + "miller-rabin": "bin/miller-rabin" + } + }, + "node_modules/miller-rabin/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "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, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/mime-db": { + "version": "1.44.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", + "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.27", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", + "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", + "dev": true, + "dependencies": { + "mime-db": "1.44.0" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mimic-fn": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-2.1.0.tgz", + "integrity": "sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/mini-css-extract-plugin": { + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-1.3.5.tgz", + "integrity": "sha512-tvmzcwqJJXau4OQE5vT72pRT18o2zF+tQJp8CWchqvfQnTlflkzS+dANYcRdyPRWUWRkfmeNTKltx0NZI/b5dQ==", + "dev": true, + "dependencies": { + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0", + "webpack-sources": "^1.1.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.4.0 || ^5.0.0" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/mini-css-extract-plugin/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "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 + }, + "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 + }, + "node_modules/minimatch": { + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", + "dependencies": { + "brace-expansion": "^1.1.7" + }, + "engines": { + "node": "*" + } + }, + "node_modules/minimist": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.5.tgz", + "integrity": "sha512-FM9nNUYrRBAELZQT3xeZQ7fmMOBg6nWNmJKTcgsJeaLstP/UODVpGsr5OhXhhXg6f+qtJ8uiZ+PUxkDWcgIXLw==" + }, + "node_modules/minipass": { + "version": "3.1.3", + "resolved": "https://registry.npmjs.org/minipass/-/minipass-3.1.3.tgz", + "integrity": "sha512-Mgd2GdMVzY+x3IJ+oHnVM+KG3lA5c8tnabyJKmHSaG2kAGpudxuOf8ToDkhumF7UzME7DecbQE9uOZhNm7PuJg==", + "dev": true, + "dependencies": { + "yallist": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-collect": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/minipass-collect/-/minipass-collect-1.0.2.tgz", + "integrity": "sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/minipass-fetch": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/minipass-fetch/-/minipass-fetch-1.3.3.tgz", + "integrity": "sha512-akCrLDWfbdAWkMLBxJEeWTdNsjML+dt5YgOI4gJ53vuO0vrmYQkUPxa6j6V65s9CcePIr2SSWqjT2EcrNseryQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.0", + "minipass-sized": "^1.0.3", + "minizlib": "^2.0.0" + }, + "engines": { + "node": ">=8" + }, + "optionalDependencies": { + "encoding": "^0.1.12" + } + }, + "node_modules/minipass-flush": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", + "integrity": "sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/minipass-json-stream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minipass-json-stream/-/minipass-json-stream-1.0.1.tgz", + "integrity": "sha512-ODqY18UZt/I8k+b7rl2AENgbWE8IDYam+undIJONvigAz8KR5GWblsFTEfQs0WODsjbSXWlm+JHEv8Gr6Tfdbg==", + "dev": true, + "dependencies": { + "jsonparse": "^1.3.1", + "minipass": "^3.0.0" + } + }, + "node_modules/minipass-pipeline": { + "version": "1.2.4", + "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", + "integrity": "sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minipass-sized": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/minipass-sized/-/minipass-sized-1.0.3.tgz", + "integrity": "sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/minizlib": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", + "integrity": "sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg==", + "dev": true, + "dependencies": { + "minipass": "^3.0.0", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/mississippi": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/mississippi/-/mississippi-3.0.0.tgz", + "integrity": "sha512-x471SsVjUtBRtcvd4BzKE9kFC+/2TeWgKCgw0bZcw1b9l2X3QX5vCWgF+KaZaYm87Ss//rHnWryupDrgLvmSkA==", + "dev": true, + "dependencies": { + "concat-stream": "^1.5.0", + "duplexify": "^3.4.2", + "end-of-stream": "^1.1.0", + "flush-write-stream": "^1.0.0", + "from2": "^2.1.0", + "parallel-transform": "^1.1.0", + "pump": "^3.0.0", + "pumpify": "^1.3.3", + "stream-each": "^1.1.0", + "through2": "^2.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "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, + "dependencies": { + "for-in": "^1.0.2", + "is-extendable": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "minimist": "^1.2.5" + }, + "bin": { + "mkdirp": "bin/cmd.js" + } + }, + "node_modules/move-concurrently": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz", + "integrity": "sha1-viwAX9oy4LKa8fBdfEszIUxwH5I=", + "dev": true, + "dependencies": { + "aproba": "^1.1.1", + "copy-concurrently": "^1.0.0", + "fs-write-stream-atomic": "^1.0.8", + "mkdirp": "^0.5.1", + "rimraf": "^2.5.4", + "run-queue": "^1.0.3" + } + }, + "node_modules/move-concurrently/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/ms": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", + "integrity": "sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==" + }, + "node_modules/multicast-dns": { + "version": "6.2.3", + "resolved": "https://registry.npmjs.org/multicast-dns/-/multicast-dns-6.2.3.tgz", + "integrity": "sha512-ji6J5enbMyGRHIAkAOu3WdV8nggqviKCEKtXcOqfphZZtQrmHKycfynJ2V7eVPUA4NhJ6V7Wf4TmGbTwKE9B6g==", + "dev": true, + "dependencies": { + "dns-packet": "^1.3.1", + "thunky": "^1.0.2" + }, + "bin": { + "multicast-dns": "cli.js" + } + }, + "node_modules/multicast-dns-service-types": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/multicast-dns-service-types/-/multicast-dns-service-types-1.1.0.tgz", + "integrity": "sha1-iZ8R2WhuXgXLkbNdXw5jt3PPyQE=", + "dev": true + }, + "node_modules/mute-stream": { + "version": "0.0.8", + "resolved": "https://registry.npmjs.org/mute-stream/-/mute-stream-0.0.8.tgz", + "integrity": "sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA==", + "dev": true + }, + "node_modules/nan": { + "version": "2.14.2", + "resolved": "https://registry.npmjs.org/nan/-/nan-2.14.2.tgz", + "integrity": "sha512-M2ufzIiINKCuDfBSAUr1vWQ+vuVcA9kqx8JJUsbQi6yf1uGRyb7HfpdfUr5qLXf3B/t8dPvcjhKMmlfnP47EzQ==", + "dev": true, + "optional": true + }, + "node_modules/nanoid": { + "version": "3.1.23", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.23.tgz", + "integrity": "sha512-FiB0kzdP0FFVGDKlRLEQ1BgDzU87dy5NnzjeW9YZNt+/c3+q82EQDUwniSAUxp/F0gFNI1ZhKU1FqYsMuqZVnw==", + "dev": true, + "bin": { + "nanoid": "bin/nanoid.cjs" + }, + "engines": { + "node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1" + } + }, + "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, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "fragment-cache": "^0.2.1", + "is-windows": "^1.0.2", + "kind-of": "^6.0.2", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/needle": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/needle/-/needle-2.6.0.tgz", + "integrity": "sha512-KKYdza4heMsEfSWD7VPUIz3zX2XDwOyX2d+geb4vrERZMT5RMU6ujjaD+I5Yr54uZxQ2w6XRTAhHBbSCyovZBg==", + "dev": true, + "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/needle/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "optional": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/neo-async": { + "version": "2.6.2", + "resolved": "https://registry.npmjs.org/neo-async/-/neo-async-2.6.2.tgz", + "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", + "dev": true + }, + "node_modules/ng2-pdf-viewer": { + "version": "6.3.2", + "resolved": "https://registry.npmjs.org/ng2-pdf-viewer/-/ng2-pdf-viewer-6.3.2.tgz", + "integrity": "sha512-H2tBhDd+Lq6CUzK2g54HsCcZDR2wTn1sDjYqKY3yF0Ydasl2R5ppCKynZBU/zge4EKvmHglJI120FbQMpJKDYQ==", + "dependencies": { + "@types/pdfjs-dist": "^2.1.4", + "pdfjs-dist": "^2.4.456", + "tslib": "^1.10.0" + } + }, + "node_modules/ng2-pdf-viewer/node_modules/tslib": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", + "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==" + }, + "node_modules/ngx-bootstrap": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/ngx-bootstrap/-/ngx-bootstrap-6.2.0.tgz", + "integrity": "sha512-5WKHo6/ltkenw4UyXZwED8rODCgp2RGbWurzYzZsF/gH1JO5SN7TJ+AL6kXYk6XM42sDA2WhN9Db+ZPNjiyHnA==", + "peerDependencies": { + "@angular/common": ">=7.0.0", + "@angular/core": ">=7.0.0" + } + }, + "node_modules/ngx-color": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/ngx-color/-/ngx-color-6.2.0.tgz", + "integrity": "sha512-n04tcMnCpOgmI24egST94YwHmnSoAxK8O1T2t3nGrTwWbvw5XBRJvImNFnoNrriBXzc4Gx4hFehH5MU8CZxp1w==", + "dependencies": { + "@ctrl/tinycolor": "^3.1.6", + "material-colors": "^1.2.6", + "tslib": "^2.0.0" + } + }, + "node_modules/ngx-cookie-service": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/ngx-cookie-service/-/ngx-cookie-service-10.1.1.tgz", + "integrity": "sha512-HvBrYHdxMN1NvFJGEIF/8EuAg2fjxj8QwqTv9h6qZGqNLU+lUba8Pb2zRPw1YA+gqKkJawOy5dYNeH0kyPyipw==", + "dependencies": { + "tslib": "^2.0.0" + } + }, + "node_modules/ngx-file-drop": { + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/ngx-file-drop/-/ngx-file-drop-11.1.0.tgz", + "integrity": "sha512-jRqnI58jh9xVYmbYY315TIaSfpzOXRwj8JQOjNALwi/r1a9SVkxp3C069tXo5Kkxi/elsMePN0Be4EmgLf4EVw==", + "dependencies": { + "tslib": "^2.0.0" + }, + "engines": { + "node": ">= 10.0.0", + "npm": ">= 6.9.0" + }, + "peerDependencies": { + "@angular/common": ">=9.0.0", + "@angular/core": ">=9.0.0" + } + }, + "node_modules/ngx-infinite-scroll": { + "version": "9.1.0", + "resolved": "https://registry.npmjs.org/ngx-infinite-scroll/-/ngx-infinite-scroll-9.1.0.tgz", + "integrity": "sha512-ZulbahgFsoPmP8cz7qPGDeFX9nKiSm74aav8vXNSI1ZoPiGYY5FQd8AK+yXqygY7tyCJRyt8Wp3DIg7zgP5dPA==", + "hasInstallScript": true, + "dependencies": { + "@scarf/scarf": "^1.1.0", + "opencollective-postinstall": "^2.0.2" + } + }, + "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_modules/node-forge": { + "version": "0.10.0", + "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.10.0.tgz", + "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA==", + "dev": true, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/node-gyp": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-7.1.2.tgz", + "integrity": "sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ==", + "dev": true, + "dependencies": { + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" + }, + "bin": { + "node-gyp": "bin/node-gyp.js" + }, + "engines": { + "node": ">= 10.12.0" + } + }, + "node_modules/node-gyp/node_modules/which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "node-which": "bin/node-which" + }, + "engines": { + "node": ">= 8" + } + }, + "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, + "dependencies": { + "assert": "^1.1.1", + "browserify-zlib": "^0.2.0", + "buffer": "^4.3.0", + "console-browserify": "^1.1.0", + "constants-browserify": "^1.0.0", + "crypto-browserify": "^3.11.0", + "domain-browser": "^1.1.1", + "events": "^3.0.0", + "https-browserify": "^1.0.0", + "os-browserify": "^0.3.0", + "path-browserify": "0.0.1", + "process": "^0.11.10", + "punycode": "^1.2.4", + "querystring-es3": "^0.2.0", + "readable-stream": "^2.3.3", + "stream-browserify": "^2.0.1", + "stream-http": "^2.7.2", + "string_decoder": "^1.0.0", + "timers-browserify": "^2.0.4", + "tty-browserify": "0.0.0", + "url": "^0.11.0", + "util": "^0.11.0", + "vm-browserify": "^1.0.1" + } + }, + "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.73", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.73.tgz", + "integrity": "sha512-uW7fodD6pyW2FZNZnp/Z3hvWKeEW1Y8R1+1CnErE8cXFXzl5blBOoVB41CvMer6P6Q0S5FXDwcHgFd1Wj0U9zg==", + "dev": true + }, + "node_modules/nopt": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", + "dev": true, + "dependencies": { + "abbrev": "1" + }, + "bin": { + "nopt": "bin/nopt.js" + }, + "engines": { + "node": ">=6" + } + }, + "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==", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-range": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/normalize-range/-/normalize-range-0.1.2.tgz", + "integrity": "sha1-LRDAa9/TEuqXd2laTShDlFa3WUI=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/normalize-url": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.0.1.tgz", + "integrity": "sha512-VU4pzAuh7Kip71XEmO9aNREYAdMHFGTVj/i+CaTImS8x0i1d3jUZkXhqluy/PRgjPLMgsLQulYY3PJ/aSbSjpQ==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/npm-bundled": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.2.tgz", + "integrity": "sha512-x5DHup0SuyQcmL3s7Rx/YQ8sbw/Hzg0rj48eN0dV7hf5cmQq5PXIeioroH3raV1QC1yh3uTYuMThvEQF3iKgGQ==", + "dev": true, + "dependencies": { + "npm-normalize-package-bin": "^1.0.1" + } + }, + "node_modules/npm-install-checks": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/npm-install-checks/-/npm-install-checks-4.0.0.tgz", + "integrity": "sha512-09OmyDkNLYwqKPOnbI8exiOZU2GVVmQp7tgez2BPi5OZC8M82elDAps7sxC4l//uSUtotWqoEIDwjRvWH4qz8w==", + "dev": true, + "dependencies": { + "semver": "^7.1.1" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-normalize-package-bin": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/npm-normalize-package-bin/-/npm-normalize-package-bin-1.0.1.tgz", + "integrity": "sha512-EPfafl6JL5/rU+ot6P3gRSCpPDW5VmIzX959Ob1+ySFUuuYHWHekXpwdUZcKP5C+DS4GEtdJluwBjnsNDl+fSA==", + "dev": true + }, + "node_modules/npm-package-arg": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.1.0.tgz", + "integrity": "sha512-/ep6QDxBkm9HvOhOg0heitSd7JHA1U7y1qhhlRlteYYAi9Pdb/ZV7FW5aHpkrpM8+P+4p/jjR8zCyKPBMBjSig==", + "dev": true, + "dependencies": { + "hosted-git-info": "^3.0.6", + "semver": "^7.0.0", + "validate-npm-package-name": "^3.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-packlist": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-2.2.2.tgz", + "integrity": "sha512-Jt01acDvJRhJGthnUJVF/w6gumWOZxO7IkpY/lsX9//zqQgnF7OJaxgQXcerd4uQOLu7W5bkb4mChL9mdfm+Zg==", + "dev": true, + "dependencies": { + "glob": "^7.1.6", + "ignore-walk": "^3.0.3", + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + }, + "bin": { + "npm-packlist": "bin/index.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-pick-manifest": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-6.1.0.tgz", + "integrity": "sha512-ygs4k6f54ZxJXrzT0x34NybRlLeZ4+6nECAIbr2i0foTnijtS1TJiyzpqtuUAJOps/hO0tNDr8fRV5g+BtRlTw==", + "dev": true, + "dependencies": { + "npm-install-checks": "^4.0.0", + "npm-package-arg": "^8.0.0", + "semver": "^7.0.0" + } + }, + "node_modules/npm-registry-fetch": { + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/npm-registry-fetch/-/npm-registry-fetch-9.0.0.tgz", + "integrity": "sha512-PuFYYtnQ8IyVl6ib9d3PepeehcUeHN9IO5N/iCRhyg9tStQcqGQBRVHmfmMWPDERU3KwZoHFvbJ4FPXPspvzbA==", + "dev": true, + "dependencies": { + "@npmcli/ci-detect": "^1.0.0", + "lru-cache": "^6.0.0", + "make-fetch-happen": "^8.0.9", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/npm-run-path": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/npm-run-path/-/npm-run-path-2.0.2.tgz", + "integrity": "sha1-NakjLfo11wZ7TLLd8jV7GHFTbF8=", + "dev": true, + "dependencies": { + "path-key": "^2.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/npmlog": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", + "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", + "dev": 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/nth-check": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.0.0.tgz", + "integrity": "sha512-i4sc/Kj8htBrAiH1viZ0TgU8Y5XqCaV/FziYK6TBczxmeKm3AEFWqqF3195yKudrarqy7Zu80Ra5dobFjn9X/Q==", + "dev": true, + "dependencies": { + "boolbase": "^1.0.0" + }, + "funding": { + "url": "https://github.com/fb55/nth-check?sponsor=1" + } + }, + "node_modules/number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": "*" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "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": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/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, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object-copy/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" + } + }, + "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-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, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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, + "engines": { + "node": ">= 0.4" + } + }, + "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, + "dependencies": { + "isobject": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/object.assign": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.0", + "define-properties": "^1.1.3", + "has-symbols": "^1.0.1", + "object-keys": "^1.1.1" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "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, + "dependencies": { + "isobject": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/obuf": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", + "integrity": "sha512-PX1wu0AmAdPqOL1mWhqmlOd8kOIZQwGZw6rh7uby9fTc5lhaOWFLX3I6R1hrF9k3zUY40e6igsLGkDXK92LJNg==", + "dev": true + }, + "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, + "dependencies": { + "ee-first": "1.1.1" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/on-headers": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz", + "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/once": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", + "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", + "dependencies": { + "wrappy": "1" + } + }, + "node_modules/onetime": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/onetime/-/onetime-5.1.2.tgz", + "integrity": "sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg==", + "dev": true, + "dependencies": { + "mimic-fn": "^2.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/open": { + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/open/-/open-7.4.0.tgz", + "integrity": "sha512-PGoBCX/lclIWlpS/R2PQuIR4NJoXh6X5AwVzE7WXnWRGvHg7+4TBCgsujUgiPpm0K1y4qvQeWnCWVTpTKZBtvA==", + "dev": true, + "dependencies": { + "is-docker": "^2.0.0", + "is-wsl": "^2.1.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/opencollective-postinstall": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/opencollective-postinstall/-/opencollective-postinstall-2.0.3.tgz", + "integrity": "sha512-8AV/sCtuzUeTo8gQK5qDZzARrulB3egtLzFgteqB2tcT4Mw7B8Kt7JcDHmltjz6FOAHsvTevk70gZEbhM4ZS9Q==", + "bin": { + "opencollective-postinstall": "index.js" + } + }, + "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, + "dependencies": { + "is-wsl": "^1.1.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/opn/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/ora": { + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.3.0.tgz", + "integrity": "sha512-zAKMgGXUim0Jyd6CXK9lraBnD3H5yPGBPPOkC23a2BG6hsm4Zu6OQSjQuEtV0BHDf4aKHcUFvJiGRrFuW3MG8g==", + "dev": true, + "dependencies": { + "bl": "^4.0.3", + "chalk": "^4.1.0", + "cli-cursor": "^3.1.0", + "cli-spinners": "^2.5.0", + "is-interactive": "^1.0.0", + "log-symbols": "^4.0.0", + "strip-ansi": "^6.0.0", + "wcwidth": "^1.0.1" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/ora/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ora/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/ora/node_modules/chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/ora/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/ora/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/ora/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/ora/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/ora/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/original": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/original/-/original-1.0.2.tgz", + "integrity": "sha512-hyBVl6iqqUOJ8FqRe+l/gS8H+kKYjrEndd5Pm1MfBtsEKA038HkkdbAl/72EAXGyonD/PFsvmVG+EvcIpliMBg==", + "dev": true, + "dependencies": { + "url-parse": "^1.4.3" + } + }, + "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 + }, + "node_modules/os-tmpdir": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", + "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/p-finally": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", + "integrity": "sha1-P7z7FbiZpEEjs0ttzBi3JDNqLK4=", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/p-limit": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-2.3.0.tgz", + "integrity": "sha512-//88mFWSJx8lxCzwdAABTJL2MyWB12+eIY7MDL2SqLmAkeKU9qxRvWuSyTjm3FUmpBEMuFfckAIqEaVGUDxb6w==", + "dev": true, + "dependencies": { + "p-try": "^2.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/p-locate": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", + "dev": true, + "dependencies": { + "p-limit": "^2.2.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/p-map": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/p-map/-/p-map-4.0.0.tgz", + "integrity": "sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ==", + "dev": true, + "dependencies": { + "aggregate-error": "^3.0.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/p-retry": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/p-retry/-/p-retry-3.0.1.tgz", + "integrity": "sha512-XE6G4+YTTkT2a0UWb2kjZe8xNwf8bIbnqpc/IS/idOBVhyves0mK5OJgeocjx7q5pvX/6m23xuzVPYT1uGM73w==", + "dev": true, + "dependencies": { + "retry": "^0.12.0" + }, + "engines": { + "node": ">=6" + } + }, + "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" + } + }, + "node_modules/pacote": { + "version": "11.2.4", + "resolved": "https://registry.npmjs.org/pacote/-/pacote-11.2.4.tgz", + "integrity": "sha512-GfTeVQGJ6WyBQbQD4t3ocHbyOmTQLmWjkCKSZPmKiGFKYKNUaM5U2gbLzUW8WG1XmS9yQFnsTFA0k3o1+q4klQ==", + "dev": true, + "dependencies": { + "@npmcli/git": "^2.0.1", + "@npmcli/installed-package-contents": "^1.0.5", + "@npmcli/promise-spawn": "^1.2.0", + "@npmcli/run-script": "^1.3.0", + "cacache": "^15.0.5", + "chownr": "^2.0.0", + "fs-minipass": "^2.1.0", + "infer-owner": "^1.0.4", + "minipass": "^3.1.3", + "mkdirp": "^1.0.3", + "npm-package-arg": "^8.0.1", + "npm-packlist": "^2.1.4", + "npm-pick-manifest": "^6.0.0", + "npm-registry-fetch": "^9.0.0", + "promise-retry": "^1.1.1", + "read-package-json-fast": "^1.1.3", + "rimraf": "^3.0.2", + "ssri": "^8.0.0", + "tar": "^6.1.0" + }, + "bin": { + "pacote": "lib/bin.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/pacote/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/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/parallel-transform": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/parallel-transform/-/parallel-transform-1.2.0.tgz", + "integrity": "sha512-P2vSmIu38uIlvdcU7fDkyrxj33gTUy/ABO5ZUbGowxNCopBq/OoD42bP4UmMrJoPyk4Uqf0mu3mtWBhHCZD8yg==", + "dev": true, + "dependencies": { + "cyclist": "^1.0.1", + "inherits": "^2.0.3", + "readable-stream": "^2.1.5" + } + }, + "node_modules/parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "dependencies": { + "callsites": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/parse-asn1": { + "version": "5.1.6", + "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.6.tgz", + "integrity": "sha512-RnZRo1EPU6JBnra2vGHj0yhp6ebyjBZpmUCLHWiFhxlzvBCCpAuZ7elsBp1PVAbQN0/04VD/19rfzlBSwLstMw==", + "dev": true, + "dependencies": { + "asn1.js": "^5.2.0", + "browserify-aes": "^1.0.0", + "evp_bytestokey": "^1.0.0", + "pbkdf2": "^3.0.3", + "safe-buffer": "^5.1.1" + } + }, + "node_modules/parse-json": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.0.0", + "error-ex": "^1.3.1", + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/parse-node-version": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-node-version/-/parse-node-version-1.0.1.tgz", + "integrity": "sha512-3YHlOa/JgH6Mnpr05jP9eDG254US9ek25LyIxZlDItp2iJtwyaXQb57lBYLdT3MowkUFYEV2XXNAYIPlESvJlA==", + "dev": true, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/parse5": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", + "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", + "dev": true + }, + "node_modules/parse5-html-rewriting-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5-html-rewriting-stream/-/parse5-html-rewriting-stream-6.0.1.tgz", + "integrity": "sha512-vwLQzynJVEfUlURxgnf51yAJDQTtVpNyGD8tKi2Za7m+akukNHxCcUQMAa/mUGLhCeicFdpy7Tlvj8ZNKadprg==", + "dev": true, + "dependencies": { + "parse5": "^6.0.1", + "parse5-sax-parser": "^6.0.1" + } + }, + "node_modules/parse5-htmlparser2-tree-adapter": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-6.0.1.tgz", + "integrity": "sha512-qPuWvbLgvDGilKc5BoicRovlT4MtYT6JfJyBOMDsKoiT+GiuP5qyrPCnR9HcPECIJJmZh5jRndyNThnhhb/vlA==", + "dev": true, + "dependencies": { + "parse5": "^6.0.1" + } + }, + "node_modules/parse5-sax-parser": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5-sax-parser/-/parse5-sax-parser-6.0.1.tgz", + "integrity": "sha512-kXX+5S81lgESA0LsDuGjAlBybImAChYRMT+/uKCEXFBFOeEhS52qUCydGhU3qLRD8D9DVjaUo821WK7DM4iCeg==", + "dev": true, + "dependencies": { + "parse5": "^6.0.1" + } + }, + "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, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/pascalcase": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/pascalcase/-/pascalcase-0.1.1.tgz", + "integrity": "sha1-s2PlXoAGym/iF4TS2yK9FdeRfxQ=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "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 + }, + "node_modules/path-exists": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/path-is-absolute": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz", + "integrity": "sha1-F0uSaHNVNP+8es5r9TpanhtcX18=", + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/path-is-inside": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/path-is-inside/-/path-is-inside-1.0.2.tgz", + "integrity": "sha1-NlQX3t5EQw0cEa9hAn+s8HS9/FM=", + "dev": true + }, + "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, + "engines": { + "node": ">=4" + } + }, + "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==" + }, + "node_modules/path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=", + "dev": true + }, + "node_modules/path-type": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-type/-/path-type-4.0.0.tgz", + "integrity": "sha512-gDKb8aZMDeD/tZWs9P6+q0J9Mwkdl6xMV8TjnGP3qJVJ06bdMgkbBlLU8IdfOsIsFz2BW1rNVT3XuNEl8zPAvw==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/pbkdf2": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz", + "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==", + "dev": true, + "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" + } + }, + "node_modules/pdfjs-dist": { + "version": "2.5.207", + "resolved": "https://registry.npmjs.org/pdfjs-dist/-/pdfjs-dist-2.5.207.tgz", + "integrity": "sha512-xGDUhnCYPfHy+unMXCLCJtlpZaaZ17Ew3WIL0tnSgKFUZXHAPD49GO9xScyszSsQMoutNDgRb+rfBXIaX/lJbw==" + }, + "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 + }, + "node_modules/picomatch": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==", + "engines": { + "node": ">=8.6" + }, + "funding": { + "url": "https://github.com/sponsors/jonschlinkert" + } + }, + "node_modules/pify": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/pify/-/pify-4.0.1.tgz", + "integrity": "sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/pinkie": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/pinkie/-/pinkie-2.0.4.tgz", + "integrity": "sha1-clVrgM+g1IqXToDnckjoDtT3+HA=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pinkie-promise": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pinkie-promise/-/pinkie-promise-2.0.1.tgz", + "integrity": "sha1-ITXW36ejWMBprJsXh3YogihFD/o=", + "dev": true, + "dependencies": { + "pinkie": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pkg-dir": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", + "dev": true, + "dependencies": { + "find-up": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/pnp-webpack-plugin": { + "version": "1.6.4", + "resolved": "https://registry.npmjs.org/pnp-webpack-plugin/-/pnp-webpack-plugin-1.6.4.tgz", + "integrity": "sha512-7Wjy+9E3WwLOEL30D+m8TSTF7qJJUJLONBnwQp0518siuMxUQUbgZwssaFX+QKlZkjHZcw/IpZCt/H0srrntSg==", + "dev": true, + "dependencies": { + "ts-pnp": "^1.1.6" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/portfinder": { + "version": "1.0.28", + "resolved": "https://registry.npmjs.org/portfinder/-/portfinder-1.0.28.tgz", + "integrity": "sha512-Se+2isanIcEqf2XMHjyUKskczxbPH7dQnlMjXX6+dybayyHvAf/TCgyMRlzf/B6QDhAEFOGes0pzRo3by4AbMA==", + "dev": true, + "dependencies": { + "async": "^2.6.2", + "debug": "^3.1.1", + "mkdirp": "^0.5.5" + }, + "engines": { + "node": ">= 0.12.0" + } + }, + "node_modules/portfinder/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/postcss": { + "version": "8.2.15", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.2.15.tgz", + "integrity": "sha512-2zO3b26eJD/8rb106Qu2o7Qgg52ND5HPjcyQiK2B98O388h43A448LCslC0dI2P97wCAQRJsFvwTRcXxTKds+Q==", + "dev": true, + "dependencies": { + "colorette": "^1.2.2", + "nanoid": "^3.1.23", + "source-map": "^0.6.1" + }, + "engines": { + "node": "^10 || ^12 || >=14" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + } + }, + "node_modules/postcss-calc": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-8.0.0.tgz", + "integrity": "sha512-5NglwDrcbiy8XXfPM11F3HeC6hoT9W7GUH/Zi5U/p7u3Irv4rHhdDcIZwG0llHXV4ftsBjpfWMXAnXNl4lnt8g==", + "dev": true, + "dependencies": { + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.0.2" + }, + "peerDependencies": { + "postcss": "^8.2.2" + } + }, + "node_modules/postcss-colormin": { + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-5.2.0.tgz", + "integrity": "sha512-+HC6GfWU3upe5/mqmxuqYZ9B2Wl4lcoUUNkoaX59nEWV4EtADCMiBqui111Bu8R8IvaZTmqmxrqOAqjbHIwXPw==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.6", + "caniuse-api": "^3.0.0", + "colord": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-convert-values": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-5.0.1.tgz", + "integrity": "sha512-C3zR1Do2BkKkCgC0g3sF8TS0koF2G+mN8xxayZx3f10cIRmTaAnpgpRQZjNekTZxM2ciSPoh2IWJm0VZx8NoQg==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-comments": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-5.0.1.tgz", + "integrity": "sha512-lgZBPTDvWrbAYY1v5GYEv8fEO/WhKOu/hmZqmCYfrpD6eyDWWzAOsl2rF29lpvziKO02Gc5GJQtlpkTmakwOWg==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-duplicates": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-5.0.1.tgz", + "integrity": "sha512-svx747PWHKOGpAXXQkCc4k/DsWo+6bc5LsVrAsw+OU+Ibi7klFZCyX54gjYzX4TH+f2uzXjRviLARxkMurA2bA==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-empty": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-5.0.1.tgz", + "integrity": "sha512-vfU8CxAQ6YpMxV2SvMcMIyF2LX1ZzWpy0lqHDsOdaKKLQVQGVP1pzhrI9JlsO65s66uQTfkQBKBD/A5gp9STFw==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-discard-overridden": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-5.0.1.tgz", + "integrity": "sha512-Y28H7y93L2BpJhrdUR2SR2fnSsT+3TVx1NmVQLbcnZWwIUpJ7mfcTC6Za9M2PG6w8j7UQRfzxqn8jU2VwFxo3Q==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-import": { + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-14.0.0.tgz", + "integrity": "sha512-gFDDzXhqr9ELmnLHgCC3TbGfA6Dm/YMb/UN8/f7Uuq4fL7VTk2vOIj6hwINEwbokEmp123bLD7a5m+E+KIetRg==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.0.0", + "read-cache": "^1.0.0", + "resolve": "^1.1.7" + }, + "engines": { + "node": ">=10.0.0" + }, + "peerDependencies": { + "postcss": "^8.0.0" + } + }, + "node_modules/postcss-loader": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-4.2.0.tgz", + "integrity": "sha512-mqgScxHqbiz1yxbnNcPdKYo/6aVt+XExURmEbQlviFVWogDbM4AJ0A/B+ZBpYsJrTRxKw7HyRazg9x0Q9SWwLA==", + "dev": true, + "dependencies": { + "cosmiconfig": "^7.0.0", + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0", + "semver": "^7.3.4" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "postcss": "^7.0.0 || ^8.0.1", + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/postcss-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/postcss-loader/node_modules/semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "dependencies": { + "lru-cache": "^6.0.0" + }, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/postcss-merge-longhand": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-5.0.2.tgz", + "integrity": "sha512-BMlg9AXSI5G9TBT0Lo/H3PfUy63P84rVz3BjCFE9e9Y9RXQZD3+h3YO1kgTNsNJy7bBc1YQp8DmSnwLIW5VPcw==", + "dev": true, + "dependencies": { + "css-color-names": "^1.0.1", + "postcss-value-parser": "^4.1.0", + "stylehacks": "^5.0.1" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-merge-rules": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-5.0.2.tgz", + "integrity": "sha512-5K+Md7S3GwBewfB4rjDeol6V/RZ8S+v4B66Zk2gChRqLTCC8yjnHQ601omj9TKftS19OPGqZ/XzoqpzNQQLwbg==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.6", + "caniuse-api": "^3.0.0", + "cssnano-utils": "^2.0.1", + "postcss-selector-parser": "^6.0.5", + "vendors": "^1.0.3" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-font-values": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-5.0.1.tgz", + "integrity": "sha512-7JS4qIsnqaxk+FXY1E8dHBDmraYFWmuL6cgt0T1SWGRO5bzJf8sUoelwa4P88LEWJZweHevAiDKxHlofuvtIoA==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-gradients": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-5.0.1.tgz", + "integrity": "sha512-odOwBFAIn2wIv+XYRpoN2hUV3pPQlgbJ10XeXPq8UY2N+9ZG42xu45lTn/g9zZ+d70NKSQD6EOi6UiCMu3FN7g==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "is-color-stop": "^1.1.0", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-params": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-5.0.1.tgz", + "integrity": "sha512-4RUC4k2A/Q9mGco1Z8ODc7h+A0z7L7X2ypO1B6V8057eVK6mZ6xwz6QN64nHuHLbqbclkX1wyzRnIrdZehTEHw==", + "dev": true, + "dependencies": { + "alphanum-sort": "^1.0.2", + "browserslist": "^4.16.0", + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0", + "uniqs": "^2.0.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-minify-selectors": { + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-5.1.0.tgz", + "integrity": "sha512-NzGBXDa7aPsAcijXZeagnJBKBPMYLaJJzB8CQh6ncvyl2sIndLVWfbcDi0SBjRWk5VqEjXvf8tYwzoKf4Z07og==", + "dev": true, + "dependencies": { + "alphanum-sort": "^1.0.2", + "postcss-selector-parser": "^6.0.5" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-modules-extract-imports": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.0.0.tgz", + "integrity": "sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-local-by-default": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.0.0.tgz", + "integrity": "sha512-sT7ihtmGSF9yhm6ggikHdV0hlziDTX7oFoXtuVWeDd3hHObNkcHRo9V3yg7vCAY7cONyxJC/XXCmmiHHcvX7bQ==", + "dev": true, + "dependencies": { + "icss-utils": "^5.0.0", + "postcss-selector-parser": "^6.0.2", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-scope": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-3.0.0.tgz", + "integrity": "sha512-hncihwFA2yPath8oZ15PZqvWGkWf+XUfQgUGamS4LqoP1anQLOsOJw0vr7J7IwLpoY9fatA2qiGUGmuZL0Iqlg==", + "dev": true, + "dependencies": { + "postcss-selector-parser": "^6.0.4" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-modules-values": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz", + "integrity": "sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==", + "dev": true, + "dependencies": { + "icss-utils": "^5.0.0" + }, + "engines": { + "node": "^10 || ^12 || >= 14" + }, + "peerDependencies": { + "postcss": "^8.1.0" + } + }, + "node_modules/postcss-normalize-charset": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-5.0.1.tgz", + "integrity": "sha512-6J40l6LNYnBdPSk+BHZ8SF+HAkS4q2twe5jnocgd+xWpz/mx/5Sa32m3W1AA8uE8XaXN+eg8trIlfu8V9x61eg==", + "dev": true, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-display-values": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-5.0.1.tgz", + "integrity": "sha512-uupdvWk88kLDXi5HEyI9IaAJTE3/Djbcrqq8YgjvAVuzgVuqIk3SuJWUisT2gaJbZm1H9g5k2w1xXilM3x8DjQ==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-positions": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-5.0.1.tgz", + "integrity": "sha512-rvzWAJai5xej9yWqlCb1OWLd9JjW2Ex2BCPzUJrbaXmtKtgfL8dBMOOMTX6TnvQMtjk3ei1Lswcs78qKO1Skrg==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-repeat-style": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-5.0.1.tgz", + "integrity": "sha512-syZ2itq0HTQjj4QtXZOeefomckiV5TaUO6ReIEabCh3wgDs4Mr01pkif0MeVwKyU/LHEkPJnpwFKRxqWA/7O3w==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-string": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-5.0.1.tgz", + "integrity": "sha512-Ic8GaQ3jPMVl1OEn2U//2pm93AXUcF3wz+OriskdZ1AOuYV25OdgS7w9Xu2LO5cGyhHCgn8dMXh9bO7vi3i9pA==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-timing-functions": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-5.0.1.tgz", + "integrity": "sha512-cPcBdVN5OsWCNEo5hiXfLUnXfTGtSFiBU9SK8k7ii8UD7OLuznzgNRYkLZow11BkQiiqMcgPyh4ZqXEEUrtQ1Q==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-unicode": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-5.0.1.tgz", + "integrity": "sha512-kAtYD6V3pK0beqrU90gpCQB7g6AOfP/2KIPCVBKJM2EheVsBQmx/Iof+9zR9NFKLAx4Pr9mDhogB27pmn354nA==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.0", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-url": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-5.0.2.tgz", + "integrity": "sha512-k4jLTPUxREQ5bpajFQZpx8bCF2UrlqOTzP9kEqcEnOfwsRshWs2+oAFIHfDQB8GO2PaUaSE0NlTAYtbluZTlHQ==", + "dev": true, + "dependencies": { + "is-absolute-url": "^3.0.3", + "normalize-url": "^6.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-normalize-whitespace": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-5.0.1.tgz", + "integrity": "sha512-iPklmI5SBnRvwceb/XH568yyzK0qRVuAG+a1HFUsFRf11lEJTiQQa03a4RSCQvLKdcpX7XsI1Gen9LuLoqwiqA==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-ordered-values": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-5.0.2.tgz", + "integrity": "sha512-8AFYDSOYWebJYLyJi3fyjl6CqMEG/UVworjiyK1r573I56kb3e879sCJLGvR3merj+fAdPpVplXKQZv+ey6CgQ==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-reduce-initial": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-5.0.1.tgz", + "integrity": "sha512-zlCZPKLLTMAqA3ZWH57HlbCjkD55LX9dsRyxlls+wfuRfqCi5mSlZVan0heX5cHr154Dq9AfbH70LyhrSAezJw==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.0", + "caniuse-api": "^3.0.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-reduce-transforms": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-5.0.1.tgz", + "integrity": "sha512-a//FjoPeFkRuAguPscTVmRQUODP+f3ke2HqFNgGPwdYnpeC29RZdCBvGRGTsKpMURb/I3p6jdKoBQ2zI+9Q7kA==", + "dev": true, + "dependencies": { + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-selector-parser": { + "version": "6.0.6", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.6.tgz", + "integrity": "sha512-9LXrvaaX3+mcv5xkg5kFwqSzSH1JIObIx51PrndZwlmznwXRfxMddDvo9gve3gVR8ZTKgoFDdWkbRFmEhT4PMg==", + "dev": true, + "dependencies": { + "cssesc": "^3.0.0", + "util-deprecate": "^1.0.2" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/postcss-svgo": { + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-5.0.2.tgz", + "integrity": "sha512-YzQuFLZu3U3aheizD+B1joQ94vzPfE6BNUcSYuceNxlVnKKsOtdo6hL9/zyC168Q8EwfLSgaDSalsUGa9f2C0A==", + "dev": true, + "dependencies": { + "postcss-value-parser": "^4.1.0", + "svgo": "^2.3.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-unique-selectors": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-5.0.1.tgz", + "integrity": "sha512-gwi1NhHV4FMmPn+qwBNuot1sG1t2OmacLQ/AX29lzyggnjd+MnVD5uqQmpXO3J17KGL2WAxQruj1qTd3H0gG/w==", + "dev": true, + "dependencies": { + "alphanum-sort": "^1.0.2", + "postcss-selector-parser": "^6.0.5", + "uniqs": "^2.0.0" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/postcss-value-parser": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.1.0.tgz", + "integrity": "sha512-97DXOFbQJhk71ne5/Mt6cOu6yxsSfM0QGQyl0L25Gca4yGWEGJaig7l7gbCX623VqTBNGLRLaVUCnNkcedlRSQ==", + "dev": true + }, + "node_modules/postcss/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/pretty-bytes": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.6.0.tgz", + "integrity": "sha512-FFw039TmrBqFK8ma/7OL3sDz/VytdtJr044/QUJtH0wK9lb9jLq9tJyIxUwtQJHwar2BqtiA4iCWSwo9JLkzFg==", + "dev": true, + "engines": { + "node": ">=6" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/process": { + "version": "0.11.10", + "resolved": "https://registry.npmjs.org/process/-/process-0.11.10.tgz", + "integrity": "sha1-czIwDoQBYb2j5podHZGn1LwW8YI=", + "dev": true, + "engines": { + "node": ">= 0.6.0" + } + }, + "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 + }, + "node_modules/promise-inflight": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/promise-inflight/-/promise-inflight-1.0.1.tgz", + "integrity": "sha1-mEcocL8igTL8vdhoEputEsPAKeM=", + "dev": true + }, + "node_modules/promise-retry": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-1.1.1.tgz", + "integrity": "sha1-ZznpaOMFHaIM5kl/srUPaRHfPW0=", + "dev": true, + "dependencies": { + "err-code": "^1.0.0", + "retry": "^0.10.0" + }, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/promise-retry/node_modules/retry": { + "version": "0.10.1", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.10.1.tgz", + "integrity": "sha1-52OI0heZLCUnUCQdPTlW/tmNj/Q=", + "dev": true, + "engines": { + "node": "*" + } + }, + "node_modules/protractor": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/protractor/-/protractor-7.0.0.tgz", + "integrity": "sha512-UqkFjivi4GcvUQYzqGYNe0mLzfn5jiLmO8w9nMhQoJRLhy2grJonpga2IWhI6yJO30LibWXJJtA4MOIZD2GgZw==", + "dev": true, + "dependencies": { + "@types/q": "^0.0.32", + "@types/selenium-webdriver": "^3.0.0", + "blocking-proxy": "^1.0.0", + "browserstack": "^1.5.1", + "chalk": "^1.1.3", + "glob": "^7.0.3", + "jasmine": "2.8.0", + "jasminewd2": "^2.1.0", + "q": "1.4.1", + "saucelabs": "^1.5.0", + "selenium-webdriver": "3.6.0", + "source-map-support": "~0.4.0", + "webdriver-js-extender": "2.1.0", + "webdriver-manager": "^12.1.7", + "yargs": "^15.3.1" + }, + "bin": { + "protractor": "bin/protractor", + "webdriver-manager": "bin/webdriver-manager" + }, + "engines": { + "node": ">=10.13.x" + } + }, + "node_modules/protractor/node_modules/@types/q": { + "version": "0.0.32", + "resolved": "https://registry.npmjs.org/@types/q/-/q-0.0.32.tgz", + "integrity": "sha1-vShOV8hPEyXacCur/IKlMoGQwMU=", + "dev": true + }, + "node_modules/protractor/node_modules/ansi-regex": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/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/protractor/node_modules/array-union": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/array-union/-/array-union-1.0.2.tgz", + "integrity": "sha1-mjRBDk9OPaI96jdb5b5w8kd47Dk=", + "dev": true, + "dependencies": { + "array-uniq": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/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/protractor/node_modules/cliui": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", + "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", + "dev": true, + "dependencies": { + "string-width": "^4.2.0", + "strip-ansi": "^6.0.0", + "wrap-ansi": "^6.2.0" + } + }, + "node_modules/protractor/node_modules/cliui/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/protractor/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/protractor/node_modules/del": { + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/del/-/del-2.2.2.tgz", + "integrity": "sha1-wSyYHQZ4RshLyvhiz/kw2Qf/0ag=", + "dev": true, + "dependencies": { + "globby": "^5.0.0", + "is-path-cwd": "^1.0.0", + "is-path-in-cwd": "^1.0.0", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0", + "rimraf": "^2.2.8" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/emoji-regex": { + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", + "dev": true + }, + "node_modules/protractor/node_modules/globby": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz", + "integrity": "sha1-69hGZ8oNuzMLmbz8aOrCvFQ3Dg0=", + "dev": true, + "dependencies": { + "array-union": "^1.0.1", + "arrify": "^1.0.0", + "glob": "^7.0.3", + "object-assign": "^4.0.1", + "pify": "^2.0.0", + "pinkie-promise": "^2.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/is-fullwidth-code-point": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/is-path-cwd": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-path-cwd/-/is-path-cwd-1.0.0.tgz", + "integrity": "sha1-0iXsIxMuie3Tj9p2dHLmLmXxEG0=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/is-path-in-cwd": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-in-cwd/-/is-path-in-cwd-1.0.1.tgz", + "integrity": "sha512-FjV1RTW48E7CWM7eE/J2NJvAEEVektecDBVBE5Hh3nM1Jd0kvhHtX68Pr3xsDf857xt3Y4AkwVULK1Vku62aaQ==", + "dev": true, + "dependencies": { + "is-path-inside": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/is-path-inside": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-path-inside/-/is-path-inside-1.0.1.tgz", + "integrity": "sha1-jvW33lBDej/cprToZe96pVy0gDY=", + "dev": true, + "dependencies": { + "path-is-inside": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/protractor/node_modules/q": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/q/-/q-1.4.1.tgz", + "integrity": "sha1-VXBbzZPF82c1MMLCy8DCs63cKG4=", + "dev": true, + "engines": { + "node": ">=0.6.0", + "teleport": ">=0.2.0" + } + }, + "node_modules/protractor/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/protractor/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/protractor/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/protractor/node_modules/source-map-support": { + "version": "0.4.18", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.4.18.tgz", + "integrity": "sha512-try0/JqxPLF9nOjvSta7tVondkP5dwgyLDjVoyMDlmjugT2lRZ1OfsrYTkCd2hkDnJTKRbO/Rl3orm8vlsUzbA==", + "dev": true, + "dependencies": { + "source-map": "^0.5.6" + } + }, + "node_modules/protractor/node_modules/string-width": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", + "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "dev": true, + "dependencies": { + "emoji-regex": "^8.0.0", + "is-fullwidth-code-point": "^3.0.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/string-width/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/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/protractor/node_modules/webdriver-manager": { + "version": "12.1.7", + "resolved": "https://registry.npmjs.org/webdriver-manager/-/webdriver-manager-12.1.7.tgz", + "integrity": "sha512-XINj6b8CYuUYC93SG3xPkxlyUc3IJbD6Vvo75CVGuG9uzsefDzWQrhz0Lq8vbPxtb4d63CZdYophF8k8Or/YiA==", + "dev": true, + "dependencies": { + "adm-zip": "^0.4.9", + "chalk": "^1.1.1", + "del": "^2.2.0", + "glob": "^7.0.3", + "ini": "^1.3.4", + "minimist": "^1.2.0", + "q": "^1.4.1", + "request": "^2.87.0", + "rimraf": "^2.5.2", + "semver": "^5.3.0", + "xml2js": "^0.4.17" + }, + "bin": { + "webdriver-manager": "bin/webdriver-manager" + }, + "engines": { + "node": ">=6.9.x" + } + }, + "node_modules/protractor/node_modules/wrap-ansi": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", + "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.0.0", + "string-width": "^4.1.0", + "strip-ansi": "^6.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/wrap-ansi/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/wrap-ansi/node_modules/strip-ansi": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", + "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", + "dev": true, + "dependencies": { + "ansi-regex": "^5.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/yargs": { + "version": "15.4.1", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", + "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", + "dev": true, + "dependencies": { + "cliui": "^6.0.0", + "decamelize": "^1.2.0", + "find-up": "^4.1.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^4.2.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^18.1.2" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/protractor/node_modules/yargs-parser": { + "version": "18.1.3", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", + "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", + "dev": true, + "dependencies": { + "camelcase": "^5.0.0", + "decamelize": "^1.2.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/proxy-addr": { + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", + "dev": true, + "dependencies": { + "forwarded": "0.2.0", + "ipaddr.js": "1.9.1" + }, + "engines": { + "node": ">= 0.10" + } + }, + "node_modules/prr": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/prr/-/prr-1.0.1.tgz", + "integrity": "sha1-0/wRS6BplaRexok/SEzrHXj19HY=", + "dev": true + }, + "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 + }, + "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, + "dependencies": { + "bn.js": "^4.1.0", + "browserify-rsa": "^4.0.0", + "create-hash": "^1.1.0", + "parse-asn1": "^5.0.0", + "randombytes": "^2.0.1", + "safe-buffer": "^5.1.2" + } + }, + "node_modules/public-encrypt/node_modules/bn.js": { + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", + "dev": true + }, + "node_modules/pump": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pump/-/pump-3.0.0.tgz", + "integrity": "sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww==", + "dev": true, + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "node_modules/pumpify": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/pumpify/-/pumpify-1.5.1.tgz", + "integrity": "sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==", + "dev": true, + "dependencies": { + "duplexify": "^3.6.0", + "inherits": "^2.0.3", + "pump": "^2.0.0" + } + }, + "node_modules/pumpify/node_modules/pump": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/pump/-/pump-2.0.1.tgz", + "integrity": "sha512-ruPMNRkN3MHP1cWJc9OWr+T/xDP0jhXYCLfJcBuX54hhfIBnaQmAUMfDcG4DM5UMWByBbJY69QSphm3jtDKIkA==", + "dev": true, + "dependencies": { + "end-of-stream": "^1.1.0", + "once": "^1.3.1" + } + }, + "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, + "engines": { + "node": ">=6" + } + }, + "node_modules/qjobs": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/qjobs/-/qjobs-1.2.0.tgz", + "integrity": "sha512-8YOJEHtxpySA3fFDyCRxA+UUV+fA+rTWnuWvylOK/NCjhY+b4ocCtmu8TtsWb+mYeU+GCHf/S66KZF/AsteKHg==", + "dev": true, + "engines": { + "node": ">=0.9" + } + }, + "node_modules/qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", + "dev": true, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/querystring": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", + "integrity": "sha1-sgmEkgO7Jd+CDadW50cAWHhSFiA=", + "dev": true, + "engines": { + "node": ">=0.4.x" + } + }, + "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, + "engines": { + "node": ">=0.4.x" + } + }, + "node_modules/querystringify": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/querystringify/-/querystringify-2.2.0.tgz", + "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==", + "dev": true + }, + "node_modules/queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ] + }, + "node_modules/randombytes": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", + "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==", + "dev": true, + "dependencies": { + "safe-buffer": "^5.1.0" + } + }, + "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, + "dependencies": { + "randombytes": "^2.0.5", + "safe-buffer": "^5.1.0" + } + }, + "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, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "dev": true, + "dependencies": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/raw-body/node_modules/bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/raw-loader": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/raw-loader/-/raw-loader-4.0.2.tgz", + "integrity": "sha512-ZnScIV3ag9A4wPX/ZayxL/jZH+euYb6FcUinPcgiQW0+UBtEv0O6Q3lGd3cqJ+GHH+rksEv3Pj99oxJ3u3VIKA==", + "dev": true, + "dependencies": { + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/raw-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/read-cache": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz", + "integrity": "sha1-5mTvMRYRZsl1HNvo28+GtftY93Q=", + "dev": true, + "dependencies": { + "pify": "^2.3.0" + } + }, + "node_modules/read-cache/node_modules/pify": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", + "integrity": "sha1-7RQaasBDqEnqWISY59yosVMw6Qw=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/read-package-json-fast": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-1.2.2.tgz", + "integrity": "sha512-39DbPJjkltEzfXJXB6D8/Ir3GFOU2YbSKa2HaB/Y3nKrc/zY+0XrALpID6/13ezWyzqvOHrBbR4t4cjQuTdBVQ==", + "dev": true, + "dependencies": { + "json-parse-even-better-errors": "^2.3.0", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "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, + "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/readdirp": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz", + "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==", + "dependencies": { + "picomatch": "^2.2.1" + }, + "engines": { + "node": ">=8.10.0" + } + }, + "node_modules/reflect-metadata": { + "version": "0.1.13", + "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", + "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==" + }, + "node_modules/regenerate": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", + "dev": true + }, + "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, + "dependencies": { + "regenerate": "^1.4.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/regenerator-runtime": { + "version": "0.13.7", + "resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.13.7.tgz", + "integrity": "sha512-a54FxoJDIr27pgf7IgeQGxmqUNYrcV338lf/6gH456HZ/PhX+5BcwHXG9ajESmwe6WRO0tAzRUrRmNONWgkrew==", + "dev": true + }, + "node_modules/regenerator-transform": { + "version": "0.14.5", + "resolved": "https://registry.npmjs.org/regenerator-transform/-/regenerator-transform-0.14.5.tgz", + "integrity": "sha512-eOf6vka5IO151Jfsw2NO9WpGX58W6wWmefK3I1zEGr0lOD0u8rwPaNqQL1aRxUaxLeKO3ArNh3VYg1KbaD+FFw==", + "dev": true, + "dependencies": { + "@babel/runtime": "^7.8.4" + } + }, + "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, + "dependencies": { + "extend-shallow": "^3.0.2", + "safe-regex": "^1.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/regex-parser": { + "version": "2.2.11", + "resolved": "https://registry.npmjs.org/regex-parser/-/regex-parser-2.2.11.tgz", + "integrity": "sha512-jbD/FT0+9MBU2XAZluI7w2OBs1RBi6p9M83nkoZayQXXU9e8Robt69FcZc7wU4eJD/YFTjn1JdCk3rbMJajz8Q==", + "dev": true + }, + "node_modules/regexp.prototype.flags": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.3.1.tgz", + "integrity": "sha512-JiBdRBq91WlY7uRJ0ds7R+dU02i6LKi8r3BuQhNXn+kmeLN+EfHhfjqMRis1zJxnlu88hq/4dx0P2OP3APRTOA==", + "dev": true, + "dependencies": { + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" + }, + "engines": { + "node": ">= 0.4" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/regexpu-core": { + "version": "4.7.1", + "resolved": "https://registry.npmjs.org/regexpu-core/-/regexpu-core-4.7.1.tgz", + "integrity": "sha512-ywH2VUraA44DZQuRKzARmw6S66mr48pQVva4LBeRhcOltJ6hExvWly5ZjFLYo67xbIxb6W1q4bAGtgfEl20zfQ==", + "dev": true, + "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" + } + }, + "node_modules/regjsgen": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/regjsgen/-/regjsgen-0.5.2.tgz", + "integrity": "sha512-OFFT3MfrH90xIW8OOSyUrk6QHD5E9JOTeGodiJeBS3J6IwlgzJMNE/1bZklWz5oTg+9dCMyEetclvCVXOPoN3A==", + "dev": true + }, + "node_modules/regjsparser": { + "version": "0.6.9", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.9.tgz", + "integrity": "sha512-ZqbNRz1SNjLAiYuwY0zoXW8Ne675IX5q+YHioAGbCw4X96Mjl2+dcX9B2ciaeyYjViDAfvIjFpQjJgLttTEERQ==", + "dev": true, + "dependencies": { + "jsesc": "~0.5.0" + }, + "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" + } + }, + "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 + }, + "node_modules/repeat-element": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.4.tgz", + "integrity": "sha512-LFiNfRcSu7KK3evMyYOuCzv3L10TW7yC1G2/+StMjK8Y6Vqd2MG7r/Qjw4ghtuCOjFvlnms/iMmLqpvW/ES/WQ==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/request": { + "version": "2.88.2", + "resolved": "https://registry.npmjs.org/request/-/request-2.88.2.tgz", + "integrity": "sha512-MsvtOrfG9ZcrOwAW+Qi+F6HbD0CWXEh9ou77uOb7FM2WPhwT7smM833PzanhJLsgXjN89Ir6V2PczXNnMpwKhw==", + "dev": true, + "dependencies": { + "aws-sign2": "~0.7.0", + "aws4": "^1.8.0", + "caseless": "~0.12.0", + "combined-stream": "~1.0.6", + "extend": "~3.0.2", + "forever-agent": "~0.6.1", + "form-data": "~2.3.2", + "har-validator": "~5.1.3", + "http-signature": "~1.2.0", + "is-typedarray": "~1.0.0", + "isstream": "~0.1.2", + "json-stringify-safe": "~5.0.1", + "mime-types": "~2.1.19", + "oauth-sign": "~0.9.0", + "performance-now": "^2.1.0", + "qs": "~6.5.2", + "safe-buffer": "^5.1.2", + "tough-cookie": "~2.5.0", + "tunnel-agent": "^0.6.0", + "uuid": "^3.3.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/request/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, + "engines": { + "node": ">=0.6" + } + }, + "node_modules/request/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, + "bin": { + "uuid": "bin/uuid" + } + }, + "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=", + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/requires-port": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/requires-port/-/requires-port-1.0.0.tgz", + "integrity": "sha1-kl0mAdOaxIXgkc8NpcbmlNw9yv8=", + "dev": true + }, + "node_modules/resolve": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz", + "integrity": "sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg==", + "dependencies": { + "is-core-module": "^2.1.0", + "path-parse": "^1.0.6" + }, + "funding": { + "url": "https://github.com/sponsors/ljharb" + } + }, + "node_modules/resolve-cwd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/resolve-cwd/-/resolve-cwd-2.0.0.tgz", + "integrity": "sha1-AKn3OHVW4nA46uIyyqNypqWbZlo=", + "dev": true, + "dependencies": { + "resolve-from": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-cwd/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-from": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", + "dev": true, + "engines": { + "node": ">=4" + } + }, + "node_modules/resolve-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 + }, + "node_modules/resolve-url-loader": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-url-loader/-/resolve-url-loader-4.0.0.tgz", + "integrity": "sha512-05VEMczVREcbtT7Bz+C+96eUO5HDNvdthIiMB34t7FcF8ehcu4wC0sSgPUubs3XW2Q3CNLJk/BJrCU9wVRymiA==", + "dev": true, + "dependencies": { + "adjust-sourcemap-loader": "^4.0.0", + "convert-source-map": "^1.7.0", + "loader-utils": "^2.0.0", + "postcss": "^7.0.35", + "source-map": "0.6.1" + }, + "engines": { + "node": ">=8.9" + }, + "peerDependencies": { + "rework": "1.0.1", + "rework-visit": "1.0.0" + }, + "peerDependenciesMeta": { + "rework": { + "optional": true + }, + "rework-visit": { + "optional": true + } + } + }, + "node_modules/resolve-url-loader/node_modules/postcss": { + "version": "7.0.36", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.36.tgz", + "integrity": "sha512-BebJSIUMwJHRH0HAQoxN4u1CN86glsrwsW0q7T+/m44eXOUAxSNdHRkNZPYz5vVUbg17hFgOQDE7fZk7li3pZw==", + "dev": true, + "dependencies": { + "chalk": "^2.4.2", + "source-map": "^0.6.1", + "supports-color": "^6.1.0" + }, + "engines": { + "node": ">=6.0.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/postcss/" + } + }, + "node_modules/resolve-url-loader/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/resolve-url-loader/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/restore-cursor": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/restore-cursor/-/restore-cursor-3.1.0.tgz", + "integrity": "sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA==", + "dev": true, + "dependencies": { + "onetime": "^5.1.0", + "signal-exit": "^3.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "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, + "engines": { + "node": ">=0.12" + } + }, + "node_modules/retry": { + "version": "0.12.0", + "resolved": "https://registry.npmjs.org/retry/-/retry-0.12.0.tgz", + "integrity": "sha1-G0KmJmoh8HQh0bC1S33BZ7AcATs=", + "dev": true, + "engines": { + "node": ">= 4" + } + }, + "node_modules/reusify": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/reusify/-/reusify-1.0.4.tgz", + "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", + "dev": true, + "engines": { + "iojs": ">=1.0.0", + "node": ">=0.10.0" + } + }, + "node_modules/rfdc": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/rfdc/-/rfdc-1.3.0.tgz", + "integrity": "sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA==", + "dev": true + }, + "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 + }, + "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 + }, + "node_modules/rimraf": { + "version": "3.0.2", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-3.0.2.tgz", + "integrity": "sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/ripemd160": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/ripemd160/-/ripemd160-2.0.2.tgz", + "integrity": "sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==", + "dev": true, + "dependencies": { + "hash-base": "^3.0.0", + "inherits": "^2.0.1" + } + }, + "node_modules/rollup": { + "version": "2.38.4", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.38.4.tgz", + "integrity": "sha512-B0LcJhjiwKkTl79aGVF/u5KdzsH8IylVfV56Ut6c9ouWLJcUK17T83aZBetNYSnZtXf2OHD4+2PbmRW+Fp5ulg==", + "dev": true, + "bin": { + "rollup": "dist/bin/rollup" + }, + "engines": { + "node": ">=10.0.0" + }, + "optionalDependencies": { + "fsevents": "~2.3.1" + } + }, + "node_modules/rollup/node_modules/fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "engines": { + "node": "^8.16.0 || ^10.6.0 || >=11.0.0" + } + }, + "node_modules/run-async": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/run-async/-/run-async-2.4.1.tgz", + "integrity": "sha512-tvVnVv01b8c1RrA6Ep7JkStj85Guv/YrMcwqYQnwjsAS2cTmmPGBBjAjpCW7RrSodNSoE2/qg9O4bceNvUuDgQ==", + "dev": true, + "engines": { + "node": ">=0.12.0" + } + }, + "node_modules/run-parallel": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "dependencies": { + "queue-microtask": "^1.2.2" + } + }, + "node_modules/run-queue": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/run-queue/-/run-queue-1.0.3.tgz", + "integrity": "sha1-6Eg5bwV9Ij8kOGkkYY4laUFh7Ec=", + "dev": true, + "dependencies": { + "aproba": "^1.1.1" + } + }, + "node_modules/rxjs": { + "version": "6.6.3", + "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.3.tgz", + "integrity": "sha512-trsQc+xYYXZ3urjOiJOuCOa5N3jAZ3eiSpQB5hIT8zGlL2QfnHLJ2r7GMkBGuIausdJN1OneaI6gQlsqNHHmZQ==", + "dependencies": { + "tslib": "^1.9.0" + }, + "engines": { + "npm": ">=2.0.0" + } + }, + "node_modules/rxjs/node_modules/tslib": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", + "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==" + }, + "node_modules/safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "node_modules/safe-regex": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/safe-regex/-/safe-regex-1.1.0.tgz", + "integrity": "sha1-QKNmnzsHfR6UPURinhV91IAjvy4=", + "dev": true, + "dependencies": { + "ret": "~0.1.10" + } + }, + "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 + }, + "node_modules/sass": { + "version": "1.32.6", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.32.6.tgz", + "integrity": "sha512-1bcDHDcSqeFtMr0JXI3xc/CXX6c4p0wHHivJdru8W7waM7a1WjKMm4m/Z5sY7CbVw4Whi2Chpcw6DFfSWwGLzQ==", + "dev": true, + "dependencies": { + "chokidar": ">=2.0.0 <4.0.0" + }, + "bin": { + "sass": "sass.js" + }, + "engines": { + "node": ">=8.9.0" + } + }, + "node_modules/sass-loader": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-10.1.1.tgz", + "integrity": "sha512-W6gVDXAd5hR/WHsPicvZdjAWHBcEJ44UahgxcIE196fW2ong0ZHMPO1kZuI5q0VlvMQZh32gpv69PLWQm70qrw==", + "dev": true, + "dependencies": { + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "neo-async": "^2.6.2", + "schema-utils": "^3.0.0", + "semver": "^7.3.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "fibers": ">= 3.1.0", + "node-sass": "^4.0.0 || ^5.0.0", + "sass": "^1.3.0", + "webpack": "^4.36.0 || ^5.0.0" + }, + "peerDependenciesMeta": { + "fibers": { + "optional": true + }, + "node-sass": { + "optional": true + }, + "sass": { + "optional": true + } + } + }, + "node_modules/sass-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/saucelabs": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/saucelabs/-/saucelabs-1.5.0.tgz", + "integrity": "sha512-jlX3FGdWvYf4Q3LFfFWS1QvPg3IGCGWxIc8QBFdPTbpTJnt/v17FHXYVAn7C8sHf1yUXo2c7yIM0isDryfYtHQ==", + "dev": true, + "dependencies": { + "https-proxy-agent": "^2.2.1" + }, + "engines": { + "node": "*" + } + }, + "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 + }, + "node_modules/schema-utils": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-2.7.1.tgz", + "integrity": "sha512-SHiNtMOUGWBQJwzISiVYKu82GiV4QYGePp3odlY1tuKO7gPtphAT5R/py0fA6xtbgLL/RvtJZnU9b8s0F1q0Xg==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.5", + "ajv": "^6.12.4", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 8.9.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/select-hose": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/select-hose/-/select-hose-2.0.0.tgz", + "integrity": "sha1-Yl2GWPhlr0Psliv8N2o3NZpJlMo=", + "dev": true + }, + "node_modules/selenium-webdriver": { + "version": "3.6.0", + "resolved": "https://registry.npmjs.org/selenium-webdriver/-/selenium-webdriver-3.6.0.tgz", + "integrity": "sha512-WH7Aldse+2P5bbFBO4Gle/nuQOdVwpHMTL6raL3uuBj/vPG07k6uzt3aiahu352ONBr5xXh0hDlM3LhtXPOC4Q==", + "dev": true, + "dependencies": { + "jszip": "^3.1.3", + "rimraf": "^2.5.4", + "tmp": "0.0.30", + "xml2js": "^0.4.17" + }, + "engines": { + "node": ">= 6.9.0" + } + }, + "node_modules/selenium-webdriver/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/selenium-webdriver/node_modules/tmp": { + "version": "0.0.30", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.30.tgz", + "integrity": "sha1-ckGdSovn1s51FI/YsyTlk6cRwu0=", + "dev": true, + "dependencies": { + "os-tmpdir": "~1.0.1" + }, + "engines": { + "node": ">=0.4.0" + } + }, + "node_modules/selfsigned": { + "version": "1.10.11", + "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-1.10.11.tgz", + "integrity": "sha512-aVmbPOfViZqOZPgRBT0+3u4yZFHpmnIghLMlAcb5/xhp5ZtB/RVnKhz5vl2M32CLXAqR4kha9zfhNg0Lf/sxKA==", + "dev": true, + "dependencies": { + "node-forge": "^0.10.0" + } + }, + "node_modules/semver": { + "version": "7.3.2", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.2.tgz", + "integrity": "sha512-OrOb32TeeambH6UrhtShmF7CRDqhL6/5XpPNp2DuRH6+9QLw/orhp72j87v8Qa1ScDkvrrBNpZcDejAirJmfXQ==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/semver-dsl": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/semver-dsl/-/semver-dsl-1.0.1.tgz", + "integrity": "sha1-02eN5VVeimH2Ke7QJTZq5fJzQKA=", + "dev": true, + "dependencies": { + "semver": "^5.3.0" + } + }, + "node_modules/semver-dsl/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/semver-intersect": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/semver-intersect/-/semver-intersect-1.4.0.tgz", + "integrity": "sha512-d8fvGg5ycKAq0+I6nfWeCx6ffaWJCsBYU0H2Rq56+/zFePYfT8mXkB3tWBSjR5BerkHNZ5eTPIk1/LBYas35xQ==", + "dev": true, + "dependencies": { + "semver": "^5.0.0" + } + }, + "node_modules/semver-intersect/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "dev": true, + "dependencies": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/send/node_modules/debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/send/node_modules/debug/node_modules/ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", + "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-javascript": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz", + "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==", + "dev": true, + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/serve-index": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/serve-index/-/serve-index-1.9.1.tgz", + "integrity": "sha1-03aNabHn2C5c4FD/9bRTvqEqkjk=", + "dev": true, + "dependencies": { + "accepts": "~1.3.4", + "batch": "0.6.1", + "debug": "2.6.9", + "escape-html": "~1.0.3", + "http-errors": "~1.6.2", + "mime-types": "~2.1.17", + "parseurl": "~1.3.2" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/serve-index/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/serve-index/node_modules/http-errors": { + "version": "1.6.3", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz", + "integrity": "sha1-i1VoC7S+KDoLW/TqLjhYC+HZMg0=", + "dev": true, + "dependencies": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.0", + "statuses": ">= 1.4.0 < 2" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/serve-index/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/serve-index/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/serve-index/node_modules/setprototypeof": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz", + "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ==", + "dev": true + }, + "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, + "dependencies": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + }, + "engines": { + "node": ">= 0.8.0" + } + }, + "node_modules/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 + }, + "node_modules/set-immediate-shim": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/set-immediate-shim/-/set-immediate-shim-1.0.1.tgz", + "integrity": "sha1-SysbJ+uAip+NzEgaWOXlb1mfP2E=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/set-value/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/setimmediate": { + "version": "1.0.5", + "resolved": "https://registry.npmjs.org/setimmediate/-/setimmediate-1.0.5.tgz", + "integrity": "sha1-KQy7Iy4waULX1+qbg3Mqt4VvgoU=", + "dev": true + }, + "node_modules/setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==", + "dev": true + }, + "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, + "dependencies": { + "inherits": "^2.0.1", + "safe-buffer": "^5.0.1" + }, + "bin": { + "sha.js": "bin.js" + } + }, + "node_modules/shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "dependencies": { + "kind-of": "^6.0.2" + }, + "engines": { + "node": ">=8" + } + }, + "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, + "dependencies": { + "shebang-regex": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/slash": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", + "integrity": "sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/smart-buffer": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/smart-buffer/-/smart-buffer-4.1.0.tgz", + "integrity": "sha512-iVICrxOzCynf/SNaBQCw34eM9jROU/s5rzIhpOvzhzuYHfJR/DhZfDkXiZSgKXfgv26HT3Yni3AV/DGw0cGnnw==", + "dev": true, + "engines": { + "node": ">= 6.0.0", + "npm": ">= 3.0.0" + } + }, + "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, + "dependencies": { + "base": "^0.11.1", + "debug": "^2.2.0", + "define-property": "^0.2.5", + "extend-shallow": "^2.0.1", + "map-cache": "^0.2.2", + "source-map": "^0.5.6", + "source-map-resolve": "^0.5.0", + "use": "^3.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "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-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, + "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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/snapdragon/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" + } + }, + "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/snapdragon/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==", + "dev": true, + "dependencies": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + }, + "node_modules/socket.io": { + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-3.1.2.tgz", + "integrity": "sha512-JubKZnTQ4Z8G4IZWtaAZSiRP3I/inpy8c/Bsx2jrwGrTbKeVU5xd6qkKMHpChYeM3dWZSO0QACiGK+obhBNwYw==", + "dev": true, + "dependencies": { + "@types/cookie": "^0.4.0", + "@types/cors": "^2.8.8", + "@types/node": ">=10.0.0", + "accepts": "~1.3.4", + "base64id": "~2.0.0", + "debug": "~4.3.1", + "engine.io": "~4.1.0", + "socket.io-adapter": "~2.1.0", + "socket.io-parser": "~4.0.3" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/socket.io-adapter": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.1.0.tgz", + "integrity": "sha512-+vDov/aTsLjViYTwS9fPy5pEtTkrbEKsw2M+oVSoFGw6OD1IpvlV1VPhUzNbofCQ8oyMbdYJqDtGdmHQK6TdPg==", + "dev": true + }, + "node_modules/socket.io-parser": { + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz", + "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==", + "dev": true, + "dependencies": { + "@types/component-emitter": "^1.2.10", + "component-emitter": "~1.3.0", + "debug": "~4.3.1" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/sockjs": { + "version": "0.3.21", + "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.21.tgz", + "integrity": "sha512-DhbPFGpxjc6Z3I+uX07Id5ZO2XwYsWOrYjaSeieES78cq+JaJvVe5q/m1uvjIQhXinhIeCFRH6JgXe+mvVMyXw==", + "dev": true, + "dependencies": { + "faye-websocket": "^0.11.3", + "uuid": "^3.4.0", + "websocket-driver": "^0.7.4" + } + }, + "node_modules/sockjs-client": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/sockjs-client/-/sockjs-client-1.5.1.tgz", + "integrity": "sha512-VnVAb663fosipI/m6pqRXakEOw7nvd7TUgdr3PlR/8V2I95QIdwT8L4nMxhyU8SmDBHYXU1TOElaKOmKLfYzeQ==", + "dev": true, + "dependencies": { + "debug": "^3.2.6", + "eventsource": "^1.0.7", + "faye-websocket": "^0.11.3", + "inherits": "^2.0.4", + "json3": "^3.3.3", + "url-parse": "^1.5.1" + } + }, + "node_modules/sockjs-client/node_modules/debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "dependencies": { + "ms": "^2.1.1" + } + }, + "node_modules/sockjs/node_modules/uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "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" + } + }, + "node_modules/socks": { + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/socks/-/socks-2.6.1.tgz", + "integrity": "sha512-kLQ9N5ucj8uIcxrDwjm0Jsqk06xdpBjGNQtpXy4Q8/QY2k+fY7nZH8CARy+hkbG+SGAovmzzuauCpBlb8FrnBA==", + "dev": true, + "dependencies": { + "ip": "^1.1.5", + "smart-buffer": "^4.1.0" + }, + "engines": { + "node": ">= 10.13.0", + "npm": ">= 3.0.0" + } + }, + "node_modules/socks-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-5.0.0.tgz", + "integrity": "sha512-lEpa1zsWCChxiynk+lCycKuC502RxDWLKJZoIhnxrWNjLSDGYRFflHA1/228VkRcnv9TIb8w98derGbpKxJRgA==", + "dev": true, + "dependencies": { + "agent-base": "6", + "debug": "4", + "socks": "^2.3.3" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/socks-proxy-agent/node_modules/agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "dependencies": { + "debug": "4" + }, + "engines": { + "node": ">= 6.0.0" + } + }, + "node_modules/source-list-map": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", + "integrity": "sha512-qnQ7gVMxGNxsiL4lEuJwe/To8UnK7fAnmbGEEH8RpLouuKbeEm0lhbQVFIrNSuB+G7tVrAlVsZgETT5nljf+Iw==", + "dev": true + }, + "node_modules/source-map": { + "version": "0.7.3", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.7.3.tgz", + "integrity": "sha512-CkCj6giN3S+n9qrYiBTX5gystlENnRW5jZeNLHpe6aue+SrHcG5VYwujhW9s4dY31mEGsxBDrHR6oI69fTXsaQ==", + "dev": true, + "engines": { + "node": ">= 8" + } + }, + "node_modules/source-map-loader": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/source-map-loader/-/source-map-loader-1.1.3.tgz", + "integrity": "sha512-6YHeF+XzDOrT/ycFJNI53cgEsp/tHTMl37hi7uVyqFAlTXW109JazaQCkbc+jjoL2637qkH1amLi+JzrIpt5lA==", + "dev": true, + "dependencies": { + "abab": "^2.0.5", + "iconv-lite": "^0.6.2", + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0", + "source-map": "^0.6.1", + "whatwg-mimetype": "^2.3.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/source-map-loader/node_modules/iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "dependencies": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/source-map-loader/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-resolve": { + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.6.0.tgz", + "integrity": "sha512-KXBr9d/fO/bWo97NXsPIAW1bFSBOuCnjbNTBMO7N59hsv5i9yzRDfcYwwt0l04+VqnKC+EwzvJZIP/qkuMgR/w==", + "dev": true, + "dependencies": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0" + } + }, + "node_modules/source-map-support": { + "version": "0.5.19", + "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz", + "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==", + "dev": true, + "dependencies": { + "buffer-from": "^1.0.0", + "source-map": "^0.6.0" + } + }, + "node_modules/source-map-support/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/source-map-url": { + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.1.tgz", + "integrity": "sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==", + "dev": true + }, + "node_modules/sourcemap-codec": { + "version": "1.4.8", + "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", + "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==" + }, + "node_modules/spdy": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/spdy/-/spdy-4.0.2.tgz", + "integrity": "sha512-r46gZQZQV+Kl9oItvl1JZZqJKGr+oEkB08A6BzkiR7593/7IbtuncXHd2YoYeTsG4157ZssMu9KYvUHLcjcDoA==", + "dev": true, + "dependencies": { + "debug": "^4.1.0", + "handle-thing": "^2.0.0", + "http-deceiver": "^1.2.7", + "select-hose": "^2.0.0", + "spdy-transport": "^3.0.0" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/spdy-transport": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/spdy-transport/-/spdy-transport-3.0.0.tgz", + "integrity": "sha512-hsLVFE5SjA6TCisWeJXFKniGGOpBgMLmerfO2aCyCU5s7nJ/rpAepqmFifv/GCbSbueEeAJJnmSQ2rKC/g8Fcw==", + "dev": true, + "dependencies": { + "debug": "^4.1.0", + "detect-node": "^2.0.4", + "hpack.js": "^2.1.6", + "obuf": "^1.1.2", + "readable-stream": "^3.0.6", + "wbuf": "^1.7.3" + } + }, + "node_modules/spdy-transport/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": { + "inherits": "^2.0.3", + "string_decoder": "^1.1.1", + "util-deprecate": "^1.0.1" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/speed-measure-webpack-plugin": { + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.4.2.tgz", + "integrity": "sha512-AtVzD0bnIy2/B0fWqJpJgmhcrfWFhBlduzSo0uwplr/QvB33ZNZj2NEth3NONgdnZJqicK0W0mSxnLSbsVCDbw==", + "dev": true, + "dependencies": { + "chalk": "^4.1.0" + }, + "engines": { + "node": ">=6.0.0" + }, + "peerDependencies": { + "webpack": "^1 || ^2 || ^3 || ^4 || ^5" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/speed-measure-webpack-plugin/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/speed-measure-webpack-plugin/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/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, + "dependencies": { + "extend-shallow": "^3.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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 + }, + "node_modules/sshpk": { + "version": "1.16.1", + "resolved": "https://registry.npmjs.org/sshpk/-/sshpk-1.16.1.tgz", + "integrity": "sha512-HXXqVUq7+pcKeLqqZj6mHFUMvXtOJt1uoUx09pFW6011inTMxqI8BA8PM95myrIyyKwdnzjdFjLiE6KBPVtJIg==", + "dev": true, + "dependencies": { + "asn1": "~0.2.3", + "assert-plus": "^1.0.0", + "bcrypt-pbkdf": "^1.0.0", + "dashdash": "^1.12.0", + "ecc-jsbn": "~0.1.1", + "getpass": "^0.1.1", + "jsbn": "~0.1.0", + "safer-buffer": "^2.0.2", + "tweetnacl": "~0.14.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/ssri": { + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", + "dev": true, + "dependencies": { + "minipass": "^3.1.1" + }, + "engines": { + "node": ">= 8" + } + }, + "node_modules/stable": { + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/stable/-/stable-0.1.8.tgz", + "integrity": "sha512-ji9qxRnOVfcuLDySj9qzhGSEFVobyt1kIOSkj1qZzYLzq7Tos/oUUWvotUPQLlrsidqsK6tBH89Bc9kL5zHA6w==", + "dev": true + }, + "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, + "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": { + "is-descriptor": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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=", + "dev": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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, + "dependencies": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/static-extend/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" + } + }, + "node_modules/statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=", + "dev": true, + "engines": { + "node": ">= 0.6" + } + }, + "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, + "dependencies": { + "inherits": "~2.0.1", + "readable-stream": "^2.0.2" + } + }, + "node_modules/stream-each": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/stream-each/-/stream-each-1.2.3.tgz", + "integrity": "sha512-vlMC2f8I2u/bZGqkdfLQW/13Zihpej/7PmSiMQsbYddxuTsJp8vRe2x2FvVExZg7FaOds43ROAuFJwPR4MTZLw==", + "dev": true, + "dependencies": { + "end-of-stream": "^1.1.0", + "stream-shift": "^1.0.0" + } + }, + "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, + "dependencies": { + "builtin-status-codes": "^3.0.0", + "inherits": "^2.0.1", + "readable-stream": "^2.3.6", + "to-arraybuffer": "^1.0.0", + "xtend": "^4.0.0" + } + }, + "node_modules/stream-shift": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/stream-shift/-/stream-shift-1.0.1.tgz", + "integrity": "sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ==", + "dev": true + }, + "node_modules/streamroller": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/streamroller/-/streamroller-2.2.4.tgz", + "integrity": "sha512-OG79qm3AujAM9ImoqgWEY1xG4HX+Lw+yY6qZj9R1K2mhF5bEmQ849wvrb+4vt4jLMLzwXttJlQbOdPOQVRv7DQ==", + "dev": true, + "dependencies": { + "date-format": "^2.1.0", + "debug": "^4.1.1", + "fs-extra": "^8.1.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/streamroller/node_modules/date-format": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/date-format/-/date-format-2.1.0.tgz", + "integrity": "sha512-bYQuGLeFxhkxNOF3rcMtiZxvCBAquGzZm6oWA1oZ0g2THUzivaRhv8uOhdr19LmoobSOLoIAxeUK2RdbM8IFTA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/streamroller/node_modules/fs-extra": { + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-8.1.0.tgz", + "integrity": "sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.2.0", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + }, + "engines": { + "node": ">=6 <7 || >=8" + } + }, + "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, + "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": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "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/strip-eof": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/strip-eof/-/strip-eof-1.0.0.tgz", + "integrity": "sha1-u0P/VZim6wXYm1n80SnJgzE2Br8=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/style-loader": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-2.0.0.tgz", + "integrity": "sha512-Z0gYUJmzZ6ZdRUqpg1r8GsaFKypE+3xAzuFeMuoHgjc9KZv3wMyCRjQIWEbhoFSq7+7yoHXySDJyyWQaPajeiQ==", + "dev": true, + "dependencies": { + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/style-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/stylehacks": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-5.0.1.tgz", + "integrity": "sha512-Es0rVnHIqbWzveU1b24kbw92HsebBepxfcqe5iix7t9j0PQqhs0IxXVXv0pY2Bxa08CgMkzD6OWql7kbGOuEdA==", + "dev": true, + "dependencies": { + "browserslist": "^4.16.0", + "postcss-selector-parser": "^6.0.4" + }, + "engines": { + "node": "^10 || ^12 || >=14.0" + }, + "peerDependencies": { + "postcss": "^8.2.15" + } + }, + "node_modules/stylus": { + "version": "0.54.8", + "resolved": "https://registry.npmjs.org/stylus/-/stylus-0.54.8.tgz", + "integrity": "sha512-vr54Or4BZ7pJafo2mpf0ZcwA74rpuYCZbxrHBsH8kbcXOwSfvBFwsRfpGO5OD5fhG5HDCFW737PKaawI7OqEAg==", + "dev": true, + "dependencies": { + "css-parse": "~2.0.0", + "debug": "~3.1.0", + "glob": "^7.1.6", + "mkdirp": "~1.0.4", + "safer-buffer": "^2.1.2", + "sax": "~1.2.4", + "semver": "^6.3.0", + "source-map": "^0.7.3" + }, + "bin": { + "stylus": "bin/stylus" + }, + "engines": { + "node": "*" + } + }, + "node_modules/stylus-loader": { + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/stylus-loader/-/stylus-loader-4.3.3.tgz", + "integrity": "sha512-PpWB5PnCXUzW4WMYhCvNzAHJBjIBPMXwsdfkkKuA9W7k8OQFMl/19/AQvaWsxz2IptxUlCseyJ6TY/eEKJ4+UQ==", + "dev": true, + "dependencies": { + "fast-glob": "^3.2.4", + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "normalize-path": "^3.0.0", + "schema-utils": "^3.0.0" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "stylus": ">=0.52.4", + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/stylus-loader/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/stylus/node_modules/debug": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", + "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "dev": true, + "dependencies": { + "ms": "2.0.0" + } + }, + "node_modules/stylus/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/stylus/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/stylus/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "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==", + "dependencies": { + "has-flag": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/svgo": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-2.3.0.tgz", + "integrity": "sha512-fz4IKjNO6HDPgIQxu4IxwtubtbSfGEAJUq/IXyTPIkGhWck/faiiwfkvsB8LnBkKLvSoyNNIY6d13lZprJMc9Q==", + "dev": true, + "dependencies": { + "@trysound/sax": "0.1.1", + "chalk": "^4.1.0", + "commander": "^7.1.0", + "css-select": "^3.1.2", + "css-tree": "^1.1.2", + "csso": "^4.2.0", + "stable": "^0.1.8" + }, + "bin": { + "svgo": "bin/svgo" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/svgo/node_modules/ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "dependencies": { + "color-convert": "^2.0.1" + }, + "engines": { + "node": ">=8" + }, + "funding": { + "url": "https://github.com/chalk/ansi-styles?sponsor=1" + } + }, + "node_modules/svgo/node_modules/chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "dependencies": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/chalk/chalk?sponsor=1" + } + }, + "node_modules/svgo/node_modules/color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "dependencies": { + "color-name": "~1.1.4" + }, + "engines": { + "node": ">=7.0.0" + } + }, + "node_modules/svgo/node_modules/color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "node_modules/svgo/node_modules/commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "dev": true, + "engines": { + "node": ">= 10" + } + }, + "node_modules/svgo/node_modules/has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/svgo/node_modules/supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "dependencies": { + "has-flag": "^4.0.0" + }, + "engines": { + "node": ">=8" + } + }, + "node_modules/symbol-observable": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-3.0.0.tgz", + "integrity": "sha512-6tDOXSHiVjuCaasQSWTmHUWn4PuG7qa3+1WT031yTc/swT7+rLiw3GOrFxaH1E3lLP09dH3bVuVDf2gK5rxG3Q==", + "dev": true, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/tapable": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.0.tgz", + "integrity": "sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/tar": { + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.0.tgz", + "integrity": "sha512-DUCttfhsnLCjwoDoFcI+B2iJgYa93vBnDUATYEeRx6sntCTdN01VnqsIuTlALXla/LWooNg0yEGeB+Y8WdFxGA==", + "dev": true, + "dependencies": { + "chownr": "^2.0.0", + "fs-minipass": "^2.0.0", + "minipass": "^3.0.0", + "minizlib": "^2.1.1", + "mkdirp": "^1.0.3", + "yallist": "^4.0.0" + }, + "engines": { + "node": ">= 10" + } + }, + "node_modules/tar/node_modules/mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true, + "bin": { + "mkdirp": "bin/cmd.js" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser": { + "version": "5.5.1", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.5.1.tgz", + "integrity": "sha512-6VGWZNVP2KTUcltUQJ25TtNjx/XgdDsBDKGt8nN0MpydU36LmbPPcMBd2kmtZNNGVVDLg44k7GKeHHj+4zPIBQ==", + "dev": true, + "dependencies": { + "commander": "^2.20.0", + "source-map": "~0.7.2", + "source-map-support": "~0.5.19" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=10" + } + }, + "node_modules/terser-webpack-plugin": { + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-4.2.3.tgz", + "integrity": "sha512-jTgXh40RnvOrLQNgIkwEKnQ8rmHjHK4u+6UBEi+W+FPmvb+uo+chJXntKe7/3lW5mNysgSWD60KyesnhW8D6MQ==", + "dev": true, + "dependencies": { + "cacache": "^15.0.5", + "find-cache-dir": "^3.3.1", + "jest-worker": "^26.5.0", + "p-limit": "^3.0.2", + "schema-utils": "^3.0.0", + "serialize-javascript": "^5.0.1", + "source-map": "^0.6.1", + "terser": "^5.3.4", + "webpack-sources": "^1.4.3" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + } + }, + "node_modules/terser-webpack-plugin/node_modules/p-limit": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", + "dev": true, + "dependencies": { + "yocto-queue": "^0.1.0" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/terser-webpack-plugin/node_modules/schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "dependencies": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + }, + "engines": { + "node": ">= 10.13.0" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + } + }, + "node_modules/terser-webpack-plugin/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/terser-webpack-plugin/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", + "dev": true + }, + "node_modules/through": { + "version": "2.3.8", + "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", + "integrity": "sha1-DdTJ/6q8NXlgsbckEV1+Doai4fU=", + "dev": true + }, + "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, + "dependencies": { + "readable-stream": "~2.3.6", + "xtend": "~4.0.1" + } + }, + "node_modules/thunky": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/thunky/-/thunky-1.1.0.tgz", + "integrity": "sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA==", + "dev": true + }, + "node_modules/timers-browserify": { + "version": "2.0.12", + "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.12.tgz", + "integrity": "sha512-9phl76Cqm6FhSX9Xe1ZUAMLtm1BLkKj2Qd5ApyWkXzsMRaA7dgr81kf4wJmQf/hAvg8EEyJxDo3du/0KlhPiKQ==", + "dev": true, + "dependencies": { + "setimmediate": "^1.0.4" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "node_modules/timsort": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/timsort/-/timsort-0.3.0.tgz", + "integrity": "sha1-QFQRqOfmM5/mTbmiNN4R3DHgK9Q=", + "dev": true + }, + "node_modules/tmp": { + "version": "0.0.33", + "resolved": "https://registry.npmjs.org/tmp/-/tmp-0.0.33.tgz", + "integrity": "sha512-jRCJlojKnZ3addtTOjdIqoRuPEKBvNXcGYqzO6zWZX8KfKEpnGY5jfggJQ3EjKuu8D4bJRr0y+cYJFmYbImXGw==", + "dev": true, + "dependencies": { + "os-tmpdir": "~1.0.2" + }, + "engines": { + "node": ">=0.6.0" + } + }, + "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 + }, + "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=", + "engines": { + "node": ">=4" + } + }, + "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, + "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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "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" + } + }, + "node_modules/to-regex-range": { + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", + "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", + "dependencies": { + "is-number": "^7.0.0" + }, + "engines": { + "node": ">=8.0" + } + }, + "node_modules/toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==", + "dev": true, + "engines": { + "node": ">=0.6" + } + }, + "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, + "dependencies": { + "psl": "^1.1.28", + "punycode": "^2.1.1" + }, + "engines": { + "node": ">=0.8" + } + }, + "node_modules/tree-kill": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", + "integrity": "sha512-L0Orpi8qGpRG//Nd+H90vFB+3iHnue1zSSGmNOOCh1GLJ7rUKVwV2HvijphGQS2UmhUZewS9VgvxYIdgr+fG1A==", + "dev": true, + "bin": { + "tree-kill": "cli.js" + } + }, + "node_modules/ts-node": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/ts-node/-/ts-node-8.3.0.tgz", + "integrity": "sha512-dyNS/RqyVTDcmNM4NIBAeDMpsAdaQ+ojdf0GOLqE6nwJOgzEkdRNzJywhDfwnuvB10oa6NLVG1rUJQCpRN7qoQ==", + "dev": true, + "dependencies": { + "arg": "^4.1.0", + "diff": "^4.0.1", + "make-error": "^1.1.1", + "source-map-support": "^0.5.6", + "yn": "^3.0.0" + }, + "bin": { + "ts-node": "dist/bin.js" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/ts-pnp": { + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/ts-pnp/-/ts-pnp-1.2.0.tgz", + "integrity": "sha512-csd+vJOb/gkzvcCHgTGSChYpy5f1/XKNsmvBGO4JXS+z1v2HobugDz4s1IeFXM3wZB44uczs+eazB5Q/ccdhQw==", + "dev": true, + "engines": { + "node": ">=6" + }, + "peerDependenciesMeta": { + "typescript": { + "optional": true + } + } + }, + "node_modules/tslib": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.0.2.tgz", + "integrity": "sha512-wAH28hcEKwna96/UacuWaVspVLkg4x1aDM9JlzqaQTOFczCktkVAb5fmXChgandR1EraDPs2w8P+ozM+oafwxg==" + }, + "node_modules/tslint": { + "version": "6.1.3", + "resolved": "https://registry.npmjs.org/tslint/-/tslint-6.1.3.tgz", + "integrity": "sha512-IbR4nkT96EQOvKE2PW/djGz8iGNeJ4rF2mBfiYaR/nvUWYKJhLwimoJKgjIFEIDibBtOevj7BqCRL4oHeWWUCg==", + "dev": true, + "dependencies": { + "@babel/code-frame": "^7.0.0", + "builtin-modules": "^1.1.1", + "chalk": "^2.3.0", + "commander": "^2.12.1", + "diff": "^4.0.1", + "glob": "^7.1.1", + "js-yaml": "^3.13.1", + "minimatch": "^3.0.4", + "mkdirp": "^0.5.3", + "resolve": "^1.3.2", + "semver": "^5.3.0", + "tslib": "^1.13.0", + "tsutils": "^2.29.0" + }, + "bin": { + "tslint": "bin/tslint" + }, + "engines": { + "node": ">=4.8.0" + } + }, + "node_modules/tslint/node_modules/semver": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "dev": true, + "bin": { + "semver": "bin/semver" + } + }, + "node_modules/tslint/node_modules/tslib": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", + "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", + "dev": true + }, + "node_modules/tsutils": { + "version": "2.29.0", + "resolved": "https://registry.npmjs.org/tsutils/-/tsutils-2.29.0.tgz", + "integrity": "sha512-g5JVHCIJwzfISaXpXE1qvNalca5Jwob6FjI4AoPlqMusJ6ftFE7IkkFoMhVLRgK+4Kx3gkzb8UZK5t5yTTvEmA==", + "dev": true, + "dependencies": { + "tslib": "^1.8.1" + } + }, + "node_modules/tsutils/node_modules/tslib": { + "version": "1.14.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", + "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", + "dev": true + }, + "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 + }, + "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, + "dependencies": { + "safe-buffer": "^5.0.1" + }, + "engines": { + "node": "*" + } + }, + "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 + }, + "node_modules/type-fest": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", + "integrity": "sha512-OdjXJxnCN1AvyLSzeKIgXTXxV+99ZuXl3Hpo9XpJAv9MBcHrrJOQ5kV7ypXOuQie+AmWG25hLbiKdwYTifzcfQ==", + "dev": true, + "engines": { + "node": ">=8" + } + }, + "node_modules/type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "dev": true, + "dependencies": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + }, + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/typedarray": { + "version": "0.0.6", + "resolved": "https://registry.npmjs.org/typedarray/-/typedarray-0.0.6.tgz", + "integrity": "sha1-hnrHTjhkGHsdPUfZlqeOxciDB3c=", + "dev": true + }, + "node_modules/typescript": { + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.1.5.tgz", + "integrity": "sha512-6OSu9PTIzmn9TCDiovULTnET6BgXtDYL4Gg4szY+cGsc3JP1dQL8qvE8kShTRx1NIw4Q9IBHlwODjkjWEtMUyA==", + "bin": { + "tsc": "bin/tsc", + "tsserver": "bin/tsserver" + }, + "engines": { + "node": ">=4.2.0" + } + }, + "node_modules/ua-parser-js": { + "version": "0.7.28", + "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.28.tgz", + "integrity": "sha512-6Gurc1n//gjp9eQNXjD9O3M/sMwVtN5S8Lv9bvOYBfKfDNiIIhqiyi01vMBO45u4zkDE420w/e0se7Vs+sIg+g==", + "dev": true, + "funding": [ + { + "type": "opencollective", + "url": "https://opencollective.com/ua-parser-js" + }, + { + "type": "paypal", + "url": "https://paypal.me/faisalman" + } + ], + "engines": { + "node": "*" + } + }, + "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, + "engines": { + "node": ">=4" + } + }, + "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, + "dependencies": { + "unicode-canonical-property-names-ecmascript": "^1.0.4", + "unicode-property-aliases-ecmascript": "^1.0.4" + }, + "engines": { + "node": ">=4" + } + }, + "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, + "engines": { + "node": ">=4" + } + }, + "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, + "engines": { + "node": ">=4" + } + }, + "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, + "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" + } + }, + "node_modules/union-value/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/uniqs": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/uniqs/-/uniqs-2.0.0.tgz", + "integrity": "sha1-/+3ks2slKQaW5uFl1KWe25mOawI=", + "dev": true + }, + "node_modules/unique-filename": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/unique-filename/-/unique-filename-1.1.1.tgz", + "integrity": "sha512-Vmp0jIp2ln35UTXuryvjzkjGdRyf9b2lTXuSYUiPmzRcl3FDtYqAwOnTJkAngD9SWhnoJzDbTKwaOrZ+STtxNQ==", + "dev": true, + "dependencies": { + "unique-slug": "^2.0.0" + } + }, + "node_modules/unique-slug": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/unique-slug/-/unique-slug-2.0.2.tgz", + "integrity": "sha512-zoWr9ObaxALD3DOPfjPSqxt4fnZiWblxHIgeWqW8x7UqDzEtHEQLzji2cuJYQFCU6KmoJikOYAZlrTHHebjx2w==", + "dev": true, + "dependencies": { + "imurmurhash": "^0.1.4" + } + }, + "node_modules/universal-analytics": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/universal-analytics/-/universal-analytics-0.4.23.tgz", + "integrity": "sha512-lgMIH7XBI6OgYn1woDEmxhGdj8yDefMKg7GkWdeATAlQZFrMrNyxSkpDzY57iY0/6fdlzTbBV03OawvvzG+q7A==", + "dev": true, + "dependencies": { + "debug": "^4.1.1", + "request": "^2.88.2", + "uuid": "^3.0.0" + } + }, + "node_modules/universal-analytics/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, + "bin": { + "uuid": "bin/uuid" + } + }, + "node_modules/universalify": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", + "engines": { + "node": ">= 4.0.0" + } + }, + "node_modules/unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "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, + "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": { + "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" + } + }, + "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, + "engines": { + "node": ">=4", + "yarn": "*" + } + }, + "node_modules/uri-js": { + "version": "4.4.0", + "resolved": "https://registry.npmjs.org/uri-js/-/uri-js-4.4.0.tgz", + "integrity": "sha512-B0yRTzYdUCCn9n+F4+Gh4yIDtMQcaJsmYBDsTSG8g/OejKBodLQ2IHfN3bM7jUsRXndopT7OIXWdYqc1fjmV6g==", + "dev": true, + "dependencies": { + "punycode": "^2.1.0" + } + }, + "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 + }, + "node_modules/url": { + "version": "0.11.0", + "resolved": "https://registry.npmjs.org/url/-/url-0.11.0.tgz", + "integrity": "sha1-ODjpfPxgUh63PFJajlW/3Z4uKPE=", + "dev": true, + "dependencies": { + "punycode": "1.3.2", + "querystring": "0.2.0" + } + }, + "node_modules/url-parse": { + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.1.tgz", + "integrity": "sha512-HOfCOUJt7iSYzEx/UqgtwKRMC6EU91NFhsCHMv9oM03VJcVo2Qrp8T8kI9D7amFf1cu+/3CEhgb3rF9zL7k85Q==", + "dev": true, + "dependencies": { + "querystringify": "^2.1.1", + "requires-port": "^1.0.0" + } + }, + "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, + "engines": { + "node": ">=0.10.0" + } + }, + "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, + "dependencies": { + "inherits": "2.0.3" + } + }, + "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 + }, + "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/utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=", + "dev": true, + "engines": { + "node": ">= 0.4.0" + } + }, + "node_modules/uuid": { + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==", + "bin": { + "uuid": "dist/bin/uuid" + } + }, + "node_modules/validate-npm-package-name": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/validate-npm-package-name/-/validate-npm-package-name-3.0.0.tgz", + "integrity": "sha1-X6kS2B630MdK/BQN5zF/DKffQ34=", + "dev": true, + "dependencies": { + "builtins": "^1.0.3" + } + }, + "node_modules/vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=", + "dev": true, + "engines": { + "node": ">= 0.8" + } + }, + "node_modules/vendors": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/vendors/-/vendors-1.0.4.tgz", + "integrity": "sha512-/juG65kTL4Cy2su4P8HjtkTxk6VmJDiOPBufWniqQ6wknac6jNiXS9vU+hO3wgusiyqWlzTbVHi0dyJqRONg3w==", + "dev": true, + "funding": { + "type": "github", + "url": "https://github.com/sponsors/wooorm" + } + }, + "node_modules/verror": { + "version": "1.10.0", + "resolved": "https://registry.npmjs.org/verror/-/verror-1.10.0.tgz", + "integrity": "sha1-OhBcoXBTr1XW4nDB+CiGguGNpAA=", + "dev": true, + "engines": [ + "node >=0.6.0" + ], + "dependencies": { + "assert-plus": "^1.0.0", + "core-util-is": "1.0.2", + "extsprintf": "^1.2.0" + } + }, + "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 + }, + "node_modules/void-elements": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/void-elements/-/void-elements-2.0.1.tgz", + "integrity": "sha1-wGavtYK7HLQSjWDqkjkulNXp2+w=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack": { + "version": "1.7.5", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.7.5.tgz", + "integrity": "sha512-9P3MWk6SrKjHsGkLT2KHXdQ/9SNkyoJbabxnKOoJepsvJjJG8uYTR3yTPxPQvNDI3w4Nz1xnE0TLHK4RIVe/MQ==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "neo-async": "^2.5.0" + }, + "optionalDependencies": { + "chokidar": "^3.4.1", + "watchpack-chokidar2": "^2.0.1" + } + }, + "node_modules/watchpack-chokidar2": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/watchpack-chokidar2/-/watchpack-chokidar2-2.0.1.tgz", + "integrity": "sha512-nCFfBIPKr5Sh61s4LPpy1Wtfi0HE8isJ3d2Yb5/Ppw2P2B/3eVSEBjKfN0fmHJSK14+31KwMKmcrzs2GM4P0Ww==", + "dev": true, + "optional": true, + "dependencies": { + "chokidar": "^2.1.8" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "deprecated": "Chokidar 2 will break on node v14+. Upgrade to chokidar 3 with 15x less dependencies.", + "dev": true, + "optional": true, + "dependencies": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" + }, + "optionalDependencies": { + "fsevents": "^1.2.7" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "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/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/node_modules/fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "dependencies": { + "bindings": "^1.5.0", + "nan": "^2.12.1" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/watchpack-chokidar2/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, + "optional": true, + "dependencies": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/wbuf": { + "version": "1.7.3", + "resolved": "https://registry.npmjs.org/wbuf/-/wbuf-1.7.3.tgz", + "integrity": "sha512-O84QOnr0icsbFGLS0O3bI5FswxzRr8/gHwWkDlQFskhSPryQXvrTMxjxGP4+iWYoauLoBvfDpkrOauZ+0iZpDA==", + "dev": true, + "dependencies": { + "minimalistic-assert": "^1.0.0" + } + }, + "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, + "dependencies": { + "defaults": "^1.0.3" + } + }, + "node_modules/webdriver-js-extender": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/webdriver-js-extender/-/webdriver-js-extender-2.1.0.tgz", + "integrity": "sha512-lcUKrjbBfCK6MNsh7xaY2UAUmZwe+/ib03AjVOpFobX4O7+83BUveSrLfU0Qsyb1DaKJdQRbuU+kM9aZ6QUhiQ==", + "dev": true, + "dependencies": { + "@types/selenium-webdriver": "^3.0.0", + "selenium-webdriver": "^3.0.1" + }, + "engines": { + "node": ">=6.9.x" + } + }, + "node_modules/webpack": { + "version": "4.44.2", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.44.2.tgz", + "integrity": "sha512-6KJVGlCxYdISyurpQ0IPTklv+DULv05rs2hseIXer6D7KrUicRDLFb4IUM1S6LUAKypPM/nSiVSuv8jHu1m3/Q==", + "dev": true, + "dependencies": { + "@webassemblyjs/ast": "1.9.0", + "@webassemblyjs/helper-module-context": "1.9.0", + "@webassemblyjs/wasm-edit": "1.9.0", + "@webassemblyjs/wasm-parser": "1.9.0", + "acorn": "^6.4.1", + "ajv": "^6.10.2", + "ajv-keywords": "^3.4.1", + "chrome-trace-event": "^1.0.2", + "enhanced-resolve": "^4.3.0", + "eslint-scope": "^4.0.3", + "json-parse-better-errors": "^1.0.2", + "loader-runner": "^2.4.0", + "loader-utils": "^1.2.3", + "memory-fs": "^0.4.1", + "micromatch": "^3.1.10", + "mkdirp": "^0.5.3", + "neo-async": "^2.6.1", + "node-libs-browser": "^2.2.1", + "schema-utils": "^1.0.0", + "tapable": "^1.1.3", + "terser-webpack-plugin": "^1.4.3", + "watchpack": "^1.7.4", + "webpack-sources": "^1.4.1" + }, + "bin": { + "webpack": "bin/webpack.js" + }, + "engines": { + "node": ">=6.11.5" + }, + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/webpack" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + }, + "webpack-command": { + "optional": true + } + } + }, + "node_modules/webpack-dev-middleware": { + "version": "3.7.2", + "resolved": "https://registry.npmjs.org/webpack-dev-middleware/-/webpack-dev-middleware-3.7.2.tgz", + "integrity": "sha512-1xC42LxbYoqLNAhV6YzTYacicgMZQTqRd27Sim9wn5hJrX3I5nxYy1SxSd4+gjUFsz1dQFj+yEe6zEVmSkeJjw==", + "dev": true, + "dependencies": { + "memory-fs": "^0.4.1", + "mime": "^2.4.4", + "mkdirp": "^0.5.1", + "range-parser": "^1.2.1", + "webpack-log": "^2.0.0" + }, + "engines": { + "node": ">= 6" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/webpack-dev-middleware/node_modules/mime": { + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.5.2.tgz", + "integrity": "sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg==", + "dev": true, + "bin": { + "mime": "cli.js" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/webpack-dev-server": { + "version": "3.11.2", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.11.2.tgz", + "integrity": "sha512-A80BkuHRQfCiNtGBS1EMf2ChTUs0x+B3wGDFmOeT4rmJOHhHTCH2naNxIHhmkr0/UillP4U3yeIyv1pNp+QDLQ==", + "dev": true, + "dependencies": { + "ansi-html": "0.0.7", + "bonjour": "^3.5.0", + "chokidar": "^2.1.8", + "compression": "^1.7.4", + "connect-history-api-fallback": "^1.6.0", + "debug": "^4.1.1", + "del": "^4.1.1", + "express": "^4.17.1", + "html-entities": "^1.3.1", + "http-proxy-middleware": "0.19.1", + "import-local": "^2.0.0", + "internal-ip": "^4.3.0", + "ip": "^1.1.5", + "is-absolute-url": "^3.0.3", + "killable": "^1.0.1", + "loglevel": "^1.6.8", + "opn": "^5.5.0", + "p-retry": "^3.0.1", + "portfinder": "^1.0.26", + "schema-utils": "^1.0.0", + "selfsigned": "^1.10.8", + "semver": "^6.3.0", + "serve-index": "^1.9.1", + "sockjs": "^0.3.21", + "sockjs-client": "^1.5.0", + "spdy": "^4.0.2", + "strip-ansi": "^3.0.1", + "supports-color": "^6.1.0", + "url": "^0.11.0", + "webpack-dev-middleware": "^3.7.2", + "webpack-log": "^2.0.0", + "ws": "^6.2.1", + "yargs": "^13.3.2" + }, + "bin": { + "webpack-dev-server": "bin/webpack-dev-server.js" + }, + "engines": { + "node": ">= 6.11.5" + }, + "peerDependencies": { + "webpack": "^4.0.0 || ^5.0.0" + }, + "peerDependenciesMeta": { + "webpack-cli": { + "optional": true + } + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "micromatch": "^3.1.4", + "normalize-path": "^2.1.1" + } + }, + "node_modules/webpack-dev-server/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": { + "remove-trailing-separator": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/chokidar": { + "version": "2.1.8", + "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-2.1.8.tgz", + "integrity": "sha512-ZmZUazfOzf0Nve7duiCKD23PFSCs4JPoYyccjUFF3aQkQadqBhfzhjkwBH2mNOG9cTBwhamM37EIsIkZw3nRgg==", + "deprecated": "Chokidar 2 will break on node v14+. Upgrade to chokidar 3 with 15x less dependencies.", + "dev": true, + "dependencies": { + "anymatch": "^2.0.0", + "async-each": "^1.0.1", + "braces": "^2.3.2", + "glob-parent": "^3.1.0", + "inherits": "^2.0.3", + "is-binary-path": "^1.0.0", + "is-glob": "^4.0.0", + "normalize-path": "^3.0.0", + "path-is-absolute": "^1.0.0", + "readdirp": "^2.2.1", + "upath": "^1.1.1" + }, + "optionalDependencies": { + "fsevents": "^1.2.7" + } + }, + "node_modules/webpack-dev-server/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, + "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/webpack-dev-server/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/node_modules/fsevents": { + "version": "1.2.13", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-1.2.13.tgz", + "integrity": "sha512-oWb1Z6mkHIskLzEJ/XWX0srkpkTQ7vaopMQkyaEIoq0fmtFVxOthb8cCxeT+p3ynTdkk/RZwbgG4brR5BeWECw==", + "deprecated": "fsevents 1 will break on node v14+ and could be using insecure binaries. Upgrade to fsevents 2.", + "dev": true, + "hasInstallScript": true, + "optional": true, + "os": [ + "darwin" + ], + "dependencies": { + "bindings": "^1.5.0", + "nan": "^2.12.1" + }, + "engines": { + "node": ">= 4.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "is-glob": "^3.1.0", + "path-dirname": "^1.0.0" + } + }, + "node_modules/webpack-dev-server/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-extglob": "^2.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "binary-extensions": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-dev-server/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, + "dependencies": { + "graceful-fs": "^4.1.11", + "micromatch": "^3.1.10", + "readable-stream": "^2.0.2" + }, + "engines": { + "node": ">=0.10" + } + }, + "node_modules/webpack-dev-server/node_modules/schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "dependencies": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/webpack-dev-server/node_modules/semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true, + "bin": { + "semver": "bin/semver.js" + } + }, + "node_modules/webpack-dev-server/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/webpack-dev-server/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, + "dependencies": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-log": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/webpack-log/-/webpack-log-2.0.0.tgz", + "integrity": "sha512-cX8G2vR/85UYG59FgkoMamwHUIkSSlV3bBMRsbxVXVUk2j6NleCKjQ/WE9eYg9WY4w25O9w8wKP4rzNZFmUcUg==", + "dev": true, + "dependencies": { + "ansi-colors": "^3.0.0", + "uuid": "^3.3.2" + }, + "engines": { + "node": ">= 6" + } + }, + "node_modules/webpack-log/node_modules/ansi-colors": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.4.tgz", + "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack-log/node_modules/uuid": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", + "integrity": "sha512-HjSDRw6gZE5JMggctHBcjVak08+KEVhSIiDzFnT9S9aegmp85S/bReBVTb4QTFaRNptJ9kuYaNhnbNEOkbKb/A==", + "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" + } + }, + "node_modules/webpack-merge": { + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.7.3.tgz", + "integrity": "sha512-6/JUQv0ELQ1igjGDzHkXbVDRxkfA57Zw7PfiupdLFJYrgFqY5ZP8xxbpp2lU3EPwYx89ht5Z/aDkD40hFCm5AA==", + "dev": true, + "dependencies": { + "clone-deep": "^4.0.1", + "wildcard": "^2.0.0" + }, + "engines": { + "node": ">=10.0.0" + } + }, + "node_modules/webpack-sources": { + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-2.2.0.tgz", + "integrity": "sha512-bQsA24JLwcnWGArOKUxYKhX3Mz/nK1Xf6hxullKERyktjNMC4x8koOeaDNTA2fEJ09BdWLbM/iTW0ithREUP0w==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.1", + "source-map": "^0.6.1" + }, + "engines": { + "node": ">=10.13.0" + } + }, + "node_modules/webpack-sources/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-subresource-integrity": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/webpack-subresource-integrity/-/webpack-subresource-integrity-1.5.2.tgz", + "integrity": "sha512-GBWYBoyalbo5YClwWop9qe6Zclp8CIXYGIz12OPclJhIrSplDxs1Ls1JDMH8xBPPrg1T6ISaTW9Y6zOrwEiAzw==", + "dev": true, + "dependencies": { + "webpack-sources": "^1.3.0" + }, + "engines": { + "node": ">=4" + }, + "peerDependencies": { + "html-webpack-plugin": ">= 2.21.0 < 5", + "webpack": ">= 1.12.11 < 6" + }, + "peerDependenciesMeta": { + "html-webpack-plugin": { + "optional": true + } + } + }, + "node_modules/webpack-subresource-integrity/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack-subresource-integrity/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/webpack/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, + "dependencies": { + "arr-flatten": "^1.1.0", + "array-unique": "^0.3.2", + "extend-shallow": "^2.0.1", + "fill-range": "^4.0.0", + "isobject": "^3.0.1", + "repeat-element": "^1.1.2", + "snapdragon": "^0.8.1", + "snapdragon-node": "^2.0.1", + "split-string": "^3.0.2", + "to-regex": "^3.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/node_modules/cacache": { + "version": "12.0.4", + "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", + "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", + "dev": true, + "dependencies": { + "bluebird": "^3.5.5", + "chownr": "^1.1.1", + "figgy-pudding": "^3.5.1", + "glob": "^7.1.4", + "graceful-fs": "^4.1.15", + "infer-owner": "^1.0.3", + "lru-cache": "^5.1.1", + "mississippi": "^3.0.0", + "mkdirp": "^0.5.1", + "move-concurrently": "^1.0.1", + "promise-inflight": "^1.0.1", + "rimraf": "^2.6.3", + "ssri": "^6.0.1", + "unique-filename": "^1.1.1", + "y18n": "^4.0.0" + } + }, + "node_modules/webpack/node_modules/chownr": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", + "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "dev": true + }, + "node_modules/webpack/node_modules/enhanced-resolve": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.5.0.tgz", + "integrity": "sha512-Nv9m36S/vxpsI+Hc4/ZGRs0n9mXqSWGGq49zxb/cJfPAQMbUtttJAlNPS4AQzaBdw/pKskw5bMbekT/Y7W/Wlg==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "memory-fs": "^0.5.0", + "tapable": "^1.0.0" + }, + "engines": { + "node": ">=6.9.0" + } + }, + "node_modules/webpack/node_modules/enhanced-resolve/node_modules/memory-fs": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", + "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "dev": true, + "dependencies": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + }, + "engines": { + "node": ">=4.3.0 <5.0.0 || >=5.10" + } + }, + "node_modules/webpack/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, + "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/webpack/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": { + "is-extendable": "^0.1.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/node_modules/find-cache-dir": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", + "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", + "dev": true, + "dependencies": { + "commondir": "^1.0.1", + "make-dir": "^2.0.0", + "pkg-dir": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack/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/webpack/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, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/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, + "dependencies": { + "kind-of": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/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": { + "is-buffer": "^1.1.5" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/webpack/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, + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/webpack/node_modules/loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/webpack/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/webpack/node_modules/lru-cache": { + "version": "5.1.1", + "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", + "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "dev": true, + "dependencies": { + "yallist": "^3.0.2" + } + }, + "node_modules/webpack/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, + "dependencies": { + "arr-diff": "^4.0.0", + "array-unique": "^0.3.2", + "braces": "^2.3.1", + "define-property": "^2.0.2", + "extend-shallow": "^3.0.2", + "extglob": "^2.0.4", + "fragment-cache": "^0.2.1", + "kind-of": "^6.0.2", + "nanomatch": "^1.2.9", + "object.pick": "^1.3.0", + "regex-not": "^1.0.0", + "snapdragon": "^0.8.1", + "to-regex": "^3.0.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/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/webpack/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/webpack/node_modules/pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "dependencies": { + "find-up": "^3.0.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack/node_modules/rimraf": { + "version": "2.7.1", + "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", + "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "dev": true, + "dependencies": { + "glob": "^7.1.3" + }, + "bin": { + "rimraf": "bin.js" + } + }, + "node_modules/webpack/node_modules/schema-utils": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", + "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "dev": true, + "dependencies": { + "ajv": "^6.1.0", + "ajv-errors": "^1.0.0", + "ajv-keywords": "^3.1.0" + }, + "engines": { + "node": ">= 4" + } + }, + "node_modules/webpack/node_modules/serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "dependencies": { + "randombytes": "^2.1.0" + } + }, + "node_modules/webpack/node_modules/source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/node_modules/ssri": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", + "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", + "dev": true, + "dependencies": { + "figgy-pudding": "^3.5.1" + } + }, + "node_modules/webpack/node_modules/tapable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", + "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/webpack/node_modules/terser": { + "version": "4.8.0", + "resolved": "https://registry.npmjs.org/terser/-/terser-4.8.0.tgz", + "integrity": "sha512-EAPipTNeWsb/3wLPeup1tVPaXfIaU68xMnVdPafIL1TV05OhASArYyIfFvnvJCNrR2NIOvDVNNTFRa+Re2MWyw==", + "dev": true, + "dependencies": { + "commander": "^2.20.0", + "source-map": "~0.6.1", + "source-map-support": "~0.5.12" + }, + "bin": { + "terser": "bin/terser" + }, + "engines": { + "node": ">=6.0.0" + } + }, + "node_modules/webpack/node_modules/terser-webpack-plugin": { + "version": "1.4.5", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-1.4.5.tgz", + "integrity": "sha512-04Rfe496lN8EYruwi6oPQkG0vo8C+HT49X687FZnpPF0qMAIHONI6HEXYPKDOE8e5HjXTyKfqRd/agHtH0kOtw==", + "dev": true, + "dependencies": { + "cacache": "^12.0.2", + "find-cache-dir": "^2.1.0", + "is-wsl": "^1.1.0", + "schema-utils": "^1.0.0", + "serialize-javascript": "^4.0.0", + "source-map": "^0.6.1", + "terser": "^4.1.2", + "webpack-sources": "^1.4.0", + "worker-farm": "^1.7.0" + }, + "engines": { + "node": ">= 6.9.0" + }, + "peerDependencies": { + "webpack": "^4.0.0" + } + }, + "node_modules/webpack/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, + "dependencies": { + "is-number": "^3.0.0", + "repeat-string": "^1.6.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, + "node_modules/webpack/node_modules/webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "dependencies": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, + "node_modules/webpack/node_modules/yallist": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", + "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "dev": true + }, + "node_modules/websocket-driver": { + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz", + "integrity": "sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==", + "dev": true, + "dependencies": { + "http-parser-js": ">=0.5.1", + "safe-buffer": ">=5.1.0", + "websocket-extensions": ">=0.1.1" + }, + "engines": { + "node": ">=0.8.0" + } + }, + "node_modules/websocket-extensions": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/websocket-extensions/-/websocket-extensions-0.1.4.tgz", + "integrity": "sha512-OqedPIGOfsDlo31UNwYbCFMSaO9m9G/0faIHj5/dZFDMFqPTcx6UwqyOy3COEaEOg/9VsGIpdqn62W5KhoKSpg==", + "dev": true, + "engines": { + "node": ">=0.8.0" + } + }, + "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 + }, + "node_modules/which": { + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", + "integrity": "sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ==", + "dev": true, + "dependencies": { + "isexe": "^2.0.0" + }, + "bin": { + "which": "bin/which" + } + }, + "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 + }, + "node_modules/wide-align": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", + "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", + "dev": true, + "dependencies": { + "string-width": "^1.0.2 || 2" + } + }, + "node_modules/wide-align/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/wide-align/node_modules/string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "dependencies": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/wide-align/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, + "dependencies": { + "ansi-regex": "^3.0.0" + }, + "engines": { + "node": ">=4" + } + }, + "node_modules/wildcard": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.0.tgz", + "integrity": "sha512-JcKqAHLPxcdb9KM49dufGXn2x3ssnfjbcaQdLlfZsL9rH9wgDQjUtDxbo8NE0F6SFvydeu1VhZe7hZuHsB2/pw==", + "dev": true + }, + "node_modules/worker-farm": { + "version": "1.7.0", + "resolved": "https://registry.npmjs.org/worker-farm/-/worker-farm-1.7.0.tgz", + "integrity": "sha512-rvw3QTZc8lAxyVrqcSGVm5yP/IJ2UcB3U0graE3LCFoZ0Yn2x4EoVSqJKdB/T5M+FLcRPjz4TDacRf3OCfNUzw==", + "dev": true, + "dependencies": { + "errno": "~0.1.7" + } + }, + "node_modules/worker-plugin": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/worker-plugin/-/worker-plugin-5.0.0.tgz", + "integrity": "sha512-AXMUstURCxDD6yGam2r4E34aJg6kW85IiaeX72hi+I1cxyaMUtrvVY6sbfpGKAj5e7f68Acl62BjQF5aOOx2IQ==", + "dev": true, + "dependencies": { + "loader-utils": "^1.1.0" + }, + "peerDependencies": { + "webpack": ">= 4" + } + }, + "node_modules/worker-plugin/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, + "dependencies": { + "minimist": "^1.2.0" + }, + "bin": { + "json5": "lib/cli.js" + } + }, + "node_modules/worker-plugin/node_modules/loader-utils": { + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", + "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", + "dev": true, + "dependencies": { + "big.js": "^5.2.2", + "emojis-list": "^3.0.0", + "json5": "^1.0.1" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "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, + "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": "^4.1.0" + }, + "engines": { + "node": ">=6" + } + }, + "node_modules/wrappy": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" + }, + "node_modules/ws": { + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.2.tgz", + "integrity": "sha512-zmhltoSR8u1cnDsD43TX59mzoMZsLKqUweyYBAIvTngR3shc0W6aOZylZmq/7hqyVxPdi+5Ud2QInblgyE72fw==", + "dev": true, + "dependencies": { + "async-limiter": "~1.0.0" + } + }, + "node_modules/xml2js": { + "version": "0.4.23", + "resolved": "https://registry.npmjs.org/xml2js/-/xml2js-0.4.23.tgz", + "integrity": "sha512-ySPiMjM0+pLDftHgXY4By0uswI3SPKLDw/i3UXbnO8M/p28zqexCUoPmQFrYD+/1BzhGJSs2i1ERWKJAtiLrug==", + "dev": true, + "dependencies": { + "sax": ">=0.6.0", + "xmlbuilder": "~11.0.0" + }, + "engines": { + "node": ">=4.0.0" + } + }, + "node_modules/xmlbuilder": { + "version": "11.0.1", + "resolved": "https://registry.npmjs.org/xmlbuilder/-/xmlbuilder-11.0.1.tgz", + "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", + "dev": true, + "engines": { + "node": ">=4.0" + } + }, + "node_modules/xtend": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", + "integrity": "sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==", + "dev": true, + "engines": { + "node": ">=0.4" + } + }, + "node_modules/y18n": { + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", + "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==", + "dev": true + }, + "node_modules/yallist": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/yallist/-/yallist-4.0.0.tgz", + "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", + "dev": true + }, + "node_modules/yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, + "node_modules/yargs": { + "version": "13.3.2", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", + "integrity": "sha512-AX3Zw5iPruN5ie6xGRIDgqkT+ZhnRlZMLMHAs8tg7nRruy2Nb+i5o9bwghAogtM08q1dpr2LVoS8KSTMYpWXUw==", + "dev": true, + "dependencies": { + "cliui": "^5.0.0", + "find-up": "^3.0.0", + "get-caller-file": "^2.0.1", + "require-directory": "^2.1.1", + "require-main-filename": "^2.0.0", + "set-blocking": "^2.0.0", + "string-width": "^3.0.0", + "which-module": "^2.0.0", + "y18n": "^4.0.0", + "yargs-parser": "^13.1.2" + } + }, + "node_modules/yargs-parser": { + "version": "13.1.2", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-13.1.2.tgz", + "integrity": "sha512-3lbsNRf/j+A4QuSZfDRA7HRSfWrzO0YjqTJd5kjAq37Zep1CEgaYmrH9Q3GwPiB9cHyd1Y1UwggGhJGoxipbzg==", + "dev": true, + "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-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/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, + "engines": { + "node": ">=4" + } + }, + "node_modules/yn": { + "version": "3.1.1", + "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", + "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", + "dev": true, + "engines": { + "node": ">=6" + } + }, + "node_modules/yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, + "node_modules/zone.js": { + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.11.4.tgz", + "integrity": "sha512-DDh2Ab+A/B+9mJyajPjHFPWfYU1H+pdun4wnnk0OcQTNjem1XQSZ2CDW+rfZEUDjv5M19SBqAkjZi0x5wuB5Qw==", + "dependencies": { + "tslib": "^2.0.0" + } + } + }, + "dependencies": { + "@angular-devkit/architect": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/architect/-/architect-0.1102.14.tgz", + "integrity": "sha512-965TVXuBtRb8RySgxRxUEO+YTd7mT0xiqVHSe+MHvMtUCmEE9vwRofFZl6axkK5ri4fiomiMnOVE19aw4spgNQ==", + "dev": true, + "requires": { + "@angular-devkit/core": "11.2.14", + "rxjs": "6.6.3" + } + }, + "@angular-devkit/build-angular": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-angular/-/build-angular-0.1102.14.tgz", + "integrity": "sha512-SyX9SK3qfpk6xNIrxpxYi8zxP/cN2kny4I+XYbkKvgGiE3qhkrC/PRJE9OWj0sloekLD0CDfFWOvIiw3GMc4Tg==", + "dev": true, + "requires": { + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/build-optimizer": "0.1102.14", + "@angular-devkit/build-webpack": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "@babel/core": "7.12.10", + "@babel/generator": "7.12.11", + "@babel/plugin-transform-async-to-generator": "7.12.1", + "@babel/plugin-transform-runtime": "7.12.10", + "@babel/preset-env": "7.12.11", + "@babel/runtime": "7.12.5", + "@babel/template": "7.12.7", + "@discoveryjs/json-ext": "0.5.2", + "@jsdevtools/coverage-istanbul-loader": "3.0.5", + "@ngtools/webpack": "11.2.14", + "ansi-colors": "4.1.1", + "autoprefixer": "10.2.4", + "babel-loader": "8.2.2", + "browserslist": "^4.9.1", + "cacache": "15.0.5", + "caniuse-lite": "^1.0.30001032", + "circular-dependency-plugin": "5.2.2", + "copy-webpack-plugin": "6.3.2", + "core-js": "3.8.3", + "critters": "0.0.7", + "css-loader": "5.0.1", + "cssnano": "5.0.2", + "file-loader": "6.2.0", + "find-cache-dir": "3.3.1", + "glob": "7.1.6", + "https-proxy-agent": "5.0.0", + "inquirer": "7.3.3", + "jest-worker": "26.6.2", + "karma-source-map-support": "1.4.0", + "less": "4.1.1", + "less-loader": "7.3.0", + "license-webpack-plugin": "2.3.11", + "loader-utils": "2.0.0", + "mini-css-extract-plugin": "1.3.5", + "minimatch": "3.0.4", + "open": "7.4.0", + "ora": "5.3.0", + "parse5-html-rewriting-stream": "6.0.1", + "pnp-webpack-plugin": "1.6.4", + "postcss": "8.2.15", + "postcss-import": "14.0.0", + "postcss-loader": "4.2.0", + "raw-loader": "4.0.2", + "regenerator-runtime": "0.13.7", + "resolve-url-loader": "4.0.0", + "rimraf": "3.0.2", + "rollup": "2.38.4", + "rxjs": "6.6.3", + "sass": "1.32.6", + "sass-loader": "10.1.1", + "semver": "7.3.4", + "source-map": "0.7.3", + "source-map-loader": "1.1.3", + "source-map-support": "0.5.19", + "speed-measure-webpack-plugin": "1.4.2", + "style-loader": "2.0.0", + "stylus": "0.54.8", + "stylus-loader": "4.3.3", + "terser": "5.5.1", + "terser-webpack-plugin": "4.2.3", + "text-table": "0.2.0", + "tree-kill": "1.2.2", + "webpack": "4.44.2", + "webpack-dev-middleware": "3.7.2", + "webpack-dev-server": "3.11.2", + "webpack-merge": "5.7.3", + "webpack-sources": "2.2.0", + "webpack-subresource-integrity": "1.5.2", + "worker-plugin": "5.0.0" + }, + "dependencies": { + "@babel/template": { + "version": "7.12.7", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.12.7.tgz", + "integrity": "sha512-GkDzmHS6GV7ZeXfJZ0tLRBhZcMcY0/Lnb+eEbXDBfCAcZCjrZKe6p3J4we/D24O9Y8enxWAg1cWwof59yLh2ow==", + "dev": true, + "requires": { + "@babel/code-frame": "^7.10.4", + "@babel/parser": "^7.12.7", + "@babel/types": "^7.12.7" + } + }, + "agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", + "dev": true, + "requires": { + "debug": "4" + } + }, + "https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", + "dev": true, + "requires": { + "agent-base": "6", + "debug": "4" + } + }, + "semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "@angular-devkit/build-optimizer": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-optimizer/-/build-optimizer-0.1102.14.tgz", + "integrity": "sha512-1j69rFqE6tPMO0lQvOH8ogF7vE+p+Ws1/OtdZKUkZPOerIbQ8A3n5wzCx6/ZzMVhBQ3sXNhaShb4b9/1YuwU/g==", + "dev": true, + "requires": { + "loader-utils": "2.0.0", + "source-map": "0.7.3", + "tslib": "2.1.0", + "typescript": "4.1.5", + "webpack-sources": "2.2.0" + }, + "dependencies": { + "tslib": { + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/tslib/-/tslib-2.1.0.tgz", + "integrity": "sha512-hcVC3wYEziELGGmEEXue7D75zbwIIVUMWAVbHItGPx0ziyXxrOMQx4rQEVEV45Ut/1IotuEvwqPopzIOkDMf0A==", + "dev": true + } + } + }, + "@angular-devkit/build-webpack": { + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/build-webpack/-/build-webpack-0.1102.14.tgz", + "integrity": "sha512-+dJvzrwjbHY0bNr8fUDVbn4D4pAT/h1YVpGVyaoX7q66LN0x61zRC3e10gJ/Mr54l3yfc26M0OPD9KG8iZRbCA==", + "dev": true, + "requires": { + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "rxjs": "6.6.3" + } + }, + "@angular-devkit/core": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-11.2.14.tgz", + "integrity": "sha512-Ad1fHqLxDwhkQgLPqq9i+G65NSOoIHXQx7ILcSPACKurV3XLS1RO9BgP/BDaqHAG+WslUAPbMStaTzzPm+9dNw==", + "dev": true, + "requires": { + "ajv": "6.12.6", + "fast-json-stable-stringify": "2.1.0", + "magic-string": "0.25.7", + "rxjs": "6.6.3", + "source-map": "0.7.3" + } + }, + "@angular-devkit/schematics": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular-devkit/schematics/-/schematics-11.2.14.tgz", + "integrity": "sha512-Ol6+0qdGKzuVJm5gCtQr47X0OCihTfAxI4h047cHYhPFIGGPSvkG/QeJMZugflgoobi2k/xcYokOu/VAkRtWbQ==", + "dev": true, + "requires": { + "@angular-devkit/core": "11.2.14", + "ora": "5.3.0", + "rxjs": "6.6.3" + } + }, + "@angular/animations": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/animations/-/animations-11.2.14.tgz", + "integrity": "sha512-Heq/nNrCmb3jbkusu+BQszOecfFI/31Oxxj+CDQkqqYpBcswk6bOJLoEE472o+vmgxaXbgeflU9qbIiCQhpMFA==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/cli": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/cli/-/cli-11.2.14.tgz", + "integrity": "sha512-8Ud7vcUK7CKjzT2Ks1glLhleAPIC5ChcrA15XtOb7k+/uMHBkMscP/UKymbVQiBjCJlglbzJoyj8cpVYTZY5KA==", + "dev": true, + "requires": { + "@angular-devkit/architect": "0.1102.14", + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", + "@schematics/angular": "11.2.14", + "@schematics/update": "0.1102.14", + "@yarnpkg/lockfile": "1.1.0", + "ansi-colors": "4.1.1", + "debug": "4.3.1", + "ini": "2.0.0", + "inquirer": "7.3.3", + "jsonc-parser": "3.0.0", + "npm-package-arg": "8.1.0", + "npm-pick-manifest": "6.1.0", + "open": "7.4.0", + "ora": "5.3.0", + "pacote": "11.2.4", + "resolve": "1.19.0", + "rimraf": "3.0.2", + "semver": "7.3.4", + "symbol-observable": "3.0.0", + "universal-analytics": "0.4.23", + "uuid": "8.3.2" + }, + "dependencies": { + "ini": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz", + "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==", + "dev": true + }, + "semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } + } + }, + "@angular/common": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/common/-/common-11.2.14.tgz", + "integrity": "sha512-ZSLV/3j7eCTyLf/8g4yBFLWySjiLz3vLJAGWscYoUpnJWMnug1VRu6zoF/COxCbtORgE+Wz6K0uhfS6MziBGVw==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/compiler": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/compiler/-/compiler-11.2.14.tgz", + "integrity": "sha512-XBOK3HgA+/y6Cz7kOX4zcJYmgJ264XnfcbXUMU2cD7Ac+mbNhLPKohWrEiSWalfcjnpf5gRfufQrQP7lpAGu0A==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@angular/compiler-cli": { + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/compiler-cli/-/compiler-cli-11.2.14.tgz", + "integrity": "sha512-A7ltnCp03/EVqK/Q3tVUDsokgz5GHW3dSPGl0Csk7Ys5uBB9ibHTmVt4eiXA4jt0+6Bk+mKxwe5BEDqLvwYFAg==", + "requires": { + "@babel/core": "^7.8.6", + "@babel/types": "^7.8.6", + "canonical-path": "1.0.0", + "chokidar": "^3.0.0", + "convert-source-map": "^1.5.1", + "dependency-graph": "^0.7.2", + "fs-extra": "4.0.2", + "magic-string": "^0.25.0", + "minimist": "^1.2.0", + "reflect-metadata": "^0.1.2", + "semver": "^6.3.0", + "source-map": "^0.6.1", + "sourcemap-codec": "^1.4.8", + "tslib": "^2.0.0", + "yargs": "^16.2.0" }, "dependencies": { "ansi-regex": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-5.0.0.tgz", - "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==", - "dev": true + "integrity": "sha512-bY6fj56OUQ0hU1KjFNDQuJFezqKdrAyFdIevADiqrWHwSlbmBNMHp5ak2f40Pm8JTFyM2mqxkG6ngkHO11f/lg==" }, "ansi-styles": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", - "dev": true, "requires": { "color-convert": "^2.0.1" } }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", - "dev": true, + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", "requires": { "string-width": "^4.2.0", "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.0" + "wrap-ansi": "^7.0.0" } }, "color-convert": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", - "dev": true, "requires": { "color-name": "~1.1.4" } @@ -420,72 +17449,32 @@ "color-name": { "version": "1.1.4", "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", - "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", - "dev": true + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==" }, "emoji-regex": { "version": "8.0.0", "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", - "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", - "dev": true - }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } + "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, "is-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", - "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", - "dev": true - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true + "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" }, "semver": { "version": "6.3.0", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", - "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", - "dev": true + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==" }, "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 + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==" }, "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", - "dev": true, + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", "requires": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", @@ -496,77 +17485,70 @@ "version": "6.0.0", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-6.0.0.tgz", "integrity": "sha512-AuvKTrTfQNYNIctbR1K/YGTR1756GycPsg7b9bdV9Duqur4gv6aKqHXah67Z8ImS7WEz5QVcOtlfW2rZEugt6w==", - "dev": true, "requires": { "ansi-regex": "^5.0.0" } }, "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", - "dev": true, + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", "requires": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, + "y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==" + }, "yargs": { - "version": "15.3.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.3.0.tgz", - "integrity": "sha512-g/QCnmjgOl1YJjGsnUg2SatC7NUYEiLXJqxNOQU9qSpjzGtGXda9b+OKccr1kLTy8BN9yqEyqfq5lxlwdc13TA==", - "dev": true, + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.0" + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" } }, "yargs-parser": { - "version": "18.1.3", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", - "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==" } } }, "@angular/core": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/core/-/core-10.1.5.tgz", - "integrity": "sha512-B8j1B5vkBmzyan78kMJhw7dfhe7znmujbeDU7qRgRcIllc9pVJv7D133Yze6JFiLVg21PfyFYs8FBJNeq39hxQ==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/core/-/core-11.2.14.tgz", + "integrity": "sha512-vpR4XqBGitk1Faph37CSpemwIYTmJ3pdIVNoHKP6jLonpWu+0azkchf0f7oD8/2ivj2F81opcIw0tcsy/D/5Vg==", "requires": { "tslib": "^2.0.0" } }, "@angular/forms": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-10.1.5.tgz", - "integrity": "sha512-fkXKCwXL0XeFMUkmzJpm+FHYrv1CCfFGxYEBQ/bzfd3Op+dFJqEPiOwK3wG943Y09THday6H509RwwEIyF/4yw==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/forms/-/forms-11.2.14.tgz", + "integrity": "sha512-4LWqY6KEIk1AZQFnk+4PJSOCamlD4tumuVN06gO4D0dZo9Cx+GcvW6pM6N0CPubRvPs3sScCnu20WT11HNWC1w==", "requires": { "tslib": "^2.0.0" } }, "@angular/localize": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-10.1.5.tgz", - "integrity": "sha512-wEiF9lLWulTwGFqFJ4dJ+mBiyylE0amJR4leUGHKu6iwjPUQUBCqReCapfvLLTfgt3nbq7DtVqlqxj7KW7w+xg==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/localize/-/localize-11.2.14.tgz", + "integrity": "sha512-ssMuquxxqxA98LgEICEO/3JdmSflWxu5rlm/HPo28bnGiZ4IzDamZjJ1cu4S6RgsonJ1drB3Z8wkidXfEYZiWA==", "requires": { "@babel/core": "7.8.3", "glob": "7.1.2", - "yargs": "15.3.0" + "yargs": "^16.2.0" }, "dependencies": { "@babel/core": { @@ -604,19 +17586,14 @@ "color-convert": "^2.0.1" } }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==" - }, "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", "requires": { "string-width": "^4.2.0", "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.0" + "wrap-ansi": "^7.0.0" } }, "color-convert": { @@ -637,15 +17614,6 @@ "resolved": "https://registry.npmjs.org/emoji-regex/-/emoji-regex-8.0.0.tgz", "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==" }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, "glob": { "version": "7.1.2", "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.2.tgz", @@ -664,27 +17632,6 @@ "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==" }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "requires": { - "p-limit": "^2.2.0" - } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==" - }, "semver": { "version": "5.7.1", "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", @@ -696,9 +17643,9 @@ "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" }, "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", "requires": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", @@ -714,102 +17661,97 @@ } }, "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", "requires": { "ansi-styles": "^4.0.0", "string-width": "^4.1.0", "strip-ansi": "^6.0.0" } }, + "y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==" + }, "yargs": { - "version": "15.3.0", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.3.0.tgz", - "integrity": "sha512-g/QCnmjgOl1YJjGsnUg2SatC7NUYEiLXJqxNOQU9qSpjzGtGXda9b+OKccr1kLTy8BN9yqEyqfq5lxlwdc13TA==", + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.0" + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" } }, "yargs-parser": { - "version": "18.1.3", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", - "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==" } } }, "@angular/platform-browser": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-10.1.5.tgz", - "integrity": "sha512-qMAoPHt6dgXMtieI4zx/s5yX7FFRRUDp1R4GMBCZHPN3p66WdEVxBJo4p5RWhZJioXpUwKz8Xvc+Rrh7r0KDBA==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/platform-browser/-/platform-browser-11.2.14.tgz", + "integrity": "sha512-fb7b7ss/gRoP8wLAN17W62leMgjynuyjEPU2eUoAAazsG9f2cgM+z3rK29GYncDVyYQxZUZYnjSqvL6GSXx86A==", "requires": { "tslib": "^2.0.0" } }, "@angular/platform-browser-dynamic": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-10.1.5.tgz", - "integrity": "sha512-wxHm1UFCtB+oU+IJ6pACGmjO9H8KVzJOLYL5hp2w0k8s7k7Zg73f6BdRgWWEEYv6uYIfF77qtKwgbH0X5H9S+w==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/platform-browser-dynamic/-/platform-browser-dynamic-11.2.14.tgz", + "integrity": "sha512-TWTPdFs6iBBcp+/YMsgCRQwdHpWGq8KjeJDJ2tfatGgBD3Gqt2YaHOMST1zPW6RkrmupytTejuVqXzeaKWFxuw==", "requires": { "tslib": "^2.0.0" } }, "@angular/router": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@angular/router/-/router-10.1.5.tgz", - "integrity": "sha512-tY88ZzoBrc9K67wi5V1NLnurd3r9bYR2csZ6/zJeOE+Vdxz9ChSaglgh9T0vQdbVEAjVGPP5QtYaFO2Xv4qOIg==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@angular/router/-/router-11.2.14.tgz", + "integrity": "sha512-3aYBmj+zrEL9yf/ntIQxHIYaWShZOBKP3U07X2mX+TPMpGlvHDnR7L6bWhQVZwewzMMz7YVR16ldg50IFuAlfA==", "requires": { "tslib": "^2.0.0" } }, "@babel/code-frame": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz", - "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.14.5.tgz", + "integrity": "sha512-9pzDqyc6OLDaqe+zbACgFkb6fKMNG6CObKpnYXChRsvYGyEdc7CA2BaqeOM+vOtCS5ndmJicPJhKAwYRI6UfFw==", "requires": { - "@babel/highlight": "^7.10.4" + "@babel/highlight": "^7.14.5" } }, "@babel/compat-data": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.12.1.tgz", - "integrity": "sha512-725AQupWJZ8ba0jbKceeFblZTY90McUBWMwHhkFQ9q1zKPJ95GUktljFcgcsIVwRnTnRKlcYzfiNImg5G9m6ZQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/compat-data/-/compat-data-7.14.5.tgz", + "integrity": "sha512-kixrYn4JwfAVPa0f2yfzc2AWti6WRRyO3XjWW5PJAvtE11qhSayrrcrEnee05KAtNaPC+EwehE8Qt1UedEVB8w==", "dev": true }, "@babel/core": { - "version": "7.11.1", - "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.11.1.tgz", - "integrity": "sha512-XqF7F6FWQdKGGWAzGELL+aCO1p+lRY5Tj5/tbT3St1G8NaH70jhhDIKknIZaDans0OQBG5wRAldROLHSt44BgQ==", - "dev": true, + "version": "7.12.10", + "resolved": "https://registry.npmjs.org/@babel/core/-/core-7.12.10.tgz", + "integrity": "sha512-eTAlQKq65zHfkHZV0sIVODCPGVgoo1HdBlbSLi9CqOzuZanMv2ihzY+4paiKr1mH+XmYESMAmJ/dpZ68eN6d8w==", "requires": { "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.11.0", - "@babel/helper-module-transforms": "^7.11.0", - "@babel/helpers": "^7.10.4", - "@babel/parser": "^7.11.1", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.11.0", - "@babel/types": "^7.11.0", + "@babel/generator": "^7.12.10", + "@babel/helper-module-transforms": "^7.12.1", + "@babel/helpers": "^7.12.5", + "@babel/parser": "^7.12.10", + "@babel/template": "^7.12.7", + "@babel/traverse": "^7.12.10", + "@babel/types": "^7.12.10", "convert-source-map": "^1.7.0", "debug": "^4.1.0", "gensync": "^1.0.0-beta.1", "json5": "^2.1.2", "lodash": "^4.17.19", - "resolve": "^1.3.2", "semver": "^5.4.1", "source-map": "^0.5.0" }, @@ -817,23 +17759,21 @@ "semver": { "version": "5.7.1", "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true + "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==" }, "source-map": { "version": "0.5.7", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", - "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", - "dev": true + "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=" } } }, "@babel/generator": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.11.0.tgz", - "integrity": "sha512-fEm3Uzw7Mc9Xi//qU20cBKatTfs2aOtKqmvy/Vm7RkJEGFQ4xc9myCfbXxqK//ZS8MR/ciOHw6meGASJuKmDfQ==", + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.11.tgz", + "integrity": "sha512-Ggg6WPOJtSi8yYQvLVjG8F/TlpWDlKx0OpS4Kt+xMQPs5OaGYWy+v1A+1TvxI6sAMGZpKWWoAQ1DaeQbImlItA==", "requires": { - "@babel/types": "^7.11.0", + "@babel/types": "^7.12.11", "jsesc": "^2.5.1", "source-map": "^0.5.0" }, @@ -846,586 +17786,376 @@ } }, "@babel/helper-annotate-as-pure": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.10.4.tgz", - "integrity": "sha512-XQlqKQP4vXFB7BN8fEEerrmYvHp3fK/rBkRFz9jaJbzK0B1DSfej9Kc7ZzE8Z/OnId1jpJdNAZ3BFQjWG68rcA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.14.5.tgz", + "integrity": "sha512-EivH9EgBIb+G8ij1B2jAwSH36WnGvkQSEC6CkX/6v6ZFlw5fVOHvsgGF4uiEHO2GzMvunZb6tDLQEQSdrdocrA==", "dev": true, "requires": { - "@babel/types": "^7.10.4" + "@babel/types": "^7.14.5" } }, "@babel/helper-builder-binary-assignment-operator-visitor": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.10.4.tgz", - "integrity": "sha512-L0zGlFrGWZK4PbT8AszSfLTM5sDU1+Az/En9VrdT8/LmEiJt4zXt+Jve9DCAnQcbqDhCI+29y/L93mrDzddCcg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-builder-binary-assignment-operator-visitor/-/helper-builder-binary-assignment-operator-visitor-7.14.5.tgz", + "integrity": "sha512-YTA/Twn0vBXDVGJuAX6PwW7x5zQei1luDDo2Pl6q1qZ7hVNl0RZrhHCQG/ArGpR29Vl7ETiB8eJyrvpuRp300w==", "dev": true, "requires": { - "@babel/helper-explode-assignable-expression": "^7.10.4", - "@babel/types": "^7.10.4" + "@babel/helper-explode-assignable-expression": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helper-compilation-targets": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.12.1.tgz", - "integrity": "sha512-jtBEif7jsPwP27GPHs06v4WBV0KrE8a/P7n0N0sSvHn2hwUCYnolP/CLmz51IzAW4NlN+HuoBtb9QcwnRo9F/g==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-compilation-targets/-/helper-compilation-targets-7.14.5.tgz", + "integrity": "sha512-v+QtZqXEiOnpO6EYvlImB6zCD2Lel06RzOPzmkz/D/XgQiUu3C/Jb1LOqSt/AIA34TYi/Q+KlT8vTQrgdxkbLw==", "dev": true, "requires": { - "@babel/compat-data": "^7.12.1", - "@babel/helper-validator-option": "^7.12.1", - "browserslist": "^4.12.0", - "semver": "^5.5.0" + "@babel/compat-data": "^7.14.5", + "@babel/helper-validator-option": "^7.14.5", + "browserslist": "^4.16.6", + "semver": "^6.3.0" }, "dependencies": { "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", "dev": true } } }, "@babel/helper-create-class-features-plugin": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.12.1.tgz", - "integrity": "sha512-hkL++rWeta/OVOBTRJc9a5Azh5mt5WgZUGAKMD8JM141YsE08K//bp1unBBieO6rUKkIPyUE0USQ30jAy3Sk1w==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.14.5.tgz", + "integrity": "sha512-Uq9z2e7ZtcnDMirRqAGLRaLwJn+Lrh388v5ETrR3pALJnElVh2zqQmdbz4W2RUJYohAPh2mtyPUgyMHMzXMncQ==", "dev": true, "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-member-expression-to-functions": "^7.12.1", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.10.4" + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-member-expression-to-functions": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5" } }, "@babel/helper-create-regexp-features-plugin": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.12.1.tgz", - "integrity": "sha512-rsZ4LGvFTZnzdNZR5HZdmJVuXK8834R5QkF3WvcnBhrlVtF0HSIUC6zbreL9MgjTywhKokn8RIYRiq99+DLAxA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.14.5.tgz", + "integrity": "sha512-TLawwqpOErY2HhWbGJ2nZT5wSkR192QpN+nBg1THfBfftrlvOh+WbhrxXCH4q4xJ9Gl16BGPR/48JA+Ryiho/A==", "dev": true, "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-regex": "^7.10.4", + "@babel/helper-annotate-as-pure": "^7.14.5", "regexpu-core": "^4.7.1" } }, - "@babel/helper-define-map": { - "version": "7.10.5", - "resolved": "https://registry.npmjs.org/@babel/helper-define-map/-/helper-define-map-7.10.5.tgz", - "integrity": "sha512-fMw4kgFB720aQFXSVaXr79pjjcW5puTCM16+rECJ/plGS+zByelE8l9nCpV1GibxTnFVmUuYG9U8wYfQHdzOEQ==", - "dev": true, - "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/types": "^7.10.5", - "lodash": "^4.17.19" - } - }, "@babel/helper-explode-assignable-expression": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.12.1.tgz", - "integrity": "sha512-dmUwH8XmlrUpVqgtZ737tK88v07l840z9j3OEhCLwKTkjlvKpfqXVIZ0wpK3aeOxspwGrf/5AP5qLx4rO3w5rA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-explode-assignable-expression/-/helper-explode-assignable-expression-7.14.5.tgz", + "integrity": "sha512-Htb24gnGJdIGT4vnRKMdoXiOIlqOLmdiUYpAQ0mYfgVT/GDm8GOYhgi4GL+hMKrkiPRohO4ts34ELFsGAPQLDQ==", "dev": true, "requires": { - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/types": "^7.14.5" } }, "@babel/helper-function-name": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.10.4.tgz", - "integrity": "sha512-YdaSyz1n8gY44EmN7x44zBn9zQ1Ry2Y+3GTA+3vH6Mizke1Vw0aWDM66FOYEPw8//qKkmqOckrGgTYa+6sceqQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-function-name/-/helper-function-name-7.14.5.tgz", + "integrity": "sha512-Gjna0AsXWfFvrAuX+VKcN/aNNWonizBj39yGwUzVDVTlMYJMK2Wp6xdpy72mfArFq5uK+NOuexfzZlzI1z9+AQ==", "requires": { - "@babel/helper-get-function-arity": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/types": "^7.10.4" + "@babel/helper-get-function-arity": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helper-get-function-arity": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.10.4.tgz", - "integrity": "sha512-EkN3YDB+SRDgiIUnNgcmiD361ti+AVbL3f3Henf6dqqUyr5dMsorno0lJWJuLhDhkI5sYEpgj6y9kB8AOU1I2A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-get-function-arity/-/helper-get-function-arity-7.14.5.tgz", + "integrity": "sha512-I1Db4Shst5lewOM4V+ZKJzQ0JGGaZ6VY1jYvMghRjqs6DWgxLCIyFt30GlnKkfUeFLpJt2vzbMVEXVSXlIFYUg==", "requires": { - "@babel/types": "^7.10.4" + "@babel/types": "^7.14.5" } }, "@babel/helper-hoist-variables": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.10.4.tgz", - "integrity": "sha512-wljroF5PgCk2juF69kanHVs6vrLwIPNp6DLD+Lrl3hoQ3PpPPikaDRNFA+0t81NOoMt2DL6WW/mdU8k4k6ZzuA==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-hoist-variables/-/helper-hoist-variables-7.14.5.tgz", + "integrity": "sha512-R1PXiz31Uc0Vxy4OEOm07x0oSjKAdPPCh3tPivn/Eo8cvz6gveAeuyUUPB21Hoiif0uoPQSSdhIPS3352nvdyQ==", "requires": { - "@babel/types": "^7.10.4" + "@babel/types": "^7.14.5" } }, "@babel/helper-member-expression-to-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.12.1.tgz", - "integrity": "sha512-k0CIe3tXUKTRSoEx1LQEPFU9vRQfqHtl+kf8eNnDqb4AUJEy5pz6aIiog+YWtVm2jpggjS1laH68bPsR+KWWPQ==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.14.5.tgz", + "integrity": "sha512-UxUeEYPrqH1Q/k0yRku1JE7dyfyehNwT6SVkMHvYvPDv4+uu627VXBckVj891BO8ruKBkiDoGnZf4qPDD8abDQ==", "requires": { - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/types": "^7.14.5" } }, "@babel/helper-module-imports": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.12.1.tgz", - "integrity": "sha512-ZeC1TlMSvikvJNy1v/wPIazCu3NdOwgYZLIkmIyAsGhqkNpiDoQQRmaCK8YP4Pq3GPTLPV9WXaPCJKvx06JxKA==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-imports/-/helper-module-imports-7.14.5.tgz", + "integrity": "sha512-SwrNHu5QWS84XlHwGYPDtCxcA0hrSlL2yhWYLgeOc0w7ccOl2qv4s/nARI0aYZW+bSwAL5CukeXA47B/1NKcnQ==", "requires": { - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/types": "^7.14.5" } }, "@babel/helper-module-transforms": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.12.1.tgz", - "integrity": "sha512-QQzehgFAZ2bbISiCpmVGfiGux8YVFXQ0abBic2Envhej22DVXV9nCFaS5hIQbkyo1AdGb+gNME2TSh3hYJVV/w==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-module-transforms/-/helper-module-transforms-7.14.5.tgz", + "integrity": "sha512-iXpX4KW8LVODuAieD7MzhNjmM6dzYY5tfRqT+R9HDXWl0jPn/djKmA+G9s/2C2T9zggw5tK1QNqZ70USfedOwA==", "requires": { - "@babel/helper-module-imports": "^7.12.1", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-simple-access": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/helper-validator-identifier": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.12.1", - "@babel/types": "^7.12.1", - "lodash": "^4.17.19" - }, - "dependencies": { - "@babel/generator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.1.tgz", - "integrity": "sha512-DB+6rafIdc9o72Yc3/Ph5h+6hUjeOp66pF0naQBgUFFuPqzQwIlPTm3xZR7YNvduIMtkDIj2t21LSQwnbCrXvg==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1", - "jsesc": "^2.5.1", - "source-map": "^0.5.0" - } - }, - "@babel/parser": { - "version": "7.12.3", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.12.3.tgz", - "integrity": "sha512-kFsOS0IbsuhO5ojF8Hc8z/8vEIOkylVBrjiZUbLTE3XFe0Qi+uu6HjzQixkFaqr0ZPAMZcBVxEwmsnsLPZ2Xsw==", - "dev": true - }, - "@babel/traverse": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.12.1.tgz", - "integrity": "sha512-MA3WPoRt1ZHo2ZmoGKNqi20YnPt0B1S0GTZEPhhd+hw2KGUzBlHuVunj6K4sNuK+reEvyiPwtp0cpaqLzJDmAw==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.12.1", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/parser": "^7.12.1", - "@babel/types": "^7.12.1", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.19" - } - }, - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - }, - "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-module-imports": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-simple-access": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", + "@babel/helper-validator-identifier": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helper-optimise-call-expression": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.10.4.tgz", - "integrity": "sha512-n3UGKY4VXwXThEiKrgRAoVPBMqeoPgHVqiHZOanAJCG9nQUL2pLRQirUzl0ioKclHGpGqRgIOkgcIJaIWLpygg==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.14.5.tgz", + "integrity": "sha512-IqiLIrODUOdnPU9/F8ib1Fx2ohlgDhxnIDU7OEVi+kAbEZcyiF7BLU8W6PfvPi9LzztjS7kcbzbmL7oG8kD6VA==", "requires": { - "@babel/types": "^7.10.4" + "@babel/types": "^7.14.5" } }, "@babel/helper-plugin-utils": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.10.4.tgz", - "integrity": "sha512-O4KCvQA6lLiMU9l2eawBPMf1xPP8xPfB3iEQw150hOVTqj/rfXz0ThTb4HEzqQfs2Bmo5Ay8BzxfzVtBrr9dVg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-plugin-utils/-/helper-plugin-utils-7.14.5.tgz", + "integrity": "sha512-/37qQCE3K0vvZKwoK4XU/irIJQdIfCJuhU5eKnNxpFDsOkgFaUAwbv+RYw6eYgsC0E4hS7r5KqGULUogqui0fQ==", "dev": true }, - "@babel/helper-regex": { - "version": "7.10.5", - "resolved": "https://registry.npmjs.org/@babel/helper-regex/-/helper-regex-7.10.5.tgz", - "integrity": "sha512-68kdUAzDrljqBrio7DYAEgCoJHxppJOERHOgOrDN7WjOzP0ZQ1LsSDRXcemzVZaLvjaJsJEESb6qt+znNuENDg==", - "dev": true, - "requires": { - "lodash": "^4.17.19" - } - }, "@babel/helper-remap-async-to-generator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.12.1.tgz", - "integrity": "sha512-9d0KQCRM8clMPcDwo8SevNs+/9a8yWVVmaE80FGJcEP8N1qToREmWEGnBn8BUlJhYRFz6fqxeRL1sl5Ogsed7A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.14.5.tgz", + "integrity": "sha512-rLQKdQU+HYlxBwQIj8dk4/0ENOUEhA/Z0l4hN8BexpvmSMN9oA9EagjnhnDpNsRdWCfjwa4mn/HyBXO9yhQP6A==", "dev": true, "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-wrap-function": "^7.10.4", - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-wrap-function": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helper-replace-supers": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.12.1.tgz", - "integrity": "sha512-zJjTvtNJnCFsCXVi5rUInstLd/EIVNmIKA1Q9ynESmMBWPWd+7sdR+G4/wdu+Mppfep0XLyG2m7EBPvjCeFyrw==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-replace-supers/-/helper-replace-supers-7.14.5.tgz", + "integrity": "sha512-3i1Qe9/8x/hCHINujn+iuHy+mMRLoc77b2nI9TB0zjH1hvn9qGlXjWlggdwUcju36PkPCy/lpM7LLUdcTyH4Ow==", "requires": { - "@babel/helper-member-expression-to-functions": "^7.12.1", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/traverse": "^7.12.1", - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/generator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.12.1.tgz", - "integrity": "sha512-DB+6rafIdc9o72Yc3/Ph5h+6hUjeOp66pF0naQBgUFFuPqzQwIlPTm3xZR7YNvduIMtkDIj2t21LSQwnbCrXvg==", - "dev": true, - "requires": { - "@babel/types": "^7.12.1", - "jsesc": "^2.5.1", - "source-map": "^0.5.0" - } - }, - "@babel/parser": { - "version": "7.12.3", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.12.3.tgz", - "integrity": "sha512-kFsOS0IbsuhO5ojF8Hc8z/8vEIOkylVBrjiZUbLTE3XFe0Qi+uu6HjzQixkFaqr0ZPAMZcBVxEwmsnsLPZ2Xsw==", - "dev": true - }, - "@babel/traverse": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.12.1.tgz", - "integrity": "sha512-MA3WPoRt1ZHo2ZmoGKNqi20YnPt0B1S0GTZEPhhd+hw2KGUzBlHuVunj6K4sNuK+reEvyiPwtp0cpaqLzJDmAw==", - "dev": true, - "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.12.1", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/parser": "^7.12.1", - "@babel/types": "^7.12.1", - "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.19" - } - }, - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - }, - "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-member-expression-to-functions": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helper-simple-access": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.12.1.tgz", - "integrity": "sha512-OxBp7pMrjVewSSC8fXDFrHrBcJATOOFssZwv16F3/6Xtc138GHybBfPbm9kfiqQHKhYQrlamWILwlDCeyMFEaA==", - "dev": true, + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-simple-access/-/helper-simple-access-7.14.5.tgz", + "integrity": "sha512-nfBN9xvmCt6nrMZjfhkl7i0oTV3yxR4/FztsbOASyTvVcoYd0TRHh7eMLdlEcCqobydC0LAF3LtC92Iwxo0wyw==", "requires": { - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/types": "^7.14.5" } }, "@babel/helper-skip-transparent-expression-wrappers": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.12.1.tgz", - "integrity": "sha512-Mf5AUuhG1/OCChOJ/HcADmvcHM42WJockombn8ATJG3OnyiSxBK/Mm5x78BQWvmtXZKHgbjdGL2kin/HOLlZGA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.14.5.tgz", + "integrity": "sha512-dmqZB7mrb94PZSAOYtr+ZN5qt5owZIAgqtoTuqiFbHFtxgEcmQlRJVI+bO++fciBunXtB6MK7HrzrfcAzIz2NQ==", "dev": true, "requires": { - "@babel/types": "^7.12.1" - }, - "dependencies": { - "@babel/types": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.12.1.tgz", - "integrity": "sha512-BzSY3NJBKM4kyatSOWh3D/JJ2O3CVzBybHWxtgxnggaxEuaSTTDqeiSb/xk9lrkw2Tbqyivw5ZU4rT+EfznQsA==", - "dev": true, - "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", - "to-fast-properties": "^2.0.0" - } - } + "@babel/types": "^7.14.5" } }, "@babel/helper-split-export-declaration": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.11.0.tgz", - "integrity": "sha512-74Vejvp6mHkGE+m+k5vHY93FX2cAtrw1zXrZXRlG4l410Nm9PxfEiVTn1PjDPV5SnmieiueY4AFg2xqhNFuuZg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-split-export-declaration/-/helper-split-export-declaration-7.14.5.tgz", + "integrity": "sha512-hprxVPu6e5Kdp2puZUmvOGjaLv9TCe58E/Fl6hRq4YiVQxIcNvuq6uTM2r1mT/oPskuS9CgR+I94sqAYv0NGKA==", "requires": { - "@babel/types": "^7.11.0" + "@babel/types": "^7.14.5" } }, "@babel/helper-validator-identifier": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz", - "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==" + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.14.5.tgz", + "integrity": "sha512-5lsetuxCLilmVGyiLEfoHBRX8UCFD+1m2x3Rj97WrW3V7H3u4RWRXA4evMjImCsin2J2YT0QaVDGf+z8ondbAg==" }, "@babel/helper-validator-option": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.12.1.tgz", - "integrity": "sha512-YpJabsXlJVWP0USHjnC/AQDTLlZERbON577YUVO/wLpqyj6HAtVYnWaQaN0iUN+1/tWn3c+uKKXjRut5115Y2A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-validator-option/-/helper-validator-option-7.14.5.tgz", + "integrity": "sha512-OX8D5eeX4XwcroVW45NMvoYaIuFI+GQpA2a8Gi+X/U/cDUIRsV37qQfF905F0htTRCREQIB4KqPeaveRJUl3Ow==", "dev": true }, "@babel/helper-wrap-function": { - "version": "7.12.3", - "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.12.3.tgz", - "integrity": "sha512-Cvb8IuJDln3rs6tzjW3Y8UeelAOdnpB8xtQ4sme2MSZ9wOxrbThporC0y/EtE16VAtoyEfLM404Xr1e0OOp+ow==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helper-wrap-function/-/helper-wrap-function-7.14.5.tgz", + "integrity": "sha512-YEdjTCq+LNuNS1WfxsDCNpgXkJaIyqco6DAelTUjT4f2KIWC1nBcaCaSdHTBqQVLnTBexBcVcFhLSU1KnYuePQ==", "dev": true, "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.10.4", - "@babel/types": "^7.10.4" + "@babel/helper-function-name": "^7.14.5", + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/helpers": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.10.4.tgz", - "integrity": "sha512-L2gX/XeUONeEbI78dXSrJzGdz4GQ+ZTA/aazfUsFaWjSe95kiCuOZ5HsXvkiw3iwF+mFHSRUfJU8t6YavocdXA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/helpers/-/helpers-7.14.5.tgz", + "integrity": "sha512-xtcWOuN9VL6nApgVHtq3PPcQv5qFBJzoSZzJ/2c0QK/IP/gxVcoWSNQwFEGvmbQsuS9rhYqjILDGGXcTkA705Q==", "requires": { - "@babel/template": "^7.10.4", - "@babel/traverse": "^7.10.4", - "@babel/types": "^7.10.4" + "@babel/template": "^7.14.5", + "@babel/traverse": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/highlight": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz", - "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.14.5.tgz", + "integrity": "sha512-qf9u2WFWVV0MppaL877j2dBtQIDgmidgjGk5VIMw3OadXvYaXn66U1BFlH2t4+t3i+8PhedppRv+i40ABzd+gg==", "requires": { - "@babel/helper-validator-identifier": "^7.10.4", + "@babel/helper-validator-identifier": "^7.14.5", "chalk": "^2.0.0", "js-tokens": "^4.0.0" } }, "@babel/parser": { - "version": "7.11.5", - "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.11.5.tgz", - "integrity": "sha512-X9rD8qqm695vgmeaQ4fvz/o3+Wk4ZzQvSHkDBgpYKxpD4qTAUm88ZKtHkVqIOsYFFbIQ6wQYhC6q7pjqVK0E0Q==" + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/parser/-/parser-7.14.5.tgz", + "integrity": "sha512-TM8C+xtH/9n1qzX+JNHi7AN2zHMTiPUtspO0ZdHflW8KaskkALhMmuMHb4bCmNdv9VAPzJX3/bXqkVLnAvsPfg==" }, "@babel/plugin-proposal-async-generator-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.12.1.tgz", - "integrity": "sha512-d+/o30tJxFxrA1lhzJqiUcEJdI6jKlNregCv5bASeGf2Q4MXmnwH7viDo7nhx1/ohf09oaH8j1GVYG/e3Yqk6A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-async-generator-functions/-/plugin-proposal-async-generator-functions-7.14.5.tgz", + "integrity": "sha512-tbD/CG3l43FIXxmu4a7RBe4zH7MLJ+S/lFowPFO7HetS2hyOZ/0nnnznegDuzFzfkyQYTxqdTH/hKmuBngaDAA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-remap-async-to-generator": "^7.12.1", - "@babel/plugin-syntax-async-generators": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-remap-async-to-generator": "^7.14.5", + "@babel/plugin-syntax-async-generators": "^7.8.4" } }, "@babel/plugin-proposal-class-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.12.1.tgz", - "integrity": "sha512-cKp3dlQsFsEs5CWKnN7BnSHOd0EOW8EKpEjkoz1pO2E5KzIDNV9Ros1b0CnmbVgAGXJubOYVBOGCT1OmJwOI7w==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-class-properties/-/plugin-proposal-class-properties-7.14.5.tgz", + "integrity": "sha512-q/PLpv5Ko4dVc1LYMpCY7RVAAO4uk55qPwrIuJ5QJ8c6cVuAmhu7I/49JOppXL6gXf7ZHzpRVEUZdYoPLM04Gg==", "dev": true, "requires": { - "@babel/helper-create-class-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-create-class-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-proposal-dynamic-import": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.12.1.tgz", - "integrity": "sha512-a4rhUSZFuq5W8/OO8H7BL5zspjnc1FLd9hlOxIK/f7qG4a0qsqk8uvF/ywgBA8/OmjsapjpvaEOYItfGG1qIvQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-dynamic-import/-/plugin-proposal-dynamic-import-7.14.5.tgz", + "integrity": "sha512-ExjiNYc3HDN5PXJx+bwC50GIx/KKanX2HiggnIUAYedbARdImiCU4RhhHfdf0Kd7JNXGpsBBBCOm+bBVy3Gb0g==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-dynamic-import": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-dynamic-import": "^7.8.3" } }, "@babel/plugin-proposal-export-namespace-from": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.12.1.tgz", - "integrity": "sha512-6CThGf0irEkzujYS5LQcjBx8j/4aQGiVv7J9+2f7pGfxqyKh3WnmVJYW3hdrQjyksErMGBPQrCnHfOtna+WLbw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-export-namespace-from/-/plugin-proposal-export-namespace-from-7.14.5.tgz", + "integrity": "sha512-g5POA32bXPMmSBu5Dx/iZGLGnKmKPc5AiY7qfZgurzrCYgIztDlHFbznSNCoQuv57YQLnQfaDi7dxCtLDIdXdA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-plugin-utils": "^7.14.5", "@babel/plugin-syntax-export-namespace-from": "^7.8.3" } }, "@babel/plugin-proposal-json-strings": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.12.1.tgz", - "integrity": "sha512-GoLDUi6U9ZLzlSda2Df++VSqDJg3CG+dR0+iWsv6XRw1rEq+zwt4DirM9yrxW6XWaTpmai1cWJLMfM8qQJf+yw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-json-strings/-/plugin-proposal-json-strings-7.14.5.tgz", + "integrity": "sha512-NSq2fczJYKVRIsUJyNxrVUMhB27zb7N7pOFGQOhBKJrChbGcgEAqyZrmZswkPk18VMurEeJAaICbfm57vUeTbQ==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-json-strings": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-json-strings": "^7.8.3" } }, "@babel/plugin-proposal-logical-assignment-operators": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.12.1.tgz", - "integrity": "sha512-k8ZmVv0JU+4gcUGeCDZOGd0lCIamU/sMtIiX3UWnUc5yzgq6YUGyEolNYD+MLYKfSzgECPcqetVcJP9Afe/aCA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-logical-assignment-operators/-/plugin-proposal-logical-assignment-operators-7.14.5.tgz", + "integrity": "sha512-YGn2AvZAo9TwyhlLvCCWxD90Xq8xJ4aSgaX3G5D/8DW94L8aaT+dS5cSP+Z06+rCJERGSr9GxMBZ601xoc2taw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-plugin-utils": "^7.14.5", "@babel/plugin-syntax-logical-assignment-operators": "^7.10.4" } }, "@babel/plugin-proposal-nullish-coalescing-operator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.12.1.tgz", - "integrity": "sha512-nZY0ESiaQDI1y96+jk6VxMOaL4LPo/QDHBqL+SF3/vl6dHkTwHlOI8L4ZwuRBHgakRBw5zsVylel7QPbbGuYgg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-nullish-coalescing-operator/-/plugin-proposal-nullish-coalescing-operator-7.14.5.tgz", + "integrity": "sha512-gun/SOnMqjSb98Nkaq2rTKMwervfdAoz6NphdY0vTfuzMfryj+tDGb2n6UkDKwez+Y8PZDhE3D143v6Gepp4Hg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-nullish-coalescing-operator": "^7.8.3" } }, "@babel/plugin-proposal-numeric-separator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.12.1.tgz", - "integrity": "sha512-MR7Ok+Af3OhNTCxYVjJZHS0t97ydnJZt/DbR4WISO39iDnhiD8XHrY12xuSJ90FFEGjir0Fzyyn7g/zY6hxbxA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-numeric-separator/-/plugin-proposal-numeric-separator-7.14.5.tgz", + "integrity": "sha512-yiclALKe0vyZRZE0pS6RXgjUOt87GWv6FYa5zqj15PvhOGFO69R5DusPlgK/1K5dVnCtegTiWu9UaBSrLLJJBg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-plugin-utils": "^7.14.5", "@babel/plugin-syntax-numeric-separator": "^7.10.4" } }, "@babel/plugin-proposal-object-rest-spread": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.12.1.tgz", - "integrity": "sha512-s6SowJIjzlhx8o7lsFx5zmY4At6CTtDvgNQDdPzkBQucle58A6b/TTeEBYtyDgmcXjUTM+vE8YOGHZzzbc/ioA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-object-rest-spread/-/plugin-proposal-object-rest-spread-7.14.5.tgz", + "integrity": "sha512-VzMyY6PWNPPT3pxc5hi9LloKNr4SSrVCg7Yr6aZpW4Ym07r7KqSU/QXYwjXLVxqwSv0t/XSXkFoKBPUkZ8vb2A==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-object-rest-spread": "^7.8.0", - "@babel/plugin-transform-parameters": "^7.12.1" + "@babel/compat-data": "^7.14.5", + "@babel/helper-compilation-targets": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-object-rest-spread": "^7.8.3", + "@babel/plugin-transform-parameters": "^7.14.5" } }, "@babel/plugin-proposal-optional-catch-binding": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.12.1.tgz", - "integrity": "sha512-hFvIjgprh9mMw5v42sJWLI1lzU5L2sznP805zeT6rySVRA0Y18StRhDqhSxlap0oVgItRsB6WSROp4YnJTJz0g==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-catch-binding/-/plugin-proposal-optional-catch-binding-7.14.5.tgz", + "integrity": "sha512-3Oyiixm0ur7bzO5ybNcZFlmVsygSIQgdOa7cTfOYCMY+wEPAYhZAJxi3mixKFCTCKUhQXuCTtQ1MzrpL3WT8ZQ==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-syntax-optional-catch-binding": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/plugin-syntax-optional-catch-binding": "^7.8.3" } }, "@babel/plugin-proposal-optional-chaining": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.12.1.tgz", - "integrity": "sha512-c2uRpY6WzaVDzynVY9liyykS+kVU+WRZPMPYpkelXH8KBt1oXoI89kPbZKKG/jDT5UK92FTW2fZkZaJhdiBabw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-optional-chaining/-/plugin-proposal-optional-chaining-7.14.5.tgz", + "integrity": "sha512-ycz+VOzo2UbWNI1rQXxIuMOzrDdHGrI23fRiz/Si2R4kv2XZQ1BK8ccdHwehMKBlcH/joGW/tzrUmo67gbJHlQ==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1", - "@babel/plugin-syntax-optional-chaining": "^7.8.0" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.14.5", + "@babel/plugin-syntax-optional-chaining": "^7.8.3" } }, "@babel/plugin-proposal-private-methods": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.12.1.tgz", - "integrity": "sha512-mwZ1phvH7/NHK6Kf8LP7MYDogGV+DKB1mryFOEwx5EBNQrosvIczzZFTUmWaeujd5xT6G1ELYWUz3CutMhjE1w==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-private-methods/-/plugin-proposal-private-methods-7.14.5.tgz", + "integrity": "sha512-838DkdUA1u+QTCplatfq4B7+1lnDa/+QMI89x5WZHBcnNv+47N8QEj2k9I2MUU9xIv8XJ4XvPCviM/Dj7Uwt9g==", "dev": true, "requires": { - "@babel/helper-create-class-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-create-class-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-proposal-unicode-property-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.12.1.tgz", - "integrity": "sha512-MYq+l+PvHuw/rKUz1at/vb6nCnQ2gmJBNaM62z0OgH7B2W1D9pvkpYtlti9bGtizNIU1K3zm4bZF9F91efVY0w==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-proposal-unicode-property-regex/-/plugin-proposal-unicode-property-regex-7.14.5.tgz", + "integrity": "sha512-6axIeOU5LnY471KenAB9vI8I5j7NQ2d652hIYwVyRfgaZT5UpiqFKCuVXCDMSrU+3VFafnu2c5m3lrWIlr6A5Q==", "dev": true, "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-syntax-async-generators": { @@ -1438,12 +18168,12 @@ } }, "@babel/plugin-syntax-class-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.1.tgz", - "integrity": "sha512-U40A76x5gTwmESz+qiqssqmeEsKvcSyvtgktrm0uzcARAmM9I1jR221f6Oq+GmHrcD+LvZDag1UTOTe2fL3TeA==", + "version": "7.12.13", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-class-properties/-/plugin-syntax-class-properties-7.12.13.tgz", + "integrity": "sha512-fm4idjKla0YahUNgFNLCB0qySdsoPiZP3iQE3rky0mBUtMZ23yDJ9SJdg6dXTSDnulOVqiF3Hgr9nbXvXTQZYA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.12.13" } }, "@babel/plugin-syntax-dynamic-import": { @@ -1528,21 +18258,21 @@ } }, "@babel/plugin-syntax-top-level-await": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.12.1.tgz", - "integrity": "sha512-i7ooMZFS+a/Om0crxZodrTzNEPJHZrlMVGMTEpFAj6rYY/bKCddB0Dk/YxfPuYXOopuhKk/e1jV6h+WUU9XN3A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-syntax-top-level-await/-/plugin-syntax-top-level-await-7.14.5.tgz", + "integrity": "sha512-hx++upLv5U1rgYfwe1xBQUhRmU41NEvpUvrp8jkrSCdvGSnM5/qdRMtylJ6PG5OFkBaHkbTAKTnd3/YyESRHFw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-arrow-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.12.1.tgz", - "integrity": "sha512-5QB50qyN44fzzz4/qxDPQMBCTHgxg3n0xRBLJUmBlLoU/sFvxVWGZF/ZUfMVDQuJUKXaBhbupxIzIfZ6Fwk/0A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.14.5.tgz", + "integrity": "sha512-KOnO0l4+tD5IfOdi4x8C1XmEIRWUjNRV8wc6K2vz/3e8yAOoZZvsRXRRIF/yo/MAOFb4QjtAw9xSxMXbSMRy8A==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-async-to-generator": { @@ -1557,242 +18287,240 @@ } }, "@babel/plugin-transform-block-scoped-functions": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.12.1.tgz", - "integrity": "sha512-5OpxfuYnSgPalRpo8EWGPzIYf0lHBWORCkj5M0oLBwHdlux9Ri36QqGW3/LR13RSVOAoUUMzoPI/jpE4ABcHoA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.14.5.tgz", + "integrity": "sha512-dtqWqdWZ5NqBX3KzsVCWfQI3A53Ft5pWFCT2eCVUftWZgjc5DpDponbIF1+c+7cSGk2wN0YK7HGL/ezfRbpKBQ==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-block-scoping": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.12.1.tgz", - "integrity": "sha512-zJyAC9sZdE60r1nVQHblcfCj29Dh2Y0DOvlMkcqSo0ckqjiCwNiUezUKw+RjOCwGfpLRwnAeQ2XlLpsnGkvv9w==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.14.5.tgz", + "integrity": "sha512-LBYm4ZocNgoCqyxMLoOnwpsmQ18HWTQvql64t3GvMUzLQrNoV1BDG0lNftC8QKYERkZgCCT/7J5xWGObGAyHDw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-classes": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.12.1.tgz", - "integrity": "sha512-/74xkA7bVdzQTBeSUhLLJgYIcxw/dpEpCdRDiHgPJ3Mv6uC11UhjpOhl72CgqbBCmt1qtssCyB2xnJm1+PFjog==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-classes/-/plugin-transform-classes-7.14.5.tgz", + "integrity": "sha512-J4VxKAMykM06K/64z9rwiL6xnBHgB1+FVspqvlgCdwD1KUbQNfszeKVVOMh59w3sztHYIZDgnhOC4WbdEfHFDA==", "dev": true, "requires": { - "@babel/helper-annotate-as-pure": "^7.10.4", - "@babel/helper-define-map": "^7.10.4", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-optimise-call-expression": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1", - "@babel/helper-split-export-declaration": "^7.10.4", + "@babel/helper-annotate-as-pure": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-optimise-call-expression": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", "globals": "^11.1.0" } }, "@babel/plugin-transform-computed-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.12.1.tgz", - "integrity": "sha512-vVUOYpPWB7BkgUWPo4C44mUQHpTZXakEqFjbv8rQMg7TC6S6ZhGZ3otQcRH6u7+adSlE5i0sp63eMC/XGffrzg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.14.5.tgz", + "integrity": "sha512-pWM+E4283UxaVzLb8UBXv4EIxMovU4zxT1OPnpHJcmnvyY9QbPPTKZfEj31EUvG3/EQRbYAGaYEUZ4yWOBC2xg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-destructuring": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.12.1.tgz", - "integrity": "sha512-fRMYFKuzi/rSiYb2uRLiUENJOKq4Gnl+6qOv5f8z0TZXg3llUwUhsNNwrwaT/6dUhJTzNpBr+CUvEWBtfNY1cw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.14.5.tgz", + "integrity": "sha512-wU9tYisEbRMxqDezKUqC9GleLycCRoUsai9ddlsq54r8QRLaeEhc+d+9DqCG+kV9W2GgQjTZESPTpn5bAFMDww==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-dotall-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.12.1.tgz", - "integrity": "sha512-B2pXeRKoLszfEW7J4Hg9LoFaWEbr/kzo3teWHmtFCszjRNa/b40f9mfeqZsIDLLt/FjwQ6pz/Gdlwy85xNckBA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.14.5.tgz", + "integrity": "sha512-loGlnBdj02MDsFaHhAIJzh7euK89lBrGIdM9EAtHFo6xKygCUGuuWe07o1oZVk287amtW1n0808sQM99aZt3gw==", "dev": true, "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-duplicate-keys": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.12.1.tgz", - "integrity": "sha512-iRght0T0HztAb/CazveUpUQrZY+aGKKaWXMJ4uf9YJtqxSUe09j3wteztCUDRHs+SRAL7yMuFqUsLoAKKzgXjw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.14.5.tgz", + "integrity": "sha512-iJjbI53huKbPDAsJ8EmVmvCKeeq21bAze4fu9GBQtSLqfvzj2oRuHVx4ZkDwEhg1htQ+5OBZh/Ab0XDf5iBZ7A==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-exponentiation-operator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.12.1.tgz", - "integrity": "sha512-7tqwy2bv48q+c1EHbXK0Zx3KXd2RVQp6OC7PbwFNt/dPTAV3Lu5sWtWuAj8owr5wqtWnqHfl2/mJlUmqkChKug==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.14.5.tgz", + "integrity": "sha512-jFazJhMBc9D27o9jDnIE5ZErI0R0m7PbKXVq77FFvqFbzvTMuv8jaAwLZ5PviOLSFttqKIW0/wxNSDbjLk0tYA==", "dev": true, "requires": { - "@babel/helper-builder-binary-assignment-operator-visitor": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-builder-binary-assignment-operator-visitor": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-for-of": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.12.1.tgz", - "integrity": "sha512-Zaeq10naAsuHo7heQvyV0ptj4dlZJwZgNAtBYBnu5nNKJoW62m0zKcIEyVECrUKErkUkg6ajMy4ZfnVZciSBhg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.14.5.tgz", + "integrity": "sha512-CfmqxSUZzBl0rSjpoQSFoR9UEj3HzbGuGNL21/iFTmjb5gFggJp3ph0xR1YBhexmLoKRHzgxuFvty2xdSt6gTA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-function-name": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.12.1.tgz", - "integrity": "sha512-JF3UgJUILoFrFMEnOJLJkRHSk6LUSXLmEFsA23aR2O5CSLUxbeUX1IZ1YQ7Sn0aXb601Ncwjx73a+FVqgcljVw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.14.5.tgz", + "integrity": "sha512-vbO6kv0fIzZ1GpmGQuvbwwm+O4Cbm2NrPzwlup9+/3fdkuzo1YqOZcXw26+YUJB84Ja7j9yURWposEHLYwxUfQ==", "dev": true, "requires": { - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.12.1.tgz", - "integrity": "sha512-+PxVGA+2Ag6uGgL0A5f+9rklOnnMccwEBzwYFL3EUaKuiyVnUipyXncFcfjSkbimLrODoqki1U9XxZzTvfN7IQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-literals/-/plugin-transform-literals-7.14.5.tgz", + "integrity": "sha512-ql33+epql2F49bi8aHXxvLURHkxJbSmMKl9J5yHqg4PLtdE6Uc48CH1GS6TQvZ86eoB/ApZXwm7jlA+B3kra7A==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-member-expression-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.12.1.tgz", - "integrity": "sha512-1sxePl6z9ad0gFMB9KqmYofk34flq62aqMt9NqliS/7hPEpURUCMbyHXrMPlo282iY7nAvUB1aQd5mg79UD9Jg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.14.5.tgz", + "integrity": "sha512-WkNXxH1VXVTKarWFqmso83xl+2V3Eo28YY5utIkbsmXoItO8Q3aZxN4BTS2k0hz9dGUloHK26mJMyQEYfkn/+Q==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-modules-amd": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.12.1.tgz", - "integrity": "sha512-tDW8hMkzad5oDtzsB70HIQQRBiTKrhfgwC/KkJeGsaNFTdWhKNt/BiE8c5yj19XiGyrxpbkOfH87qkNg1YGlOQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.14.5.tgz", + "integrity": "sha512-3lpOU8Vxmp3roC4vzFpSdEpGUWSMsHFreTWOMMLzel2gNGfHE5UWIh/LN6ghHs2xurUp4jRFYMUIZhuFbody1g==", "dev": true, "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", "babel-plugin-dynamic-import-node": "^2.3.3" } }, "@babel/plugin-transform-modules-commonjs": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.12.1.tgz", - "integrity": "sha512-dY789wq6l0uLY8py9c1B48V8mVL5gZh/+PQ5ZPrylPYsnAvnEMjqsUXkuoDVPeVK+0VyGar+D08107LzDQ6pag==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.14.5.tgz", + "integrity": "sha512-en8GfBtgnydoao2PS+87mKyw62k02k7kJ9ltbKe0fXTHrQmG6QZZflYuGI1VVG7sVpx4E1n7KBpNlPb8m78J+A==", "dev": true, "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-simple-access": "^7.12.1", + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-simple-access": "^7.14.5", "babel-plugin-dynamic-import-node": "^2.3.3" } }, "@babel/plugin-transform-modules-systemjs": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.12.1.tgz", - "integrity": "sha512-Hn7cVvOavVh8yvW6fLwveFqSnd7rbQN3zJvoPNyNaQSvgfKmDBO9U1YL9+PCXGRlZD9tNdWTy5ACKqMuzyn32Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.14.5.tgz", + "integrity": "sha512-mNMQdvBEE5DcMQaL5LbzXFMANrQjd2W7FPzg34Y4yEz7dBgdaC+9B84dSO+/1Wba98zoDbInctCDo4JGxz1VYA==", "dev": true, "requires": { - "@babel/helper-hoist-variables": "^7.10.4", - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-validator-identifier": "^7.10.4", + "@babel/helper-hoist-variables": "^7.14.5", + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-validator-identifier": "^7.14.5", "babel-plugin-dynamic-import-node": "^2.3.3" } }, "@babel/plugin-transform-modules-umd": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.12.1.tgz", - "integrity": "sha512-aEIubCS0KHKM0zUos5fIoQm+AZUMt1ZvMpqz0/H5qAQ7vWylr9+PLYurT+Ic7ID/bKLd4q8hDovaG3Zch2uz5Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.14.5.tgz", + "integrity": "sha512-RfPGoagSngC06LsGUYyM9QWSXZ8MysEjDJTAea1lqRjNECE3y0qIJF/qbvJxc4oA4s99HumIMdXOrd+TdKaAAA==", "dev": true, "requires": { - "@babel/helper-module-transforms": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-module-transforms": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-named-capturing-groups-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.12.1.tgz", - "integrity": "sha512-tB43uQ62RHcoDp9v2Nsf+dSM8sbNodbEicbQNA53zHz8pWUhsgHSJCGpt7daXxRydjb0KnfmB+ChXOv3oADp1Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.14.5.tgz", + "integrity": "sha512-+Xe5+6MWFo311U8SchgeX5c1+lJM+eZDBZgD+tvXu9VVQPXwwVzeManMMjYX6xw2HczngfOSZjoFYKwdeB/Jvw==", "dev": true, "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1" + "@babel/helper-create-regexp-features-plugin": "^7.14.5" } }, "@babel/plugin-transform-new-target": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.12.1.tgz", - "integrity": "sha512-+eW/VLcUL5L9IvJH7rT1sT0CzkdUTvPrXC2PXTn/7z7tXLBuKvezYbGdxD5WMRoyvyaujOq2fWoKl869heKjhw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.14.5.tgz", + "integrity": "sha512-Nx054zovz6IIRWEB49RDRuXGI4Gy0GMgqG0cII9L3MxqgXz/+rgII+RU58qpo4g7tNEx1jG7rRVH4ihZoP4esQ==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-object-super": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.12.1.tgz", - "integrity": "sha512-AvypiGJH9hsquNUn+RXVcBdeE3KHPZexWRdimhuV59cSoOt5kFBmqlByorAeUlGG2CJWd0U+4ZtNKga/TB0cAw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.14.5.tgz", + "integrity": "sha512-MKfOBWzK0pZIrav9z/hkRqIk/2bTv9qvxHzPQc12RcVkMOzpIKnFCNYJip00ssKWYkd8Sf5g0Wr7pqJ+cmtuFg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-replace-supers": "^7.12.1" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-replace-supers": "^7.14.5" } }, "@babel/plugin-transform-parameters": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.12.1.tgz", - "integrity": "sha512-xq9C5EQhdPK23ZeCdMxl8bbRnAgHFrw5EOC3KJUsSylZqdkCaFEXxGSBuTSObOpiiHHNyb82es8M1QYgfQGfNg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.14.5.tgz", + "integrity": "sha512-Tl7LWdr6HUxTmzQtzuU14SqbgrSKmaR77M0OKyq4njZLQTPfOvzblNKyNkGwOfEFCEx7KeYHQHDI0P3F02IVkA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-property-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.12.1.tgz", - "integrity": "sha512-6MTCR/mZ1MQS+AwZLplX4cEySjCpnIF26ToWo942nqn8hXSm7McaHQNeGx/pt7suI1TWOWMfa/NgBhiqSnX0cQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.14.5.tgz", + "integrity": "sha512-r1uilDthkgXW8Z1vJz2dKYLV1tuw2xsbrp3MrZmD99Wh9vsfKoob+JTgri5VUb/JqyKRXotlOtwgu4stIYCmnw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-regenerator": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.12.1.tgz", - "integrity": "sha512-gYrHqs5itw6i4PflFX3OdBPMQdPbF4bj2REIUxlMRUFk0/ZOAIpDFuViuxPjUL7YC8UPnf+XG7/utJvqXdPKng==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.14.5.tgz", + "integrity": "sha512-NVIY1W3ITDP5xQl50NgTKlZ0GrotKtLna08/uGY6ErQt6VEQZXla86x/CTddm5gZdcr+5GSsvMeTmWA5Ii6pkg==", "dev": true, "requires": { "regenerator-transform": "^0.14.2" } }, "@babel/plugin-transform-reserved-words": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.12.1.tgz", - "integrity": "sha512-pOnUfhyPKvZpVyBHhSBoX8vfA09b7r00Pmm1sH+29ae2hMTKVmSp4Ztsr8KBKjLjx17H0eJqaRC3bR2iThM54A==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.14.5.tgz", + "integrity": "sha512-cv4F2rv1nD4qdexOGsRQXJrOcyb5CrgjUH9PKrrtyhSDBNWGxd0UIitjyJiWagS+EbUGjG++22mGH1Pub8D6Vg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-runtime": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.11.0.tgz", - "integrity": "sha512-LFEsP+t3wkYBlis8w6/kmnd6Kb1dxTd+wGJ8MlxTGzQo//ehtqlVL4S9DNUa53+dtPSQobN2CXx4d81FqC58cw==", + "version": "7.12.10", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-runtime/-/plugin-transform-runtime-7.12.10.tgz", + "integrity": "sha512-xOrUfzPxw7+WDm9igMgQCbO3cJKymX7dFdsgRr1eu9n3KjjyU4pptIXbXPseQDquw+W+RuJEJMHKHNsPNNm3CA==", "dev": true, "requires": { - "@babel/helper-module-imports": "^7.10.4", + "@babel/helper-module-imports": "^7.12.5", "@babel/helper-plugin-utils": "^7.10.4", - "resolve": "^1.8.1", "semver": "^5.5.1" }, "dependencies": { @@ -1805,96 +18533,96 @@ } }, "@babel/plugin-transform-shorthand-properties": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.12.1.tgz", - "integrity": "sha512-GFZS3c/MhX1OusqB1MZ1ct2xRzX5ppQh2JU1h2Pnfk88HtFTM+TWQqJNfwkmxtPQtb/s1tk87oENfXJlx7rSDw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.14.5.tgz", + "integrity": "sha512-xLucks6T1VmGsTB+GWK5Pl9Jl5+nRXD1uoFdA5TSO6xtiNjtXTjKkmPdFXVLGlK5A2/or/wQMKfmQ2Y0XJfn5g==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-spread": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.12.1.tgz", - "integrity": "sha512-vuLp8CP0BE18zVYjsEBZ5xoCecMK6LBMMxYzJnh01rxQRvhNhH1csMMmBfNo5tGpGO+NhdSNW2mzIvBu3K1fng==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-spread/-/plugin-transform-spread-7.14.5.tgz", + "integrity": "sha512-/3iqoQdiWergnShZYl0xACb4ADeYCJ7X/RgmwtXshn6cIvautRPAFzhd58frQlokLO6Jb4/3JXvmm6WNTPtiTw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-skip-transparent-expression-wrappers": "^7.12.1" + "@babel/helper-plugin-utils": "^7.14.5", + "@babel/helper-skip-transparent-expression-wrappers": "^7.14.5" } }, "@babel/plugin-transform-sticky-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.12.1.tgz", - "integrity": "sha512-CiUgKQ3AGVk7kveIaPEET1jNDhZZEl1RPMWdTBE1799bdz++SwqDHStmxfCtDfBhQgCl38YRiSnrMuUMZIWSUQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.14.5.tgz", + "integrity": "sha512-Z7F7GyvEMzIIbwnziAZmnSNpdijdr4dWt+FJNBnBLz5mwDFkqIXU9wmBcWWad3QeJF5hMTkRe4dAq2sUZiG+8A==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4", - "@babel/helper-regex": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-template-literals": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.12.1.tgz", - "integrity": "sha512-b4Zx3KHi+taXB1dVRBhVJtEPi9h1THCeKmae2qP0YdUHIFhVjtpqqNfxeVAa1xeHVhAy4SbHxEwx5cltAu5apw==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.14.5.tgz", + "integrity": "sha512-22btZeURqiepOfuy/VkFr+zStqlujWaarpMErvay7goJS6BWwdd6BY9zQyDLDa4x2S3VugxFb162IZ4m/S/+Gg==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-typeof-symbol": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.12.1.tgz", - "integrity": "sha512-EPGgpGy+O5Kg5pJFNDKuxt9RdmTgj5sgrus2XVeMp/ZIbOESadgILUbm50SNpghOh3/6yrbsH+NB5+WJTmsA7Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.14.5.tgz", + "integrity": "sha512-lXzLD30ffCWseTbMQzrvDWqljvZlHkXU+CnseMhkMNqU1sASnCsz3tSzAaH3vCUXb9PHeUb90ZT1BdFTm1xxJw==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-unicode-escapes": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.12.1.tgz", - "integrity": "sha512-I8gNHJLIc7GdApm7wkVnStWssPNbSRMPtgHdmH3sRM1zopz09UWPS4x5V4n1yz/MIWTVnJ9sp6IkuXdWM4w+2Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.14.5.tgz", + "integrity": "sha512-crTo4jATEOjxj7bt9lbYXcBAM3LZaUrbP2uUdxb6WIorLmjNKSpHfIybgY4B8SRpbf8tEVIWH3Vtm7ayCrKocA==", "dev": true, "requires": { - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/plugin-transform-unicode-regex": { - "version": "7.12.1", - "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.12.1.tgz", - "integrity": "sha512-SqH4ClNngh/zGwHZOOQMTD+e8FGWexILV+ePMyiDJttAWRh5dhDL8rcl5lSgU3Huiq6Zn6pWTMvdPAb21Dwdyg==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.14.5.tgz", + "integrity": "sha512-UygduJpC5kHeCiRw/xDVzC+wj8VaYSoKl5JNVmbP7MadpNinAm3SvZCxZ42H37KZBKztz46YC73i9yV34d0Tzw==", "dev": true, "requires": { - "@babel/helper-create-regexp-features-plugin": "^7.12.1", - "@babel/helper-plugin-utils": "^7.10.4" + "@babel/helper-create-regexp-features-plugin": "^7.14.5", + "@babel/helper-plugin-utils": "^7.14.5" } }, "@babel/preset-env": { - "version": "7.11.0", - "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.11.0.tgz", - "integrity": "sha512-2u1/k7rG/gTh02dylX2kL3S0IJNF+J6bfDSp4DI2Ma8QN6Y9x9pmAax59fsCk6QUQG0yqH47yJWA+u1I1LccAg==", + "version": "7.12.11", + "resolved": "https://registry.npmjs.org/@babel/preset-env/-/preset-env-7.12.11.tgz", + "integrity": "sha512-j8Tb+KKIXKYlDBQyIOy4BLxzv1NUOwlHfZ74rvW+Z0Gp4/cI2IMDPBWAgWceGcE7aep9oL/0K9mlzlMGxA8yNw==", "dev": true, "requires": { - "@babel/compat-data": "^7.11.0", - "@babel/helper-compilation-targets": "^7.10.4", - "@babel/helper-module-imports": "^7.10.4", + "@babel/compat-data": "^7.12.7", + "@babel/helper-compilation-targets": "^7.12.5", + "@babel/helper-module-imports": "^7.12.5", "@babel/helper-plugin-utils": "^7.10.4", - "@babel/plugin-proposal-async-generator-functions": "^7.10.4", - "@babel/plugin-proposal-class-properties": "^7.10.4", - "@babel/plugin-proposal-dynamic-import": "^7.10.4", - "@babel/plugin-proposal-export-namespace-from": "^7.10.4", - "@babel/plugin-proposal-json-strings": "^7.10.4", - "@babel/plugin-proposal-logical-assignment-operators": "^7.11.0", - "@babel/plugin-proposal-nullish-coalescing-operator": "^7.10.4", - "@babel/plugin-proposal-numeric-separator": "^7.10.4", - "@babel/plugin-proposal-object-rest-spread": "^7.11.0", - "@babel/plugin-proposal-optional-catch-binding": "^7.10.4", - "@babel/plugin-proposal-optional-chaining": "^7.11.0", - "@babel/plugin-proposal-private-methods": "^7.10.4", - "@babel/plugin-proposal-unicode-property-regex": "^7.10.4", + "@babel/helper-validator-option": "^7.12.11", + "@babel/plugin-proposal-async-generator-functions": "^7.12.1", + "@babel/plugin-proposal-class-properties": "^7.12.1", + "@babel/plugin-proposal-dynamic-import": "^7.12.1", + "@babel/plugin-proposal-export-namespace-from": "^7.12.1", + "@babel/plugin-proposal-json-strings": "^7.12.1", + "@babel/plugin-proposal-logical-assignment-operators": "^7.12.1", + "@babel/plugin-proposal-nullish-coalescing-operator": "^7.12.1", + "@babel/plugin-proposal-numeric-separator": "^7.12.7", + "@babel/plugin-proposal-object-rest-spread": "^7.12.1", + "@babel/plugin-proposal-optional-catch-binding": "^7.12.1", + "@babel/plugin-proposal-optional-chaining": "^7.12.7", + "@babel/plugin-proposal-private-methods": "^7.12.1", + "@babel/plugin-proposal-unicode-property-regex": "^7.12.1", "@babel/plugin-syntax-async-generators": "^7.8.0", - "@babel/plugin-syntax-class-properties": "^7.10.4", + "@babel/plugin-syntax-class-properties": "^7.12.1", "@babel/plugin-syntax-dynamic-import": "^7.8.0", "@babel/plugin-syntax-export-namespace-from": "^7.8.3", "@babel/plugin-syntax-json-strings": "^7.8.0", @@ -1904,45 +18632,42 @@ "@babel/plugin-syntax-object-rest-spread": "^7.8.0", "@babel/plugin-syntax-optional-catch-binding": "^7.8.0", "@babel/plugin-syntax-optional-chaining": "^7.8.0", - "@babel/plugin-syntax-top-level-await": "^7.10.4", - "@babel/plugin-transform-arrow-functions": "^7.10.4", - "@babel/plugin-transform-async-to-generator": "^7.10.4", - "@babel/plugin-transform-block-scoped-functions": "^7.10.4", - "@babel/plugin-transform-block-scoping": "^7.10.4", - "@babel/plugin-transform-classes": "^7.10.4", - "@babel/plugin-transform-computed-properties": "^7.10.4", - "@babel/plugin-transform-destructuring": "^7.10.4", - "@babel/plugin-transform-dotall-regex": "^7.10.4", - "@babel/plugin-transform-duplicate-keys": "^7.10.4", - "@babel/plugin-transform-exponentiation-operator": "^7.10.4", - "@babel/plugin-transform-for-of": "^7.10.4", - "@babel/plugin-transform-function-name": "^7.10.4", - "@babel/plugin-transform-literals": "^7.10.4", - "@babel/plugin-transform-member-expression-literals": "^7.10.4", - "@babel/plugin-transform-modules-amd": "^7.10.4", - "@babel/plugin-transform-modules-commonjs": "^7.10.4", - "@babel/plugin-transform-modules-systemjs": "^7.10.4", - "@babel/plugin-transform-modules-umd": "^7.10.4", - "@babel/plugin-transform-named-capturing-groups-regex": "^7.10.4", - "@babel/plugin-transform-new-target": "^7.10.4", - "@babel/plugin-transform-object-super": "^7.10.4", - "@babel/plugin-transform-parameters": "^7.10.4", - "@babel/plugin-transform-property-literals": "^7.10.4", - "@babel/plugin-transform-regenerator": "^7.10.4", - "@babel/plugin-transform-reserved-words": "^7.10.4", - "@babel/plugin-transform-shorthand-properties": "^7.10.4", - "@babel/plugin-transform-spread": "^7.11.0", - "@babel/plugin-transform-sticky-regex": "^7.10.4", - "@babel/plugin-transform-template-literals": "^7.10.4", - "@babel/plugin-transform-typeof-symbol": "^7.10.4", - "@babel/plugin-transform-unicode-escapes": "^7.10.4", - "@babel/plugin-transform-unicode-regex": "^7.10.4", + "@babel/plugin-syntax-top-level-await": "^7.12.1", + "@babel/plugin-transform-arrow-functions": "^7.12.1", + "@babel/plugin-transform-async-to-generator": "^7.12.1", + "@babel/plugin-transform-block-scoped-functions": "^7.12.1", + "@babel/plugin-transform-block-scoping": "^7.12.11", + "@babel/plugin-transform-classes": "^7.12.1", + "@babel/plugin-transform-computed-properties": "^7.12.1", + "@babel/plugin-transform-destructuring": "^7.12.1", + "@babel/plugin-transform-dotall-regex": "^7.12.1", + "@babel/plugin-transform-duplicate-keys": "^7.12.1", + "@babel/plugin-transform-exponentiation-operator": "^7.12.1", + "@babel/plugin-transform-for-of": "^7.12.1", + "@babel/plugin-transform-function-name": "^7.12.1", + "@babel/plugin-transform-literals": "^7.12.1", + "@babel/plugin-transform-member-expression-literals": "^7.12.1", + "@babel/plugin-transform-modules-amd": "^7.12.1", + "@babel/plugin-transform-modules-commonjs": "^7.12.1", + "@babel/plugin-transform-modules-systemjs": "^7.12.1", + "@babel/plugin-transform-modules-umd": "^7.12.1", + "@babel/plugin-transform-named-capturing-groups-regex": "^7.12.1", + "@babel/plugin-transform-new-target": "^7.12.1", + "@babel/plugin-transform-object-super": "^7.12.1", + "@babel/plugin-transform-parameters": "^7.12.1", + "@babel/plugin-transform-property-literals": "^7.12.1", + "@babel/plugin-transform-regenerator": "^7.12.1", + "@babel/plugin-transform-reserved-words": "^7.12.1", + "@babel/plugin-transform-shorthand-properties": "^7.12.1", + "@babel/plugin-transform-spread": "^7.12.1", + "@babel/plugin-transform-sticky-regex": "^7.12.7", + "@babel/plugin-transform-template-literals": "^7.12.1", + "@babel/plugin-transform-typeof-symbol": "^7.12.10", + "@babel/plugin-transform-unicode-escapes": "^7.12.1", + "@babel/plugin-transform-unicode-regex": "^7.12.1", "@babel/preset-modules": "^0.1.3", - "@babel/types": "^7.11.0", - "browserslist": "^4.12.0", - "core-js-compat": "^3.6.2", - "invariant": "^2.2.2", - "levenary": "^1.1.1", + "@babel/types": "^7.12.11", + "core-js-compat": "^3.8.0", "semver": "^5.5.0" }, "dependencies": { @@ -1968,46 +18693,46 @@ } }, "@babel/runtime": { - "version": "7.11.2", - "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.11.2.tgz", - "integrity": "sha512-TeWkU52so0mPtDcaCTxNBI/IHiz0pZgr8VEFqXFtZWpYD08ZB6FaSwVAS8MKRQAP3bYKiVjwysOJgMFY28o6Tw==", + "version": "7.12.5", + "resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.12.5.tgz", + "integrity": "sha512-plcc+hbExy3McchJCEQG3knOsuh3HH+Prx1P6cLIkET/0dLuQDEnrT+s27Axgc9bqfsmNUNHfscgMUdBpC9xfg==", "dev": true, "requires": { "regenerator-runtime": "^0.13.4" } }, "@babel/template": { - "version": "7.10.4", - "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.10.4.tgz", - "integrity": "sha512-ZCjD27cGJFUB6nmCB1Enki3r+L5kJveX9pq1SvAUKoICy6CZ9yD8xO086YXdYhvNjBdnekm4ZnaP5yC8Cs/1tA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/template/-/template-7.14.5.tgz", + "integrity": "sha512-6Z3Po85sfxRGachLULUhOmvAaOo7xCvqGQtxINai2mEGPFm6pQ4z5QInFnUrRpfoSV60BnjyF5F3c+15fxFV1g==", "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/parser": "^7.10.4", - "@babel/types": "^7.10.4" + "@babel/code-frame": "^7.14.5", + "@babel/parser": "^7.14.5", + "@babel/types": "^7.14.5" } }, "@babel/traverse": { - "version": "7.11.5", - "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.11.5.tgz", - "integrity": "sha512-EjiPXt+r7LiCZXEfRpSJd+jUMnBd4/9OUv7Nx3+0u9+eimMwJmG0Q98lw4/289JCoxSE8OolDMNZaaF/JZ69WQ==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/traverse/-/traverse-7.14.5.tgz", + "integrity": "sha512-G3BiS15vevepdmFqmUc9X+64y0viZYygubAMO8SvBmKARuF6CPSZtH4Ng9vi/lrWlZFGe3FWdXNy835akH8Glg==", "requires": { - "@babel/code-frame": "^7.10.4", - "@babel/generator": "^7.11.5", - "@babel/helper-function-name": "^7.10.4", - "@babel/helper-split-export-declaration": "^7.11.0", - "@babel/parser": "^7.11.5", - "@babel/types": "^7.11.5", + "@babel/code-frame": "^7.14.5", + "@babel/generator": "^7.14.5", + "@babel/helper-function-name": "^7.14.5", + "@babel/helper-hoist-variables": "^7.14.5", + "@babel/helper-split-export-declaration": "^7.14.5", + "@babel/parser": "^7.14.5", + "@babel/types": "^7.14.5", "debug": "^4.1.0", - "globals": "^11.1.0", - "lodash": "^4.17.19" + "globals": "^11.1.0" }, "dependencies": { "@babel/generator": { - "version": "7.11.6", - "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.11.6.tgz", - "integrity": "sha512-DWtQ1PV3r+cLbySoHrwn9RWEgKMBLLma4OBQloPRyDYvc5msJM9kvTLo1YnlJd1P/ZuKbdli3ijr5q3FvAF3uA==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/generator/-/generator-7.14.5.tgz", + "integrity": "sha512-y3rlP+/G25OIX3mYKKIOlQRcqj7YgrvHxOLbVmyLJ9bPmi5ttvUmpydVjcFjZphOktWuA7ovbx91ECloWTfjIA==", "requires": { - "@babel/types": "^7.11.5", + "@babel/types": "^7.14.5", "jsesc": "^2.5.1", "source-map": "^0.5.0" } @@ -2020,19 +18745,29 @@ } }, "@babel/types": { - "version": "7.11.5", - "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.11.5.tgz", - "integrity": "sha512-bvM7Qz6eKnJVFIn+1LPtjlBFPVN5jNDc1XmN15vWe7Q3DPBufWWsLiIvUu7xW87uTG6QoggpIDnUgLQvPheU+Q==", + "version": "7.14.5", + "resolved": "https://registry.npmjs.org/@babel/types/-/types-7.14.5.tgz", + "integrity": "sha512-M/NzBpEL95I5Hh4dwhin5JlE7EzO5PHMAuzjxss3tiOBD46KfQvVedN/3jEPZvdRvtsK2222XfdHogNIttFgcg==", "requires": { - "@babel/helper-validator-identifier": "^7.10.4", - "lodash": "^4.17.19", + "@babel/helper-validator-identifier": "^7.14.5", "to-fast-properties": "^2.0.0" } }, + "@ctrl/tinycolor": { + "version": "3.4.0", + "resolved": "https://registry.npmjs.org/@ctrl/tinycolor/-/tinycolor-3.4.0.tgz", + "integrity": "sha512-JZButFdZ1+/xAfpguQHoabIXkcqRRKpMrWKBkpEZZyxfY9C1DpADFB8PEqGSTeFr135SaTRfKqGKx5xSCLI7ZQ==" + }, + "@discoveryjs/json-ext": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.2.tgz", + "integrity": "sha512-HyYEUDeIj5rRQU2Hk5HTB2uHsbRQpF70nvMhVzi+VJR0X+xNEhjPui4/kBf3VeH/wqD28PT4sVOm8qqLjBrSZg==", + "dev": true + }, "@istanbuljs/schema": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.2.tgz", - "integrity": "sha512-tsAQNx32a8CoFhjhijUIhI4kccIAgmGhy8LZMZgGfmXcpMbPRUqn5LWmgRttILi6yeGmBJd2xsPkFMs0PzgPCw==", + "version": "0.1.3", + "resolved": "https://registry.npmjs.org/@istanbuljs/schema/-/schema-0.1.3.tgz", + "integrity": "sha512-ZXRY4jNvVgSVQ8DL3LTcakaAtXwTVUxE81hslsyD2AtoXW/wVob10HkOJ1X/pAlcI7D+2YoZKg5do8G/w6RYgA==", "dev": true }, "@jsdevtools/coverage-istanbul-loader": { @@ -2049,87 +18784,172 @@ } }, "@ng-bootstrap/ng-bootstrap": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/@ng-bootstrap/ng-bootstrap/-/ng-bootstrap-8.0.0.tgz", - "integrity": "sha512-v77Gfd8xHH+exq0WqIqVRlxbUEHdA/2+RUJenUP2IDTQN9E1rWl7O461/kosr+0XPuxPArHQJxhh/WsCYckcNg==", + "version": "9.1.2", + "resolved": "https://registry.npmjs.org/@ng-bootstrap/ng-bootstrap/-/ng-bootstrap-9.1.2.tgz", + "integrity": "sha512-oZt05AjwIY/oagS3QtST2GIlW9LQgTM7gX+wsuLQvHQt8bh9THOHAws8rDobeZi9fTUR3OvuxK56Mvj1jvyljw==", + "requires": { + "tslib": "^2.0.0" + } + }, + "@ng-select/ng-select": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/@ng-select/ng-select/-/ng-select-7.0.0.tgz", + "integrity": "sha512-YKQcohssQzLZgItCsEb0vbJR6xGRTIVq4FRp+5M7fkaX7wWM+o2o1R4LxsiWhz0CKajslkoR0VDytyiumf8QiQ==", "requires": { "tslib": "^2.0.0" } }, "@ngtools/webpack": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-10.2.0.tgz", - "integrity": "sha512-W4SSFNQhIiC8JRhIn3c4mb1+fsFKiHp+THVMAUNo+wRZEt/rgzsCdnqv0EmQJJojZhnilUIyB/wVYJu2+S/Bxg==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@ngtools/webpack/-/webpack-11.2.14.tgz", + "integrity": "sha512-6q57tEWtUJRsxfTKE19L20iXvNesfVy8hrVdyzVk64DZQh0lIl4/xZT4d5bJCWOuQQDaAeZK4YbEFcYJn7k1yw==", "dev": true, "requires": { - "@angular-devkit/core": "10.2.0", - "enhanced-resolve": "4.3.0", - "webpack-sources": "1.4.3" - }, - "dependencies": { - "@angular-devkit/core": { - "version": "10.2.0", - "resolved": "https://registry.npmjs.org/@angular-devkit/core/-/core-10.2.0.tgz", - "integrity": "sha512-XAszFhSF3mZw1VjoOsYGbArr5NJLcStjOvcCGjBPl1UBM2AKpuCQXHxI9XJGYKL3B93Vp5G58d8qkHvamT53OA==", - "dev": true, - "requires": { - "ajv": "6.12.4", - "fast-json-stable-stringify": "2.1.0", - "magic-string": "0.25.7", - "rxjs": "6.6.2", - "source-map": "0.7.3" - } - }, - "rxjs": { - "version": "6.6.2", - "resolved": "https://registry.npmjs.org/rxjs/-/rxjs-6.6.2.tgz", - "integrity": "sha512-BHdBMVoWC2sL26w//BCu3YzKT4s2jip/WhwsGEDmeKYBhKDZeYezVUnHatYB7L85v5xs0BAQmg6BEYJEKxBabg==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - } - }, - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true - } + "@angular-devkit/core": "11.2.14", + "enhanced-resolve": "5.7.0", + "webpack-sources": "2.2.0" } }, "@nodelib/fs.scandir": { - "version": "2.1.3", - "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.3.tgz", - "integrity": "sha512-eGmwYQn3gxo4r7jdQnkrrN6bY478C3P+a/y72IJukF8LjB6ZHeB3c+Ehacj3sYeSmUXGlnA67/PmbM9CVwL7Dw==", + "version": "2.1.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz", + "integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==", "dev": true, "requires": { - "@nodelib/fs.stat": "2.0.3", + "@nodelib/fs.stat": "2.0.5", "run-parallel": "^1.1.9" } }, "@nodelib/fs.stat": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.3.tgz", - "integrity": "sha512-bQBFruR2TAwoevBEd/NWMoAAtNGzTRgdrqnYCc7dhzfoNvqPzLyqlEQnzZ3kVnNrSp25iyxE00/3h2fqGAGArA==", + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz", + "integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==", "dev": true }, "@nodelib/fs.walk": { - "version": "1.2.4", - "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.4.tgz", - "integrity": "sha512-1V9XOY4rDW0rehzbrcqAmHnz8e7SKvX27gh8Gt2WgB0+pdzdiLV83p72kZPU+jvMbS1qU5mauP2iOvO8rhmurQ==", + "version": "1.2.7", + "resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.7.tgz", + "integrity": "sha512-BTIhocbPBSrRmHxOAJFtR18oLhxTtAFDAvL8hY1S3iU8k+E60W/YFs4jrixGzQjMpF4qPXxIQHcjVD9dz1C2QA==", "dev": true, "requires": { - "@nodelib/fs.scandir": "2.1.3", + "@nodelib/fs.scandir": "2.1.5", "fastq": "^1.6.0" } }, - "@npmcli/move-file": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/@npmcli/move-file/-/move-file-1.0.1.tgz", - "integrity": "sha512-Uv6h1sT+0DrblvIrolFtbvM1FgWm+/sy4B3pvLp67Zys+thcukzS5ekn7HsZFGpWP4Q3fYJCljbWQE/XivMRLw==", + "@npmcli/ci-detect": { + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/@npmcli/ci-detect/-/ci-detect-1.3.0.tgz", + "integrity": "sha512-oN3y7FAROHhrAt7Rr7PnTSwrHrZVRTS2ZbyxeQwSSYD0ifwM3YNgQqbaRmjcWoPyq77MjchusjJDspbzMmip1Q==", + "dev": true + }, + "@npmcli/git": { + "version": "2.0.9", + "resolved": "https://registry.npmjs.org/@npmcli/git/-/git-2.0.9.tgz", + "integrity": "sha512-hTMbMryvOqGLwnmMBKs5usbPsJtyEsMsgXwJbmNrsEuQQh1LAIMDU77IoOrwkCg+NgQWl+ySlarJASwM3SutCA==", "dev": true, "requires": { - "mkdirp": "^1.0.4" + "@npmcli/promise-spawn": "^1.3.2", + "lru-cache": "^6.0.0", + "mkdirp": "^1.0.4", + "npm-pick-manifest": "^6.1.1", + "promise-inflight": "^1.0.1", + "promise-retry": "^2.0.1", + "semver": "^7.3.5", + "which": "^2.0.2" + }, + "dependencies": { + "err-code": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", + "dev": true + }, + "hosted-git-info": { + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-4.0.2.tgz", + "integrity": "sha512-c9OGXbZ3guC/xOlCg1Ci/VgWlwsqDv1yMQL1CWqXDL0hDjXuNcq0zuR4xqPSuasI3kqFDhqSyTjREz5gzq0fXg==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", + "dev": true + }, + "npm-package-arg": { + "version": "8.1.4", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.1.4.tgz", + "integrity": "sha512-xLokoCFqj/rPdr3LvcdDL6Kj6ipXGEDHD/QGpzwU6/pibYUOXmp5DBmg76yukFyx4ZDbrXNOTn+BPyd8TD4Jlw==", + "dev": true, + "requires": { + "hosted-git-info": "^4.0.1", + "semver": "^7.3.4", + "validate-npm-package-name": "^3.0.0" + } + }, + "npm-pick-manifest": { + "version": "6.1.1", + "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-6.1.1.tgz", + "integrity": "sha512-dBsdBtORT84S8V8UTad1WlUyKIY9iMsAmqxHbLdeEeBNMLQDlDWWra3wYUx9EBEIiG/YwAy0XyNHDd2goAsfuA==", + "dev": true, + "requires": { + "npm-install-checks": "^4.0.0", + "npm-normalize-package-bin": "^1.0.1", + "npm-package-arg": "^8.1.2", + "semver": "^7.3.4" + } + }, + "promise-retry": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", + "dev": true, + "requires": { + "err-code": "^2.0.2", + "retry": "^0.12.0" + } + }, + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + }, + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, + "@npmcli/installed-package-contents": { + "version": "1.0.7", + "resolved": "https://registry.npmjs.org/@npmcli/installed-package-contents/-/installed-package-contents-1.0.7.tgz", + "integrity": "sha512-9rufe0wnJusCQoLpV9ZPKIVP55itrM5BxOXs10DmdbRfgWtHy1LDyskbwRnBghuB0PrF7pNPOqREVtpz4HqzKw==", + "dev": true, + "requires": { + "npm-bundled": "^1.1.1", + "npm-normalize-package-bin": "^1.0.1" + } + }, + "@npmcli/move-file": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@npmcli/move-file/-/move-file-1.1.2.tgz", + "integrity": "sha512-1SUf/Cg2GzGDyaf15aR9St9TWlb+XvbZXWpDx8YKs7MLzMH/BCeopv+y9vzrzgkfykCGuWOlSu3mZhj2+FQcrg==", + "dev": true, + "requires": { + "mkdirp": "^1.0.4", + "rimraf": "^3.0.2" }, "dependencies": { "mkdirp": { @@ -2140,38 +18960,125 @@ } } }, + "@npmcli/node-gyp": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/@npmcli/node-gyp/-/node-gyp-1.0.2.tgz", + "integrity": "sha512-yrJUe6reVMpktcvagumoqD9r08fH1iRo01gn1u0zoCApa9lnZGEigVKUd2hzsCId4gdtkZZIVscLhNxMECKgRg==", + "dev": true + }, + "@npmcli/promise-spawn": { + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/@npmcli/promise-spawn/-/promise-spawn-1.3.2.tgz", + "integrity": "sha512-QyAGYo/Fbj4MXeGdJcFzZ+FkDkomfRBrPM+9QYJSg+PxgAUL+LU3FneQk37rKR2/zjqkCV1BLHccX98wRXG3Sg==", + "dev": true, + "requires": { + "infer-owner": "^1.0.4" + } + }, + "@npmcli/run-script": { + "version": "1.8.5", + "resolved": "https://registry.npmjs.org/@npmcli/run-script/-/run-script-1.8.5.tgz", + "integrity": "sha512-NQspusBCpTjNwNRFMtz2C5MxoxyzlbuJ4YEhxAKrIonTiirKDtatsZictx9RgamQIx6+QuHMNmPl0wQdoESs9A==", + "dev": true, + "requires": { + "@npmcli/node-gyp": "^1.0.2", + "@npmcli/promise-spawn": "^1.3.2", + "infer-owner": "^1.0.4", + "node-gyp": "^7.1.0", + "read-package-json-fast": "^2.0.1" + }, + "dependencies": { + "read-package-json-fast": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-2.0.2.tgz", + "integrity": "sha512-5fyFUyO9B799foVk4n6ylcoAktG/FbE3jwRKxvwaeSrIunaoMc0u81dzXxjeAFKOce7O5KncdfwpGvvs6r5PsQ==", + "dev": true, + "requires": { + "json-parse-even-better-errors": "^2.3.0", + "npm-normalize-package-bin": "^1.0.1" + } + } + } + }, "@scarf/scarf": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-1.1.0.tgz", "integrity": "sha512-b2iE8kjjzzUo2WZ0xuE2N77kfnTds7ClrDxcz3Atz7h2XrNVoAPUoT75i7CY0st5x++70V91Y+c6RpBX9MX7Jg==" }, "@schematics/angular": { - "version": "10.1.5", - "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-10.1.5.tgz", - "integrity": "sha512-3VRcMB9WpjcMvlZ1y+78WGuZ4Ehp9pGw/T+zAR1VG9/16XHDQyfObsMuaU2EnEoufiHbTe3UpvVpYOu6tOCJrA==", + "version": "11.2.14", + "resolved": "https://registry.npmjs.org/@schematics/angular/-/angular-11.2.14.tgz", + "integrity": "sha512-nErn5BFYp4HB7mOkt23kF+dyM6zPxolejM8eXQ5vd/rdhcc6ROaMZ0EmeEAWkfqB3+vqaSDz/D2Nm/IjJlyW/Q==", "dev": true, "requires": { - "@angular-devkit/core": "10.1.5", - "@angular-devkit/schematics": "10.1.5", - "jsonc-parser": "2.3.0" + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", + "jsonc-parser": "3.0.0" } }, "@schematics/update": { - "version": "0.1001.5", - "resolved": "https://registry.npmjs.org/@schematics/update/-/update-0.1001.5.tgz", - "integrity": "sha512-DSomJ5IMs/5HUPx0RdPYubPWXh7kToxXUZbJywe0Q+TWTd+1xFfg8++O1DG4iW7E/Boqojx5VenAOzWY9jDWjA==", + "version": "0.1102.14", + "resolved": "https://registry.npmjs.org/@schematics/update/-/update-0.1102.14.tgz", + "integrity": "sha512-OsWuC0iyNjpST1+hVUUZAegXAFpEFpS5uKYSQF3jsbyw8XHx7oA5/HbEwyr2WkX2EdV1tKrDLz6BrD5b8W6EYw==", "dev": true, "requires": { - "@angular-devkit/core": "10.1.5", - "@angular-devkit/schematics": "10.1.5", + "@angular-devkit/core": "11.2.14", + "@angular-devkit/schematics": "11.2.14", "@yarnpkg/lockfile": "1.1.0", - "ini": "1.3.5", + "ini": "2.0.0", "npm-package-arg": "^8.0.0", - "pacote": "9.5.12", - "semver": "7.3.2", + "pacote": "11.2.4", + "semver": "7.3.4", "semver-intersect": "1.4.0" + }, + "dependencies": { + "ini": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ini/-/ini-2.0.0.tgz", + "integrity": "sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA==", + "dev": true + }, + "semver": { + "version": "7.3.4", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.4.tgz", + "integrity": "sha512-tCfb2WLjqFAtXn4KEdxIhalnRtoKFN7nAwj0B3ZXCbQloV2tq5eDbcTmT68JJD3nRJq24/XgxtQKFIpQdtvmVw==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" + } + } } }, + "@tootallnate/once": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/@tootallnate/once/-/once-1.1.2.tgz", + "integrity": "sha512-RbzJvlNzmRq5c3O09UipeuXno4tA1FE6ikOjxZK0tuxVv3412l64l5t1W5pj4+rJq9vpkm/kwiR07aZXnsKPxw==", + "dev": true + }, + "@trysound/sax": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/@trysound/sax/-/sax-0.1.1.tgz", + "integrity": "sha512-Z6DoceYb/1xSg5+e+ZlPZ9v0N16ZvZ+wYMraFue4HYrE4ttONKtsvruIRf6t9TBR0YvSOfi1hUU0fJfBLCDYow==", + "dev": true + }, + "@types/component-emitter": { + "version": "1.2.10", + "resolved": "https://registry.npmjs.org/@types/component-emitter/-/component-emitter-1.2.10.tgz", + "integrity": "sha512-bsjleuRKWmGqajMerkzox19aGbscQX5rmmvvXl3wlIp5gMG1HgkiwPxsN5p070fBDKTNSPgojVbuY1+HWMbFhg==", + "dev": true + }, + "@types/cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/@types/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-y7mImlc/rNkvCRmg8gC3/lj87S7pTUIJ6QGjwHR9WQJcFs+ZMTOaoPrkdFA/YdbuqVEmEbb5RdhVxMkAcgOnpg==", + "dev": true + }, + "@types/cors": { + "version": "2.8.10", + "resolved": "https://registry.npmjs.org/@types/cors/-/cors-2.8.10.tgz", + "integrity": "sha512-C7srjHiVG3Ey1nR6d511dtDkCEjxuN9W1HWAEjGq8kpcwmNM6JJkpC0xvabM7BXTG2wDq8Eu33iH9aQKa7IvLQ==", + "dev": true + }, "@types/glob": { "version": "7.1.3", "resolved": "https://registry.npmjs.org/@types/glob/-/glob-7.1.3.tgz", @@ -2183,9 +19090,9 @@ } }, "@types/jasmine": { - "version": "3.5.14", - "resolved": "https://registry.npmjs.org/@types/jasmine/-/jasmine-3.5.14.tgz", - "integrity": "sha512-Fkgk536sHPqcOtd+Ow+WiUNuk0TSo/BntKkF8wSvcd6M2FvPjeXcUE6Oz/bwDZiUZEaXLslAgw00Q94Pnx6T4w==", + "version": "3.6.11", + "resolved": "https://registry.npmjs.org/@types/jasmine/-/jasmine-3.6.11.tgz", + "integrity": "sha512-S6pvzQDvMZHrkBz2Mcn/8Du7cpr76PlRJBAoHnSDNbulULsH5dp0Gns+WRyNX5LHejz/ljxK4/vIHK/caHt6SQ==", "dev": true }, "@types/jasminewd2": { @@ -2198,15 +19105,15 @@ } }, "@types/json-schema": { - "version": "7.0.6", - "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.6.tgz", - "integrity": "sha512-3c+yGKvVP5Y9TYBEibGNR+kLtijnj7mYrXRg+WpFb2X9xm04g/DXYkfg4hmzJQosc9snFNUPkbYIhu+KAm6jJw==", + "version": "7.0.7", + "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.7.tgz", + "integrity": "sha512-cxWFQVseBm6O9Gbw1IWb8r6OS4OhSt3hPZLkFApLjM8TEXROBuQGLAH2i2gZpcXdLBIrpXuTDhH7Vbm1iXmNGA==", "dev": true }, "@types/minimatch": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.3.tgz", - "integrity": "sha512-tHq6qdbT9U1IRSGf14CL0pUlULksvY9OZ+5eEgl1N7t+OA3tGvNpxJCzuKQlsNgCVwbAs670L1vcVQi8j9HjnA==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/@types/minimatch/-/minimatch-3.0.4.tgz", + "integrity": "sha512-1z8k4wzFnNjVK/tlxvrWuK5WMt6mydWWP7+zvH5eFep4oj+UkrfiJTRtjCeBXNpwaA/FYqqtb4/QS4ianFpIRA==", "dev": true }, "@types/node": { @@ -2215,17 +19122,17 @@ "integrity": "sha512-UV1/ZJMC+HcP902wWdpC43cAcGu0IQk/I5bXjP2aSuCjsk3cE74mDvFrLKga7oDC170ugOAYBwfT4DSQW3akDA==", "dev": true }, + "@types/parse-json": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/@types/parse-json/-/parse-json-4.0.0.tgz", + "integrity": "sha512-//oorEZjL6sbPcKUaCdIGlIUeH26mgzimjBB77G6XRgnDl/L5wOnpyBGRe/Mmf5CVW3PwEBE1NjiMZ/ssFh4wA==", + "dev": true + }, "@types/pdfjs-dist": { "version": "2.1.7", "resolved": "https://registry.npmjs.org/@types/pdfjs-dist/-/pdfjs-dist-2.1.7.tgz", "integrity": "sha512-nQIwcPUhkAIyn7x9NS0lR/qxYfd5unRtfGkMjvpgF4Sh28IXftRymaNmFKTTdejDNY25NDGSIyjwj/BRwAPexg==" }, - "@types/q": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/@types/q/-/q-1.5.4.tgz", - "integrity": "sha512-1HcDas8SEj4z1Wc696tH56G8OlRaH/sqZOynNNB+HF0WOeXPaxTtbYzJY2oEfiUxjSKjhCKr+MvR7dCHcEelug==", - "dev": true - }, "@types/selenium-webdriver": { "version": "3.0.17", "resolved": "https://registry.npmjs.org/@types/selenium-webdriver/-/selenium-webdriver-3.0.17.tgz", @@ -2450,22 +19357,18 @@ "integrity": "sha512-GpSwvyXOcOOlV70vbnzjj4fW5xW/FdUF6nQEt1ENy7m4ZCczi1+/buVUPAqmGfqznsORNFzUMjctTIp8a9tuCQ==", "dev": true }, - "JSONStream": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/JSONStream/-/JSONStream-1.3.5.tgz", - "integrity": "sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==", - "dev": true, - "requires": { - "jsonparse": "^1.2.0", - "through": ">=2.2.7 <3" - } - }, "abab": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/abab/-/abab-2.0.5.tgz", "integrity": "sha512-9IK9EadsbHo6jLWIpxpR6pL0sazTXV6+SQv25ZB+F7Bj9mJNaOc4nCRabwd5M/JwmUa8idz6Eci6eKfJryPs6Q==", "dev": true }, + "abbrev": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/abbrev/-/abbrev-1.1.1.tgz", + "integrity": "sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q==", + "dev": true + }, "accepts": { "version": "1.3.7", "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", @@ -2483,9 +19386,9 @@ "dev": true }, "adjust-sourcemap-loader": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/adjust-sourcemap-loader/-/adjust-sourcemap-loader-3.0.0.tgz", - "integrity": "sha512-YBrGyT2/uVQ/c6Rr+t6ZJXniY03YtHGMJQYal368burRGYKqhx9qGTWqcBU5s1CwYY9E/ri63RYyG1IacMZtqw==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/adjust-sourcemap-loader/-/adjust-sourcemap-loader-4.0.0.tgz", + "integrity": "sha512-OXwN5b9pCUXNQHJpwwD2qP40byEmSgzj8B4ydSN0uMNYWiFmJ6x6KwUllMmfk8Rwu/HJDFR7U8ubsWBoN0Xp0A==", "dev": true, "requires": { "loader-utils": "^2.0.0", @@ -2498,12 +19401,6 @@ "integrity": "sha512-TFi4HBKSGfIKsK5YCkKaaFG2m4PEDyViZmEwof3MTIgzimHLto6muaHVpbrljdIvIrFZzEq/p4nafOeLcYegrg==", "dev": true }, - "after": { - "version": "0.8.2", - "resolved": "https://registry.npmjs.org/after/-/after-0.8.2.tgz", - "integrity": "sha1-/ts5T58OAqqXaOcCvaI7UF+ufh8=", - "dev": true - }, "agent-base": { "version": "4.3.0", "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.3.0.tgz", @@ -2514,11 +19411,13 @@ } }, "agentkeepalive": { - "version": "3.5.2", - "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-3.5.2.tgz", - "integrity": "sha512-e0L/HNe6qkQ7H19kTlRRqUibEAwDK5AFk6y3PtMsuut2VAH6+Q4xZml1tNDJD7kSAyqmbG/K08K5WEJYtUrSlQ==", + "version": "4.1.4", + "resolved": "https://registry.npmjs.org/agentkeepalive/-/agentkeepalive-4.1.4.tgz", + "integrity": "sha512-+V/rGa3EuU74H6wR04plBb7Ks10FbtUQgRj/FQOG7uUIEuaINI+AiqJR1k6t3SVNs7o7ZjIdus6706qqzVq8jQ==", "dev": true, "requires": { + "debug": "^4.1.0", + "depd": "^1.1.2", "humanize-ms": "^1.2.1" } }, @@ -2533,9 +19432,9 @@ } }, "ajv": { - "version": "6.12.4", - "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.4.tgz", - "integrity": "sha512-eienB2c9qVQs2KWexhkrdMLVDoIQCz5KSeLxwg9Lzk4DOfBtIK9PQwwufcsn1jjGuf9WZmqPMbGxOzfcuphJCQ==", + "version": "6.12.6", + "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", + "integrity": "sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g==", "dev": true, "requires": { "fast-deep-equal": "^3.1.1", @@ -2548,13 +19447,15 @@ "version": "1.0.1", "resolved": "https://registry.npmjs.org/ajv-errors/-/ajv-errors-1.0.1.tgz", "integrity": "sha512-DCRfO/4nQ+89p/RK43i8Ezd41EqdGIU4ld7nGF8OQ14oc/we5rEntLCUa7+jrn3nn83BosfwZA0wb4pon2o8iQ==", - "dev": true + "dev": true, + "requires": {} }, "ajv-keywords": { "version": "3.5.2", "resolved": "https://registry.npmjs.org/ajv-keywords/-/ajv-keywords-3.5.2.tgz", "integrity": "sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ==", - "dev": true + "dev": true, + "requires": {} }, "alphanum-sort": { "version": "1.0.2", @@ -2563,9 +19464,9 @@ "dev": true }, "ansi-colors": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.4.tgz", - "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-4.1.1.tgz", + "integrity": "sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==", "dev": true }, "ansi-escapes": { @@ -2601,7 +19502,6 @@ "version": "3.1.1", "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz", "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==", - "dev": true, "requires": { "normalize-path": "^3.0.0", "picomatch": "^2.0.4" @@ -2619,6 +19519,16 @@ "integrity": "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw==", "dev": true }, + "are-we-there-yet": { + "version": "1.1.5", + "resolved": "https://registry.npmjs.org/are-we-there-yet/-/are-we-there-yet-1.1.5.tgz", + "integrity": "sha512-5hYdAkZlcG8tOLujVDTgCT+uPX0VnpAH28gWsLfzpXYm7wP6mp5Q/gYyR7YQ0cKVJcXJnl3j2kpBan13PtQf6w==", + "dev": true, + "requires": { + "delegates": "^1.0.0", + "readable-stream": "^2.0.6" + } + }, "arg": { "version": "4.1.3", "resolved": "https://registry.npmjs.org/arg/-/arg-4.1.3.tgz", @@ -2644,12 +19554,6 @@ "commander": "^2.11.0" } }, - "arity-n": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/arity-n/-/arity-n-1.0.4.tgz", - "integrity": "sha1-2edrEXM+CFacCEeuezmyhgswt0U=", - "dev": true - }, "arr-diff": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/arr-diff/-/arr-diff-4.0.0.tgz", @@ -2692,24 +19596,12 @@ "integrity": "sha1-qJS3XUvE9s1nnvMkSp/Y9Gri1Cg=", "dev": true }, - "arraybuffer.slice": { - "version": "0.0.7", - "resolved": "https://registry.npmjs.org/arraybuffer.slice/-/arraybuffer.slice-0.0.7.tgz", - "integrity": "sha512-wGUIVQXuehL5TCqQun8OW81jGzAWycqzFF8lFp+GOM5BXLYj3bKNsYC4daB7n6XjCqxQA/qgTJ+8ANR3acjrog==", - "dev": true - }, "arrify": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/arrify/-/arrify-1.0.1.tgz", "integrity": "sha1-iYUI2iIm84DfkEcoRWhJwVAaSw0=", "dev": true }, - "asap": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/asap/-/asap-2.0.6.tgz", - "integrity": "sha1-5QNHYR1+aQlDIIu9r+vLwvuGbUY=", - "dev": true - }, "asn1": { "version": "0.2.4", "resolved": "https://registry.npmjs.org/asn1/-/asn1-0.2.4.tgz", @@ -2732,9 +19624,9 @@ }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -2818,17 +19710,16 @@ "dev": true }, "autoprefixer": { - "version": "9.8.6", - "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-9.8.6.tgz", - "integrity": "sha512-XrvP4VVHdRBCdX1S3WXVD8+RyG9qeb1D5Sn1DeLiG2xfSpzellk5k54xbUERJ3M5DggQxes39UGOTP8CFrEGbg==", + "version": "10.2.4", + "resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.2.4.tgz", + "integrity": "sha512-DCCdUQiMD+P/as8m3XkeTUkUKuuRqLGcwD0nll7wevhqoJfMRpJlkFd1+MQh1pvupjiQuip42lc/VFvfUTMSKw==", "dev": true, "requires": { - "browserslist": "^4.12.0", - "caniuse-lite": "^1.0.30001109", + "browserslist": "^4.16.1", + "caniuse-lite": "^1.0.30001181", "colorette": "^1.2.1", + "fraction.js": "^4.0.13", "normalize-range": "^0.1.2", - "num2fraction": "^1.2.2", - "postcss": "^7.0.32", "postcss-value-parser": "^4.1.0" } }, @@ -2854,29 +19745,17 @@ } }, "babel-loader": { - "version": "8.1.0", - "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.1.0.tgz", - "integrity": "sha512-7q7nC1tYOrqvUrN3LQK4GwSk/TQorZSOlO9C+RZDZpODgyN4ZlCqE5q9cDsyWOliN+aU9B4JX01xK9eJXowJLw==", + "version": "8.2.2", + "resolved": "https://registry.npmjs.org/babel-loader/-/babel-loader-8.2.2.tgz", + "integrity": "sha512-JvTd0/D889PQBtUXJ2PXaKU/pjZDMtHA9V2ecm+eNRmmBCMR09a+fmpGTNwnJtFmFl5Ei7Vy47LjBb+L0wQ99g==", "dev": true, "requires": { - "find-cache-dir": "^2.1.0", + "find-cache-dir": "^3.3.1", "loader-utils": "^1.4.0", - "mkdirp": "^0.5.3", - "pify": "^4.0.1", + "make-dir": "^3.1.0", "schema-utils": "^2.6.5" }, "dependencies": { - "find-cache-dir": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/find-cache-dir/-/find-cache-dir-2.1.0.tgz", - "integrity": "sha512-Tq6PixE0w/VMFfCgbONnkiQIVol/JJL7nRMi20fqzA4NRs9AfeqMGeRdPi3wIhYkxjeBaWh2rxwapn5Tu3IqOQ==", - "dev": true, - "requires": { - "commondir": "^1.0.1", - "make-dir": "^2.0.0", - "pkg-dir": "^3.0.0" - } - }, "json5": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", @@ -2896,6 +19775,21 @@ "emojis-list": "^3.0.0", "json5": "^1.0.1" } + }, + "make-dir": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", + "integrity": "sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw==", + "dev": true, + "requires": { + "semver": "^6.0.0" + } + }, + "semver": { + "version": "6.3.0", + "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", + "integrity": "sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw==", + "dev": true } } }, @@ -2908,12 +19802,6 @@ "object.assign": "^4.1.0" } }, - "backo2": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/backo2/-/backo2-1.0.2.tgz", - "integrity": "sha1-MasayLEpNjRj41s+u2n038+6eUc=", - "dev": true - }, "balanced-match": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", @@ -2942,35 +19830,6 @@ "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" - } } } }, @@ -3007,15 +19866,6 @@ "tweetnacl": "^0.14.3" } }, - "better-assert": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/better-assert/-/better-assert-1.0.2.tgz", - "integrity": "sha1-QIZrnhueC1W0gYlDEeaPr/rrxSI=", - "dev": true, - "requires": { - "callsite": "1.0.0" - } - }, "big.js": { "version": "5.2.2", "resolved": "https://registry.npmjs.org/big.js/-/big.js-5.2.2.tgz", @@ -3025,8 +19875,7 @@ "binary-extensions": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.1.0.tgz", - "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==", - "dev": true + "integrity": "sha512-1Yj8h9Q+QDF5FzhMs/c9+6UntbD5MkRfRwac8DoEm9ZfUBZ7tZ55YcGVAzEe4bXsdQHEk+s9S5wsOKVdZrw0tQ==" }, "bindings": { "version": "1.5.0", @@ -3038,11 +19887,39 @@ "file-uri-to-path": "1.0.0" } }, - "blob": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/blob/-/blob-0.0.5.tgz", - "integrity": "sha512-gaqbzQPqOoamawKg0LGVd7SzLgXS+JH61oWprSLH+P+abTczqJbhTR8CmJ2u9/bUYNmHTGJx/UEmn6doAvvuig==", - "dev": true + "bl": { + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz", + "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==", + "dev": true, + "requires": { + "buffer": "^5.5.0", + "inherits": "^2.0.4", + "readable-stream": "^3.4.0" + }, + "dependencies": { + "buffer": { + "version": "5.7.1", + "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz", + "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==", + "dev": true, + "requires": { + "base64-js": "^1.3.1", + "ieee754": "^1.1.13" + } + }, + "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" + } + } + } }, "blocking-proxy": { "version": "1.0.1", @@ -3060,9 +19937,9 @@ "dev": true }, "bn.js": { - "version": "5.1.3", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.1.3.tgz", - "integrity": "sha512-GkTiFpjFtUzU9CbMeJ5iazkCzGL3jrhzerzZIuqLABjbwRaFt33I9tUdSNryIptM+RxDet6OKm2WnLXzW51KsQ==", + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-5.2.0.tgz", + "integrity": "sha512-D7iWRBvnZE8ecXiLj/9wbxH7Tk79fAh8IHaTNq1RWRixsS02W+5qS+iE9yq6RYl0asXx5tw0bLhmT5pIfbSquw==", "dev": true }, "body-parser": { @@ -3098,15 +19975,6 @@ "ms": "2.0.0" } }, - "iconv-lite": { - "version": "0.4.24", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", - "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", - "dev": true, - "requires": { - "safer-buffer": ">= 2.1.2 < 3" - } - }, "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -3153,7 +20021,6 @@ "version": "3.0.2", "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz", "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==", - "dev": true, "requires": { "fill-range": "^7.0.1" } @@ -3202,21 +20069,13 @@ } }, "browserify-rsa": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.0.1.tgz", - "integrity": "sha1-IeCr+vbyApzy+vsTNWenAdQTVSQ=", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/browserify-rsa/-/browserify-rsa-4.1.0.tgz", + "integrity": "sha512-AdEER0Hkspgno2aR97SAf6vi0y0k8NuOpGnVH3O99rcA5Q6sh8QxcngtHuJ6uXwnfAXNM4Gn1Gb7/MV1+Ymbog==", "dev": true, "requires": { - "bn.js": "^4.1.0", + "bn.js": "^5.0.0", "randombytes": "^2.0.1" - }, - "dependencies": { - "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", - "dev": true - } } }, "browserify-sign": { @@ -3265,15 +20124,16 @@ } }, "browserslist": { - "version": "4.14.5", - "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.14.5.tgz", - "integrity": "sha512-Z+vsCZIvCBvqLoYkBFTwEYH3v5MCQbsAjp50ERycpOjnPmolg1Gjy4+KaWWpm8QOJt9GHkhdqAl14NpCX73CWA==", + "version": "4.16.6", + "resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.16.6.tgz", + "integrity": "sha512-Wspk/PqO+4W9qp5iUTJsa1B/QrYn1keNCcEP5OvP7WBwT4KaDly0uONYmC6Xa3Z5IqnUgS0KcgLYu1l74x0ZXQ==", "dev": true, "requires": { - "caniuse-lite": "^1.0.30001135", - "electron-to-chromium": "^1.3.571", - "escalade": "^3.1.0", - "node-releases": "^1.1.61" + "caniuse-lite": "^1.0.30001219", + "colorette": "^1.2.2", + "electron-to-chromium": "^1.3.723", + "escalade": "^3.1.1", + "node-releases": "^1.1.71" } }, "browserstack": { @@ -3388,40 +20248,26 @@ "unset-value": "^1.0.0" } }, - "caller-callsite": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/caller-callsite/-/caller-callsite-2.0.0.tgz", - "integrity": "sha1-hH4PzgoiN1CpoCfFSzNzGtMVQTQ=", + "call-bind": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/call-bind/-/call-bind-1.0.2.tgz", + "integrity": "sha512-7O+FbCihrB5WGbFYesctwmTKae6rOiIzmz1icreWJ+0aA7LJfuqhEso2T9ncpcFtzMQtzXf2QGGueWJGTYsqrA==", "dev": true, "requires": { - "callsites": "^2.0.0" + "function-bind": "^1.1.1", + "get-intrinsic": "^1.0.2" } }, - "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": { - "caller-callsite": "^2.0.0" - } - }, - "callsite": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/callsite/-/callsite-1.0.0.tgz", - "integrity": "sha1-KAOY5dZkvXQDi28JBRU+borxvCA=", - "dev": true - }, "callsites": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/callsites/-/callsites-2.0.0.tgz", - "integrity": "sha1-BuuE8A7qQT2oav/vrL/7Ngk7PFA=", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", + "integrity": "sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ==", "dev": true }, "camelcase": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.1.0.tgz", - "integrity": "sha512-WCMml9ivU60+8rEJgELlFp1gxFcEGxwYleE3bziHEDeqsqAWGHdimB7beBFGjLzVNgPGyDsfgXLQEYMpmIFnVQ==", + "version": "5.3.1", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", + "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", "dev": true }, "caniuse-api": { @@ -3437,16 +20283,15 @@ } }, "caniuse-lite": { - "version": "1.0.30001150", - "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001150.tgz", - "integrity": "sha512-kiNKvihW0m36UhAFnl7bOAv0i1K1f6wpfVtTF5O5O82XzgtBnb05V0XeV3oZ968vfg2sRNChsHw8ASH2hDfoYQ==", + "version": "1.0.30001237", + "resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001237.tgz", + "integrity": "sha512-pDHgRndit6p1NR2GhzMbQ6CkRrp4VKuSsqbcLeOQppYPKOYkKT/6ZvZDvKJUqcmtyWIAHuZq3SVS2vc1egCZzw==", "dev": true }, "canonical-path": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/canonical-path/-/canonical-path-1.0.0.tgz", - "integrity": "sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg==", - "dev": true + "integrity": "sha512-feylzsbDxi1gPZ1IjystzIQZagYYLvfKrSuygUCgf7z6x790VEzze5QEkdSV1U58RA7Hi0+v6fv4K54atOzATg==" }, "caseless": { "version": "0.12.0", @@ -3474,7 +20319,6 @@ "version": "3.4.2", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.2.tgz", "integrity": "sha512-IZHaDeBeI+sZJRX7lGcXsdzgvZqKv6sECqsbErJA4mHWfpRrD8B97kSFN4cQz6nGBGiuFia1MKR4d6c1o8Cv7A==", - "dev": true, "requires": { "anymatch": "~3.1.1", "braces": "~3.0.2", @@ -3493,21 +20337,10 @@ "dev": true }, "chrome-trace-event": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.2.tgz", - "integrity": "sha512-9e/zx1jw7B4CO+c/RXoCsfg/x1AfUBioy4owYH0bJprEYAx5hRFLRhWBqHAG57D0ZM4H7vxbP7bPe0VwhQRYDQ==", - "dev": true, - "requires": { - "tslib": "^1.9.0" - }, - "dependencies": { - "tslib": { - "version": "1.14.1", - "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.1.tgz", - "integrity": "sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==", - "dev": true - } - } + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/chrome-trace-event/-/chrome-trace-event-1.0.3.tgz", + "integrity": "sha512-p3KULyQg4S7NIHixdwbGX+nFHkoBiA4YQmyWtjb8XngSKV124nJmRysgAeujbUVb15vh+RvFUfCPqU7rXk+hZg==", + "dev": true }, "cipher-base": { "version": "1.0.4", @@ -3520,10 +20353,11 @@ } }, "circular-dependency-plugin": { - "version": "5.2.0", - "resolved": "https://registry.npmjs.org/circular-dependency-plugin/-/circular-dependency-plugin-5.2.0.tgz", - "integrity": "sha512-7p4Kn/gffhQaavNfyDFg7LS5S/UT1JAjyGd4UqR2+jzoYF02eDkj0Ec3+48TsIa4zghjLY87nQHIh/ecK9qLdw==", - "dev": true + "version": "5.2.2", + "resolved": "https://registry.npmjs.org/circular-dependency-plugin/-/circular-dependency-plugin-5.2.2.tgz", + "integrity": "sha512-g38K9Cm5WRwlaH6g03B9OEz/0qRizI+2I7n+Gz+L5DxXJAPAiWQvwlYNm1V1jkdpUv95bOe/ASm2vfi/G560jQ==", + "dev": true, + "requires": {} }, "class-utils": { "version": "0.3.6", @@ -3545,6 +20379,63 @@ "requires": { "is-descriptor": "^0.1.0" } + }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-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": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "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 } } }, @@ -3564,9 +20455,9 @@ } }, "cli-spinners": { - "version": "2.4.0", - "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.4.0.tgz", - "integrity": "sha512-sJAofoarcm76ZGpuooaO0eDy8saEy+YoZBLjC4h8srt4jeBnkYeOgqxgsJQTpyt2LjI5PTfLJHSL+41Yu4fEJA==", + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/cli-spinners/-/cli-spinners-2.6.0.tgz", + "integrity": "sha512-t+4/y50K/+4xcCRosKkA7W4gTr1MySvLV0q+PxmG7FJ5g+66ChKurYjxBCjHggHH3HA5Hh9cy+lcUGWDqVH+4Q==", "dev": true }, "cli-width": { @@ -3604,22 +20495,28 @@ } }, "clone": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/clone/-/clone-2.1.2.tgz", - "integrity": "sha1-G39Ln1kfHo+DZwQBYANFoCiHQ18=", + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/clone/-/clone-1.0.4.tgz", + "integrity": "sha1-2jCcwmPfFZlMaIypAheco8fNfH4=", "dev": true }, - "coa": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/coa/-/coa-2.0.2.tgz", - "integrity": "sha512-q5/jG+YQnSy4nRTV4F7lPepBJZ8qBNJJDBuJdoejDyLXgmL7IEo+Le2JDZudFTFt7mrCqIRaSjws4ygRCTCAXA==", + "clone-deep": { + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/clone-deep/-/clone-deep-4.0.1.tgz", + "integrity": "sha512-neHB9xuzh/wk0dIHweyAXv2aPGZIVk3pLMe+/RNzINf17fe0OG96QroktYAUm7SM1PBnzTabaLboqqxDyMU+SQ==", "dev": true, "requires": { - "@types/q": "^1.5.1", - "chalk": "^2.4.1", - "q": "^1.1.2" + "is-plain-object": "^2.0.4", + "kind-of": "^6.0.2", + "shallow-clone": "^3.0.0" } }, + "code-point-at": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", + "dev": true + }, "codelyzer": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/codelyzer/-/codelyzer-6.0.1.tgz", @@ -3671,6 +20568,12 @@ "resolved": "https://registry.npmjs.org/tslib/-/tslib-1.14.0.tgz", "integrity": "sha512-+Zw5lu0D9tvBMjGP8LpvMb0u2WW2QV3y+D8mO6J+cNzCYIN4sVy43Bf9vl92nqFahutN0I8zHa7cc4vihIshnw==", "dev": true + }, + "zone.js": { + "version": "0.10.3", + "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.10.3.tgz", + "integrity": "sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg==", + "dev": true } } }, @@ -3684,16 +20587,6 @@ "object-visit": "^1.0.0" } }, - "color": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/color/-/color-3.1.3.tgz", - "integrity": "sha512-xgXAcTHa2HeFCGLE9Xs/R82hujGtu9Jd9x4NW3T34+OMs7VoPsjwzRczKHvTAHeJwWFwX5j15+MgAppE8ztObQ==", - "dev": true, - "requires": { - "color-convert": "^1.9.1", - "color-string": "^1.5.4" - } - }, "color-convert": { "version": "1.9.3", "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz", @@ -3707,20 +20600,16 @@ "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz", "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=" }, - "color-string": { - "version": "1.5.4", - "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.5.4.tgz", - "integrity": "sha512-57yF5yt8Xa3czSEW1jfQDE79Idk0+AkN/4KWad6tbdxUmAs3MvjxlWSWD4deYytcRfoZ9nhKyFl1kj5tBvidbw==", - "dev": true, - "requires": { - "color-name": "^1.0.0", - "simple-swizzle": "^0.2.2" - } + "colord": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/colord/-/colord-2.0.1.tgz", + "integrity": "sha512-vm5YpaWamD0Ov6TSG0GGmUIwstrWcfKQV/h2CmbR7PbNu41+qdB5PW9lpzhjedrpm08uuYvcXi0Oel1RLZIJuA==", + "dev": true }, "colorette": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.1.tgz", - "integrity": "sha512-puCDz0CzydiSYOrnXpz/PKd69zRrribezjtE9yd4zvytoRc8+RY/KJPvtPFKZS3E3wP6neGyMe0vOTlHO5L3Pw==", + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/colorette/-/colorette-1.2.2.tgz", + "integrity": "sha512-MKGMzyfeuutC/ZJ1cba9NqcNpfeqMUcYmyF1ZFY6/Cn7CNSAKx6a+s48sqLqyAiZuaP2TcqMhoo+dlwFnVxT9w==", "dev": true }, "colors": { @@ -3750,33 +20639,12 @@ "integrity": "sha1-3dgA2gxmEnOTzKWVDqloo6rxJTs=", "dev": true }, - "component-bind": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/component-bind/-/component-bind-1.0.0.tgz", - "integrity": "sha1-AMYIq33Nk4l8AAllGx06jh5zu9E=", - "dev": true - }, "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 }, - "component-inherit": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/component-inherit/-/component-inherit-0.0.3.tgz", - "integrity": "sha1-ZF/ErfWLcrZJ1crmUTVhnbJv8UM=", - "dev": true - }, - "compose-function": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/compose-function/-/compose-function-3.0.3.tgz", - "integrity": "sha1-ntZ18TzFRQHTCVCkhv9qe6OrGF8=", - "dev": true, - "requires": { - "arity-n": "^1.0.4" - } - }, "compressible": { "version": "2.0.18", "resolved": "https://registry.npmjs.org/compressible/-/compressible-2.0.18.tgz", @@ -3876,6 +20744,12 @@ "integrity": "sha512-ZMkYO/LkF17QvCPqM0gxw8yUzigAOZOSWSHg91FH6orS7vcEj5dVZTidN2fQ14yBSdg97RqhSNwLUXInd52OTA==", "dev": true }, + "console-control-strings": { + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", + "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=", + "dev": true + }, "constants-browserify": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/constants-browserify/-/constants-browserify-1.0.0.tgz", @@ -3917,6 +20791,15 @@ "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=", "dev": true }, + "copy-anything": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/copy-anything/-/copy-anything-2.0.3.tgz", + "integrity": "sha512-GK6QUtisv4fNS+XcI7shX0Gx9ORg7QqIznyfho79JTnX1XhLiyZHfftvGiziqzRiEi/Bjhgpi+D2o7HxJFPnDQ==", + "dev": true, + "requires": { + "is-what": "^3.12.0" + } + }, "copy-concurrently": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/copy-concurrently/-/copy-concurrently-1.0.5.tgz", @@ -3949,48 +20832,75 @@ "dev": true }, "copy-webpack-plugin": { - "version": "6.0.3", - "resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-6.0.3.tgz", - "integrity": "sha512-q5m6Vz4elsuyVEIUXr7wJdIdePWTubsqVbEMvf1WQnHGv0Q+9yPRu7MtYFPt+GBOXRav9lvIINifTQ1vSCs+eA==", + "version": "6.3.2", + "resolved": "https://registry.npmjs.org/copy-webpack-plugin/-/copy-webpack-plugin-6.3.2.tgz", + "integrity": "sha512-MgJ1uouLIbDg4ST1GzqrGQyKoXY5iPqi6fghFqarijam7FQcBa/r6Rg0VkoIuzx75Xq8iAMghyOueMkWUQ5OaA==", "dev": true, "requires": { - "cacache": "^15.0.4", + "cacache": "^15.0.5", "fast-glob": "^3.2.4", "find-cache-dir": "^3.3.1", "glob-parent": "^5.1.1", "globby": "^11.0.1", "loader-utils": "^2.0.0", "normalize-path": "^3.0.0", - "p-limit": "^3.0.1", - "schema-utils": "^2.7.0", - "serialize-javascript": "^4.0.0", + "p-limit": "^3.0.2", + "schema-utils": "^3.0.0", + "serialize-javascript": "^5.0.1", "webpack-sources": "^1.4.3" }, "dependencies": { "p-limit": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.0.2.tgz", - "integrity": "sha512-iwqZSOoWIW+Ew4kAGUlN16J4M7OB3ysMLSZtnhmqx7njIHFPlxWBX8xo3lVTyFVq6mI/lL9qt2IsN1sHwaxJkg==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", "dev": true, "requires": { - "p-try": "^2.0.0" + "yocto-queue": "^0.1.0" + } + }, + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, + "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 + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" } } } }, "core-js": { - "version": "3.6.4", - "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.6.4.tgz", - "integrity": "sha512-4paDGScNgZP2IXXilaffL9X7968RuvwlkK3xWtZRVqgd8SYNiVKRJvkFd1aqqEuPfN7E68ZHEp9hDj6lHj4Hyw==", + "version": "3.8.3", + "resolved": "https://registry.npmjs.org/core-js/-/core-js-3.8.3.tgz", + "integrity": "sha512-KPYXeVZYemC2TkNEkX/01I+7yd+nX3KddKwZ1Ww7SKWdI2wQprSgLmrTddT8nw92AjEklTsPBoSdQBhbI1bQ6Q==", "dev": true }, "core-js-compat": { - "version": "3.6.5", - "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.6.5.tgz", - "integrity": "sha512-7ItTKOhOZbznhXAQ2g/slGg1PJV5zDO/WdkTwi7UEOJmkvsE32PWvx6mKtDjiMpjnR2CNf6BAD6sSxIlv7ptng==", + "version": "3.14.0", + "resolved": "https://registry.npmjs.org/core-js-compat/-/core-js-compat-3.14.0.tgz", + "integrity": "sha512-R4NS2eupxtiJU+VwgkF9WTpnSfZW4pogwKHd8bclWU2sp93Pr5S1uYJI84cMOubJRou7bcfL0vmwtLslWN5p3A==", "dev": true, "requires": { - "browserslist": "^4.8.5", + "browserslist": "^4.16.6", "semver": "7.0.0" }, "dependencies": { @@ -4008,16 +20918,27 @@ "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac=", "dev": true }, - "cosmiconfig": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-5.2.1.tgz", - "integrity": "sha512-H65gsXo1SKjf8zmrJ67eJk8aIRKV5ff2D4uKZIBZShbhGSpEmsQOPW/SKMKYhSTrqR7ufy6RP69rPogdaPh/kA==", + "cors": { + "version": "2.8.5", + "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz", + "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==", "dev": true, "requires": { - "import-fresh": "^2.0.0", - "is-directory": "^0.3.1", - "js-yaml": "^3.13.1", - "parse-json": "^4.0.0" + "object-assign": "^4", + "vary": "^1" + } + }, + "cosmiconfig": { + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/cosmiconfig/-/cosmiconfig-7.0.0.tgz", + "integrity": "sha512-pondGvTuVYDk++upghXJabWzL6Kxu6f26ljFw64Swq9v6sQPUL3EUlVDV56diOjpCayKihL6hVe8exIACU4XcA==", + "dev": true, + "requires": { + "@types/parse-json": "^4.0.0", + "import-fresh": "^3.2.1", + "parse-json": "^5.0.0", + "path-type": "^4.0.0", + "yaml": "^1.10.0" } }, "create-ecdh": { @@ -4031,9 +20952,9 @@ }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -4065,6 +20986,70 @@ "sha.js": "^2.4.8" } }, + "critters": { + "version": "0.0.7", + "resolved": "https://registry.npmjs.org/critters/-/critters-0.0.7.tgz", + "integrity": "sha512-qUF2SaAWFYjNPdCcPpu68p2DnHiosia84yx5mPTlUMQjkjChR+n6sO1/I7yn2U2qNDgSPTd2SoaTIDQcUL+EwQ==", + "dev": true, + "requires": { + "chalk": "^4.1.0", + "css": "^3.0.0", + "parse5": "^6.0.1", + "parse5-htmlparser2-tree-adapter": "^6.0.1", + "pretty-bytes": "^5.3.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } + } + }, "cross-spawn": { "version": "6.0.5", "resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-6.0.5.tgz", @@ -4106,15 +21091,14 @@ } }, "css": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/css/-/css-2.2.4.tgz", - "integrity": "sha512-oUnjmWpy0niI3x/mPL8dVEI1l7MnG3+HHyRPHf+YFSbK+svOhXpmSOcDURUh2aOCgl2grzrOPt1nHLuCVFULLw==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/css/-/css-3.0.0.tgz", + "integrity": "sha512-DG9pFfwOrzc+hawpmqX/dHYHJG+Bsdb0klhyi1sDneOgGOXy9wQIC8hzyVp1e4NRYDBdxcylvywPkkXCHAzTyQ==", "dev": true, "requires": { - "inherits": "^2.0.3", + "inherits": "^2.0.4", "source-map": "^0.6.1", - "source-map-resolve": "^0.5.2", - "urix": "^0.1.0" + "source-map-resolve": "^0.6.0" }, "dependencies": { "source-map": { @@ -4126,39 +21110,57 @@ } }, "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=", + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/css-color-names/-/css-color-names-1.0.1.tgz", + "integrity": "sha512-/loXYOch1qU1biStIFsHH8SxTmOseh1IJqFvy8IujXOm1h+QjUdDhkzOrR5HG8K8mlxREj0yfi8ewCHx0eMxzA==", "dev": true }, "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==", + "version": "6.0.3", + "resolved": "https://registry.npmjs.org/css-declaration-sorter/-/css-declaration-sorter-6.0.3.tgz", + "integrity": "sha512-52P95mvW1SMzuRZegvpluT6yEv0FqQusydKQPZsNN5Q7hh8EwQvN8E2nwuJ16BBvNN6LcoIZXu/Bk58DAhrrxw==", "dev": true, "requires": { - "postcss": "^7.0.1", "timsort": "^0.3.0" } }, "css-loader": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-4.2.2.tgz", - "integrity": "sha512-omVGsTkZPVwVRpckeUnLshPp12KsmMSLqYxs12+RzM9jRR5Y+Idn/tBffjXRvOE+qW7if24cuceFJqYR5FmGBg==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/css-loader/-/css-loader-5.0.1.tgz", + "integrity": "sha512-cXc2ti9V234cq7rJzFKhirb2L2iPy8ZjALeVJAozXYz9te3r4eqLSixNAbMDJSgJEQywqXzs8gonxaboeKqwiw==", "dev": true, "requires": { - "camelcase": "^6.0.0", + "camelcase": "^6.2.0", "cssesc": "^3.0.0", - "icss-utils": "^4.1.1", + "icss-utils": "^5.0.0", "loader-utils": "^2.0.0", - "postcss": "^7.0.32", - "postcss-modules-extract-imports": "^2.0.0", - "postcss-modules-local-by-default": "^3.0.3", - "postcss-modules-scope": "^2.2.0", - "postcss-modules-values": "^3.0.0", + "postcss": "^8.1.4", + "postcss-modules-extract-imports": "^3.0.0", + "postcss-modules-local-by-default": "^4.0.0", + "postcss-modules-scope": "^3.0.0", + "postcss-modules-values": "^4.0.0", "postcss-value-parser": "^4.1.0", - "schema-utils": "^2.7.0", + "schema-utils": "^3.0.0", "semver": "^7.3.2" + }, + "dependencies": { + "camelcase": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-6.2.0.tgz", + "integrity": "sha512-c7wVvbw3f37nuobQNtgsgG9POC9qMbNuMQmTCqZv23b6MIz0fcYpBiOlv9gEN/hdLdnZTDQhg6e9Dq5M1vKvfg==", + "dev": true + }, + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, "css-parse": { @@ -4168,26 +21170,54 @@ "dev": true, "requires": { "css": "^2.0.0" + }, + "dependencies": { + "css": { + "version": "2.2.4", + "resolved": "https://registry.npmjs.org/css/-/css-2.2.4.tgz", + "integrity": "sha512-oUnjmWpy0niI3x/mPL8dVEI1l7MnG3+HHyRPHf+YFSbK+svOhXpmSOcDURUh2aOCgl2grzrOPt1nHLuCVFULLw==", + "dev": true, + "requires": { + "inherits": "^2.0.3", + "source-map": "^0.6.1", + "source-map-resolve": "^0.5.2", + "urix": "^0.1.0" + } + }, + "source-map": { + "version": "0.6.1", + "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", + "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", + "dev": true + }, + "source-map-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==", + "dev": true, + "requires": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } + } } }, "css-select": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/css-select/-/css-select-2.1.0.tgz", - "integrity": "sha512-Dqk7LQKpwLoH3VovzZnkzegqNSuAziQyNZUcrdDM401iY+R5NkGBXGmtO05/yaXQziALuPogeG0b7UAgjnTJTQ==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/css-select/-/css-select-3.1.2.tgz", + "integrity": "sha512-qmss1EihSuBNWNNhHjxzxSfJoFBM/lERB/Q4EnsJQQC62R2evJDW481091oAdOr9uh46/0n4nrg0It5cAnj1RA==", "dev": true, "requires": { "boolbase": "^1.0.0", - "css-what": "^3.2.1", - "domutils": "^1.7.0", - "nth-check": "^1.0.2" + "css-what": "^4.0.0", + "domhandler": "^4.0.0", + "domutils": "^2.4.3", + "nth-check": "^2.0.0" } }, - "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": { "version": "0.7.3", "resolved": "https://registry.npmjs.org/css-selector-tokenizer/-/css-selector-tokenizer-0.7.3.tgz", @@ -4199,12 +21229,12 @@ } }, "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==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/css-tree/-/css-tree-1.1.3.tgz", + "integrity": "sha512-tRpdppF7TRazZrjJ6v3stzv93qxRcSsFmW6cX0Zm2NVKpxE1WV1HblnghVv9TreireHkqI/VDEsfolRF1p6y7Q==", "dev": true, "requires": { - "mdn-data": "2.0.4", + "mdn-data": "2.0.14", "source-map": "^0.6.1" }, "dependencies": { @@ -4217,9 +21247,9 @@ } }, "css-what": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/css-what/-/css-what-3.4.2.tgz", - "integrity": "sha512-ACUm3L0/jiZTqfzRM3Hi9Q8eZqd6IK37mMWPLz9PJxkLWllYeRf+EHUSHYEtFop2Eqytaq1FizFVh7XfBnXCDQ==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/css-what/-/css-what-4.0.0.tgz", + "integrity": "sha512-teijzG7kwYfNVsUh2H/YN62xW3KK9YhXEgSlbxMlcyjPNvdKJqFx5lrwlJgoFP1ZHlB89iGDlo/JyshKeRhv5A==", "dev": true }, "cssauron": { @@ -4238,113 +21268,67 @@ "dev": true }, "cssnano": { - "version": "4.1.10", - "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-4.1.10.tgz", - "integrity": "sha512-5wny+F6H4/8RgNlaqab4ktc3e0/blKutmq8yNlBFXA//nSFFAqAngjNVRzUvCgYROULmZZUoosL/KSoZo5aUaQ==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/cssnano/-/cssnano-5.0.2.tgz", + "integrity": "sha512-8JK3EnPsjQsULme9/e5M2hF564f/480hwsdcHvQ7ZtAIMfQ1O3SCfs+b8Mjf5KJxhYApyRshR2QSovEJi2K72Q==", "dev": true, "requires": { - "cosmiconfig": "^5.0.0", - "cssnano-preset-default": "^4.0.7", - "is-resolvable": "^1.0.0", - "postcss": "^7.0.0" + "cosmiconfig": "^7.0.0", + "cssnano-preset-default": "^5.0.1", + "is-resolvable": "^1.1.0" } }, "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==", + "version": "5.1.3", + "resolved": "https://registry.npmjs.org/cssnano-preset-default/-/cssnano-preset-default-5.1.3.tgz", + "integrity": "sha512-qo9tX+t4yAAZ/yagVV3b+QBKeLklQbmgR3wI7mccrDcR+bEk9iHgZN1E7doX68y9ThznLya3RDmR+nc7l6/2WQ==", "dev": true, "requires": { - "css-declaration-sorter": "^4.0.1", - "cssnano-util-raw-cache": "^4.0.1", - "postcss": "^7.0.0", - "postcss-calc": "^7.0.1", - "postcss-colormin": "^4.0.3", - "postcss-convert-values": "^4.0.1", - "postcss-discard-comments": "^4.0.2", - "postcss-discard-duplicates": "^4.0.2", - "postcss-discard-empty": "^4.0.1", - "postcss-discard-overridden": "^4.0.1", - "postcss-merge-longhand": "^4.0.11", - "postcss-merge-rules": "^4.0.3", - "postcss-minify-font-values": "^4.0.2", - "postcss-minify-gradients": "^4.0.2", - "postcss-minify-params": "^4.0.2", - "postcss-minify-selectors": "^4.0.2", - "postcss-normalize-charset": "^4.0.1", - "postcss-normalize-display-values": "^4.0.2", - "postcss-normalize-positions": "^4.0.2", - "postcss-normalize-repeat-style": "^4.0.2", - "postcss-normalize-string": "^4.0.2", - "postcss-normalize-timing-functions": "^4.0.2", - "postcss-normalize-unicode": "^4.0.1", - "postcss-normalize-url": "^4.0.1", - "postcss-normalize-whitespace": "^4.0.2", - "postcss-ordered-values": "^4.1.2", - "postcss-reduce-initial": "^4.0.3", - "postcss-reduce-transforms": "^4.0.2", - "postcss-svgo": "^4.0.2", - "postcss-unique-selectors": "^4.0.1" + "css-declaration-sorter": "^6.0.3", + "cssnano-utils": "^2.0.1", + "postcss-calc": "^8.0.0", + "postcss-colormin": "^5.2.0", + "postcss-convert-values": "^5.0.1", + "postcss-discard-comments": "^5.0.1", + "postcss-discard-duplicates": "^5.0.1", + "postcss-discard-empty": "^5.0.1", + "postcss-discard-overridden": "^5.0.1", + "postcss-merge-longhand": "^5.0.2", + "postcss-merge-rules": "^5.0.2", + "postcss-minify-font-values": "^5.0.1", + "postcss-minify-gradients": "^5.0.1", + "postcss-minify-params": "^5.0.1", + "postcss-minify-selectors": "^5.1.0", + "postcss-normalize-charset": "^5.0.1", + "postcss-normalize-display-values": "^5.0.1", + "postcss-normalize-positions": "^5.0.1", + "postcss-normalize-repeat-style": "^5.0.1", + "postcss-normalize-string": "^5.0.1", + "postcss-normalize-timing-functions": "^5.0.1", + "postcss-normalize-unicode": "^5.0.1", + "postcss-normalize-url": "^5.0.2", + "postcss-normalize-whitespace": "^5.0.1", + "postcss-ordered-values": "^5.0.2", + "postcss-reduce-initial": "^5.0.1", + "postcss-reduce-transforms": "^5.0.1", + "postcss-svgo": "^5.0.2", + "postcss-unique-selectors": "^5.0.1" } }, - "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 - }, - "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 - }, - "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==", + "cssnano-utils": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/cssnano-utils/-/cssnano-utils-2.0.1.tgz", + "integrity": "sha512-i8vLRZTnEH9ubIyfdZCAdIdgnHAUeQeByEeQ2I7oTilvP9oHO6RScpeq3GsFUVqeB8uZgOQ9pw8utofNn32hhQ==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } - }, - "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 + "requires": {} }, "csso": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/csso/-/csso-4.0.3.tgz", - "integrity": "sha512-NL3spysxUkcrOgnpsT4Xdl2aiEiBG6bXswAABQVHcMrfjjBisFOKwLDOmf4wf32aPdcJws1zds2B0Rg+jqMyHQ==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/csso/-/csso-4.2.0.tgz", + "integrity": "sha512-wvlcdIbf6pwKEk7vHj8/Bkc0B4ylXZruLvOgs9doS5eOsOpuodOV2zJChSpkp+pRpYQLQMeF04nr3Z68Sta9jA==", "dev": true, "requires": { - "css-tree": "1.0.0-alpha.39" - }, - "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 - }, - "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 - } + "css-tree": "^1.1.2" } }, "custom-event": { @@ -4359,16 +21343,6 @@ "integrity": "sha1-WW6WmP0MgOEgOMK4LW6xs1tiJNk=", "dev": true }, - "d": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/d/-/d-1.0.1.tgz", - "integrity": "sha512-m62ShEObQ39CfralilEQRjH6oAMtNCV1xJyEx5LpRYUVN+EviphDgUc/F3hnYbADmkiNs67Y+3ylmlG7Lnu+FA==", - "dev": true, - "requires": { - "es5-ext": "^0.10.50", - "type": "^1.0.1" - } - }, "damerau-levenshtein": { "version": "1.0.6", "resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.6.tgz", @@ -4384,17 +21358,6 @@ "assert-plus": "^1.0.0" } }, - "data-urls": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/data-urls/-/data-urls-2.0.0.tgz", - "integrity": "sha512-X5eWTSXO/BJmpdIKCRuKUgSCgAN0OwliVK3yPKbwIWU1Tdw5BRajxlzMidvh+gwko9AfQ9zIj52pzF91Q3YAvQ==", - "dev": true, - "requires": { - "abab": "^2.0.3", - "whatwg-mimetype": "^2.3.0", - "whatwg-url": "^8.0.0" - } - }, "date-format": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/date-format/-/date-format-3.0.0.tgz", @@ -4402,23 +21365,18 @@ "dev": true }, "debug": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.2.0.tgz", - "integrity": "sha512-IX2ncY78vDTjZMFUdmsvIRFY2Cf4FnD0wRs+nQwJU8Lu99/tPFdb0VybiiMTPe3I6rQmwsqQqRBvxU+bZ/I8sg==", + "version": "4.3.1", + "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.1.tgz", + "integrity": "sha512-doEwdvm4PCeK4K3RQN2ZC2BYUBaxwLARCqZmMjtF8a51J2Rb0xpVloFRnCODwqjpwnAoao4pelN8l3RJdv3gRQ==", "requires": { "ms": "2.1.2" } }, - "debuglog": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/debuglog/-/debuglog-1.0.1.tgz", - "integrity": "sha1-qiT/uaw9+aI1GDfPstJ5NgzXhJI=", - "dev": true - }, "decamelize": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/decamelize/-/decamelize-1.2.0.tgz", - "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=" + "integrity": "sha1-9lNNFRSCabIDUue+4m9QH5oZEpA=", + "dev": true }, "decode-uri-component": { "version": "0.2.0", @@ -4457,14 +21415,6 @@ "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 - } } }, "define-properties": { @@ -4484,37 +21434,6 @@ "requires": { "is-descriptor": "^1.0.2", "isobject": "^3.0.1" - }, - "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" - } - } } }, "del": { @@ -4585,6 +21504,12 @@ "integrity": "sha1-3zrhmayt+31ECqrgsp4icrJOxhk=", "dev": true }, + "delegates": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delegates/-/delegates-1.0.0.tgz", + "integrity": "sha1-hMbhWbgZBP3KWaDvRM2HDTElD5o=", + "dev": true + }, "depd": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", @@ -4594,8 +21519,7 @@ "dependency-graph": { "version": "0.7.2", "resolved": "https://registry.npmjs.org/dependency-graph/-/dependency-graph-0.7.2.tgz", - "integrity": "sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ==", - "dev": true + "integrity": "sha512-KqtH4/EZdtdfWX0p6MGP9jljvxSY6msy/pRUD4jgNwVpv3v1QmNLlsB3LDSSUg79BRVSn7jI1QPRtArGABovAQ==" }, "des.js": { "version": "1.0.1", @@ -4614,21 +21538,11 @@ "dev": true }, "detect-node": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.0.4.tgz", - "integrity": "sha512-ZIzRpLJrOj7jjP2miAtgqIfmzbxa4ZOr5jJc601zklsfEx9oTzmmj2nVpIPRpNlRTIh8lc1kyViIY7BWSGNmKw==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/detect-node/-/detect-node-2.1.0.tgz", + "integrity": "sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g==", "dev": true }, - "dezalgo": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/dezalgo/-/dezalgo-1.0.3.tgz", - "integrity": "sha1-f3Qt4Gb8dIvI24IFad3c5Jvw1FY=", - "dev": true, - "requires": { - "asap": "^2.0.0", - "wrappy": "1" - } - }, "di": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/di/-/di-0.0.1.tgz", @@ -4653,9 +21567,9 @@ }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -4676,9 +21590,9 @@ "dev": true }, "dns-packet": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.1.tgz", - "integrity": "sha512-0UxfQkMhYAUaZI+xrNZOz/as5KgDU0M/fQ9b6SpkyLbk3GEswDi6PADJVaYJradtRVsRIlF1zLyOodbcTCDzUg==", + "version": "1.3.4", + "resolved": "https://registry.npmjs.org/dns-packet/-/dns-packet-1.3.4.tgz", + "integrity": "sha512-BQ6F4vycLXBvdrJZ6S3gZewt6rcrks9KBgM9vrhW+knGRqc8uEdT7fuCwloc7nny5xNoMJ17HGH0R/6fpo8ECA==", "dev": true, "requires": { "ip": "^1.1.0", @@ -4707,21 +21621,14 @@ } }, "dom-serializer": { - "version": "0.2.2", - "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-0.2.2.tgz", - "integrity": "sha512-2/xPb3ORsQ42nHYiSunXkDjPLBaEj/xTwUO4B7XCZQTRk7EBtTOPaygh10YAAh2OI1Qrp6NWfpAhzswj0ydt9g==", + "version": "1.3.2", + "resolved": "https://registry.npmjs.org/dom-serializer/-/dom-serializer-1.3.2.tgz", + "integrity": "sha512-5c54Bk5Dw4qAxNOI1pFEizPSjVsx5+bpJKmL2kPn8JhBUq2q09tTCa3mjijun2NfK78NMouDYNMBkOrPZiS+ig==", "dev": true, "requires": { "domelementtype": "^2.0.1", + "domhandler": "^4.2.0", "entities": "^2.0.0" - }, - "dependencies": { - "domelementtype": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.0.2.tgz", - "integrity": "sha512-wFwTwCVebUrMgGeAwRL/NhZtHAUyT9n9yg4IMDwf10+6iCMxSkVq9MGCVEH+QZWo1nNidy8kNvwmv4zWHDTqvA==", - "dev": true - } } }, "domain-browser": { @@ -4731,28 +21638,29 @@ "dev": true }, "domelementtype": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-1.3.1.tgz", - "integrity": "sha512-BSKB+TSpMpFI/HOxCNr1O8aMOTZ8hT3pM3GQ0w/mWRmkhEDSFJkkyzz4XQsBV44BChwGkrDfMyjVD0eA2aFV3w==", + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/domelementtype/-/domelementtype-2.2.0.tgz", + "integrity": "sha512-DtBMo82pv1dFtUmHyr48beiuq792Sxohr+8Hm9zoxklYPfa6n0Z3Byjj2IV7bmr2IyqClnqEQhfgHJJ5QF0R5A==", "dev": true }, - "domutils": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/domutils/-/domutils-1.7.0.tgz", - "integrity": "sha512-Lgd2XcJ/NjEw+7tFvfKxOzCYKZsdct5lczQ2ZaQY8Djz7pfAD3Gbp8ySJWtreII/vDlMVmxwa6pHmdxIYgttDg==", + "domhandler": { + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/domhandler/-/domhandler-4.2.0.tgz", + "integrity": "sha512-zk7sgt970kzPks2Bf+dwT/PLzghLnsivb9CcxkvR8Mzr66Olr0Ofd8neSbglHJHaHa2MadfoSdNlKYAaafmWfA==", "dev": true, "requires": { - "dom-serializer": "0", - "domelementtype": "1" + "domelementtype": "^2.2.0" } }, - "dot-prop": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/dot-prop/-/dot-prop-5.3.0.tgz", - "integrity": "sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q==", + "domutils": { + "version": "2.7.0", + "resolved": "https://registry.npmjs.org/domutils/-/domutils-2.7.0.tgz", + "integrity": "sha512-8eaHa17IwJUPAiB+SoTYBo5mCdeMgdcAoXJ59m6DT1vw+5iLS3gNoqYaRowaBKtGVrOF1Jz4yDTgYKLK2kvfJg==", "dev": true, "requires": { - "is-obj": "^2.0.0" + "dom-serializer": "^1.0.1", + "domelementtype": "^2.2.0", + "domhandler": "^4.2.0" } }, "duplexify": { @@ -4784,30 +21692,30 @@ "dev": true }, "electron-to-chromium": { - "version": "1.3.583", - "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.583.tgz", - "integrity": "sha512-L9BwLwJohjZW9mQESI79HRzhicPk1DFgM+8hOCfGgGCFEcA3Otpv7QK6SGtYoZvfQfE3wKLh0Hd5ptqUFv3gvQ==", + "version": "1.3.752", + "resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.3.752.tgz", + "integrity": "sha512-2Tg+7jSl3oPxgsBsWKh5H83QazTkmWG/cnNwJplmyZc7KcN61+I10oUgaXSVk/NwfvN3BdkKDR4FYuRBQQ2v0A==", "dev": true }, "elliptic": { - "version": "6.5.3", - "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.3.tgz", - "integrity": "sha512-IMqzv5wNQf+E6aHeIqATs0tOLeOTwj1QKbRcS3jBbYkl5oLAserA8yJTT7/VyHUYG91PRmPyeQDObKLPpeS4dw==", + "version": "6.5.4", + "resolved": "https://registry.npmjs.org/elliptic/-/elliptic-6.5.4.tgz", + "integrity": "sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==", "dev": true, "requires": { - "bn.js": "^4.4.0", - "brorand": "^1.0.1", + "bn.js": "^4.11.9", + "brorand": "^1.1.0", "hash.js": "^1.0.0", - "hmac-drbg": "^1.0.0", - "inherits": "^2.0.1", - "minimalistic-assert": "^1.0.0", - "minimalistic-crypto-utils": "^1.0.0" + "hmac-drbg": "^1.0.1", + "inherits": "^2.0.4", + "minimalistic-assert": "^1.0.1", + "minimalistic-crypto-utils": "^1.0.1" }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -4835,8 +21743,21 @@ "resolved": "https://registry.npmjs.org/encoding/-/encoding-0.1.13.tgz", "integrity": "sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A==", "dev": true, + "optional": true, "requires": { "iconv-lite": "^0.6.2" + }, + "dependencies": { + "iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "optional": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + } + } } }, "end-of-stream": { @@ -4849,121 +21770,52 @@ } }, "engine.io": { - "version": "3.4.2", - "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-3.4.2.tgz", - "integrity": "sha512-b4Q85dFkGw+TqgytGPrGgACRUhsdKc9S9ErRAXpPGy/CXKs4tYoHDkvIRdsseAF7NjfVwjRFIn6KTnbw7LwJZg==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/engine.io/-/engine.io-4.1.1.tgz", + "integrity": "sha512-t2E9wLlssQjGw0nluF6aYyfX8LwYU8Jj0xct+pAhfWfv/YrBn6TSNtEYsgxHIfaMqfrLx07czcMg9bMN6di+3w==", "dev": true, "requires": { "accepts": "~1.3.4", "base64id": "2.0.0", - "cookie": "0.3.1", - "debug": "~4.1.0", - "engine.io-parser": "~2.2.0", - "ws": "^7.1.2" + "cookie": "~0.4.1", + "cors": "~2.8.5", + "debug": "~4.3.1", + "engine.io-parser": "~4.0.0", + "ws": "~7.4.2" }, "dependencies": { "cookie": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz", - "integrity": "sha1-5+Ch+e9DtMi6klxcWpboBtFoc7s=", - "dev": true - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "ws": { - "version": "7.3.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-7.3.1.tgz", - "integrity": "sha512-D3RuNkynyHmEJIpD2qrgVkc9DQ23OrN/moAwZX4L8DfvszsJxpjQuUq3LMx6HoYji9fbIOBY18XWBsAux1ZZUA==", - "dev": true - } - } - }, - "engine.io-client": { - "version": "3.4.4", - "resolved": "https://registry.npmjs.org/engine.io-client/-/engine.io-client-3.4.4.tgz", - "integrity": "sha512-iU4CRr38Fecj8HoZEnFtm2EiKGbYZcPn3cHxqNGl/tmdWRf60KhK+9vE0JeSjgnlS/0oynEfLgKbT9ALpim0sQ==", - "dev": true, - "requires": { - "component-emitter": "~1.3.0", - "component-inherit": "0.0.3", - "debug": "~3.1.0", - "engine.io-parser": "~2.2.0", - "has-cors": "1.1.0", - "indexof": "0.0.1", - "parseqs": "0.0.6", - "parseuri": "0.0.6", - "ws": "~6.1.0", - "xmlhttprequest-ssl": "~1.5.4", - "yeast": "0.1.2" - }, - "dependencies": { - "debug": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", - "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - }, - "parseqs": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.6.tgz", - "integrity": "sha512-jeAGzMDbfSHHA091hr0r31eYfTig+29g3GKKE/PPbEQ65X0lmMwlEoqmhzu0iztID5uJpZsFlUPDP8ThPL7M8w==", - "dev": true - }, - "parseuri": { - "version": "0.0.6", - "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.6.tgz", - "integrity": "sha512-AUjen8sAkGgao7UyCX6Ahv0gIK2fABKmYjvP4xmy5JaKvcbTRueIqIPHLAfq30xJddqSE033IOMUSOMCcK3Sow==", + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.1.tgz", + "integrity": "sha512-ZwrFkGJxUR3EIoXtO+yVE69Eb7KlixbaeAWfBQB9vVsNn/o+Yw69gBWSSDK825hQNdN+wF8zELf3dFNl/kxkUA==", "dev": true }, "ws": { - "version": "6.1.4", - "resolved": "https://registry.npmjs.org/ws/-/ws-6.1.4.tgz", - "integrity": "sha512-eqZfL+NE/YQc1/ZynhojeV8q+H050oR8AZ2uIev7RU10svA9ZnJUddHcOUZTJLinZ9yEfdA2kSATS2qZK5fhJA==", + "version": "7.4.6", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.4.6.tgz", + "integrity": "sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==", "dev": true, - "requires": { - "async-limiter": "~1.0.0" - } + "requires": {} } } }, "engine.io-parser": { - "version": "2.2.1", - "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-2.2.1.tgz", - "integrity": "sha512-x+dN/fBH8Ro8TFwJ+rkB2AmuVw9Yu2mockR/p3W8f8YtExwFgDvBDi0GWyb4ZLkpahtDGZgtr3zLovanJghPqg==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/engine.io-parser/-/engine.io-parser-4.0.2.tgz", + "integrity": "sha512-sHfEQv6nmtJrq6TKuIz5kyEKH/qSdK56H/A+7DnAuUPWosnIZAS2NHNcPLmyjtY3cGS/MqJdZbUjW97JU72iYg==", "dev": true, "requires": { - "after": "0.8.2", - "arraybuffer.slice": "~0.0.7", - "base64-arraybuffer": "0.1.4", - "blob": "0.0.5", - "has-binary2": "~1.0.2" + "base64-arraybuffer": "0.1.4" } }, "enhanced-resolve": { - "version": "4.3.0", - "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.3.0.tgz", - "integrity": "sha512-3e87LvavsdxyoCfGusJnrZ5G8SLPOFeHSNpZI/ATL9a5leXo2k0w6MKnbqhdBad9qTobSfB20Ld7UmgoNbAZkQ==", + "version": "5.7.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-5.7.0.tgz", + "integrity": "sha512-6njwt/NsZFUKhM6j9U8hzVyD4E4r0x7NQzhTCbcWOJ0IQjNSAoalWmb0AE51Wn+fwan5qVESWi7t2ToBxs9vrw==", "dev": true, "requires": { - "graceful-fs": "^4.1.2", - "memory-fs": "^0.5.0", - "tapable": "^1.0.0" + "graceful-fs": "^4.2.4", + "tapable": "^2.2.0" } }, "ent": { @@ -4973,9 +21825,15 @@ "dev": true }, "entities": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/entities/-/entities-2.1.0.tgz", - "integrity": "sha512-hCx1oky9PFrJ611mf0ifBLBRW8lUUVRlFolb5gWRfIELabBlbp9xZvrqZLZAs+NxFnbfQoeGd8wDkygjg7U85w==", + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/entities/-/entities-2.2.0.tgz", + "integrity": "sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==", + "dev": true + }, + "env-paths": { + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/env-paths/-/env-paths-2.2.1.tgz", + "integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==", "dev": true }, "err-code": { @@ -4985,9 +21843,9 @@ "dev": true }, "errno": { - "version": "0.1.7", - "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.7.tgz", - "integrity": "sha512-MfrRBDWzIWifgq6tJj60gkAwtLNb6sQPlcFrSOflcP1aFmmruKQ2wRnze/8V6kgyz7H3FF8Npzv78mZ7XLLflg==", + "version": "0.1.8", + "resolved": "https://registry.npmjs.org/errno/-/errno-0.1.8.tgz", + "integrity": "sha512-dJ6oBr5SQ1VSd9qkk7ByRgb/1SH4JZjCHSW/mr63/QcXO9zLVxvJ6Oy13nio03rxpSnVDDjFor75SjVeZWPW/A==", "dev": true, "requires": { "prr": "~1.0.1" @@ -5002,59 +21860,6 @@ "is-arrayish": "^0.2.1" } }, - "es-abstract": { - "version": "1.18.0-next.1", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.18.0-next.1.tgz", - "integrity": "sha512-I4UGspA0wpZXWENrdA0uHbnhte683t3qT/1VFH9aX2dA5PPSf6QW5HHXf5HImaqPmjXaVeVk4RGWnaylmV7uAA==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-negative-zero": "^2.0.0", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - }, - "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": { - "is-callable": "^1.1.4", - "is-date-object": "^1.0.1", - "is-symbol": "^1.0.2" - } - }, - "es5-ext": { - "version": "0.10.53", - "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.53.tgz", - "integrity": "sha512-Xs2Stw6NiNHWypzRTY1MtaG/uJlwCk8kH81920ma8mvN8Xq1gsfhZvpkImLQArw8AHnv8MT2I45J3c0R8slE+Q==", - "dev": true, - "requires": { - "es6-iterator": "~2.0.3", - "es6-symbol": "~3.1.3", - "next-tick": "~1.0.0" - } - }, - "es6-iterator": { - "version": "2.0.3", - "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz", - "integrity": "sha1-p96IkUGgWpSwhUQDstCg+/qY87c=", - "dev": true, - "requires": { - "d": "1", - "es5-ext": "^0.10.35", - "es6-symbol": "^3.1.1" - } - }, "es6-promise": { "version": "4.2.8", "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", @@ -5070,21 +21875,10 @@ "es6-promise": "^4.0.3" } }, - "es6-symbol": { - "version": "3.1.3", - "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.3.tgz", - "integrity": "sha512-NJ6Yn3FuDinBaBRWl/q5X/s4koRHBrgKAu+yGI6JCBeiu3qrcbJhwT2GeR/EXVfylRk8dpQVJoLEFhK+Mu31NA==", - "dev": true, - "requires": { - "d": "^1.0.1", - "ext": "^1.1.2" - } - }, "escalade": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", - "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==", - "dev": true + "integrity": "sha512-k0er2gUkLf8O0zKJiAhmkTnJlTvINGv7ygDNPbeIsX/TJjGJZHuh9B2UxbsaEkmlEo9MfhrSzmhIlhRlI2GXnw==" }, "escape-html": { "version": "1.0.3", @@ -5155,15 +21949,15 @@ "dev": true }, "events": { - "version": "3.2.0", - "resolved": "https://registry.npmjs.org/events/-/events-3.2.0.tgz", - "integrity": "sha512-/46HWwbfCX2xTawVfkKLGxMifJYQBWMwY1mjywRtb4c9x8l5NP3KoJtnIOiL1hfdRkIuYhETxQlo62IF8tcnlg==", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/events/-/events-3.3.0.tgz", + "integrity": "sha512-mQw+2fkQbALzQ7V0MY0IqdnXNOeTtP4r0lN9z7AAawCXgqea7bDii20AYrIBrFd/Hx0M2Ocz6S111CaFkUcb0Q==", "dev": true }, "eventsource": { - "version": "1.0.7", - "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-1.0.7.tgz", - "integrity": "sha512-4Ln17+vVT0k8aWq+t/bF5arcS3EpT9gYtW66EPacdj/mAFevznsnyoHLPy2BA8gbIQeIHoPsvwmfBftfcG//BQ==", + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/eventsource/-/eventsource-1.1.0.tgz", + "integrity": "sha512-VSJjT5oCNrFvCS6igjzPAt5hBzQ2qPBFIbJ03zLI9SE0mxwZpMw6BfJrbFHm1a141AavMEB8JHmBhWAd66PfCg==", "dev": true, "requires": { "original": "^1.0.0" @@ -5242,6 +22036,69 @@ "is-extendable": "^0.1.0" } }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-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": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "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 + }, + "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 + }, "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -5311,23 +22168,6 @@ } } }, - "ext": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/ext/-/ext-1.4.0.tgz", - "integrity": "sha512-Key5NIsUxdqKg3vIsdw9dSuXpPCQ297y6wBjL30edxwPgt2E44WcWBZey/ZvUc6sERLTxKdyCu4gZFmUbk1Q7A==", - "dev": true, - "requires": { - "type": "^2.0.0" - }, - "dependencies": { - "type": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/type/-/type-2.1.0.tgz", - "integrity": "sha512-G9absDWvhAWCV2gmF1zKud3OyC61nZDwWvBL2DApaVFogI07CprggiQAOOjvp2NRjYWFzPyu7vwtDrQFq8jeSA==", - "dev": true - } - } - }, "extend": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/extend/-/extend-3.0.2.tgz", @@ -5342,17 +22182,6 @@ "requires": { "assign-symbols": "^1.0.0", "is-extendable": "^1.0.1" - }, - "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" - } - } } }, "external-editor": { @@ -5364,17 +22193,6 @@ "chardet": "^0.7.0", "iconv-lite": "^0.4.24", "tmp": "^0.0.33" - }, - "dependencies": { - "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": { - "safer-buffer": ">= 2.1.2 < 3" - } - } } }, "extglob": { @@ -5411,34 +22229,11 @@ "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": { + "version": "0.1.1", + "resolved": "https://registry.npmjs.org/is-extendable/-/is-extendable-0.1.1.tgz", + "integrity": "sha1-YrEQ4omkcUGOPsNqYX1HLjAd/Ik=", + "dev": true } } }, @@ -5455,9 +22250,9 @@ "dev": true }, "fast-glob": { - "version": "3.2.4", - "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.4.tgz", - "integrity": "sha512-kr/Oo6PX51265qeuCYsyGypiO5uJFgBS0jksyG7FUeCyQzNwYnzrNIMR1NXfkZXsMYXYLRAHgISHBz8gQcxKHQ==", + "version": "3.2.5", + "resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.2.5.tgz", + "integrity": "sha512-2DtFcgT68wiTTiwZ2hNdJfcHNke9XOfnwmBRWXhmeKM8rF0TGwmC/Qto3S7RoZKp5cilZbxzO5iTNTQsJ+EeDg==", "dev": true, "requires": { "@nodelib/fs.stat": "^2.0.2", @@ -5481,18 +22276,18 @@ "dev": true }, "fastq": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.8.0.tgz", - "integrity": "sha512-SMIZoZdLh/fgofivvIkmknUXyPnvxRE3DhtZ5Me3Mrsk5gyPL42F0xr51TdRXskBxHfMp+07bcYzfsYEsSQA9Q==", + "version": "1.11.0", + "resolved": "https://registry.npmjs.org/fastq/-/fastq-1.11.0.tgz", + "integrity": "sha512-7Eczs8gIPDrVzT+EksYBcupqMyxSHXXrHOLRRxU2/DicV8789MRBRR8+Hc2uWzUupOs4YS4JzBmBxjjCVBxD/g==", "dev": true, "requires": { "reusify": "^1.0.4" } }, "faye-websocket": { - "version": "0.10.0", - "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.10.0.tgz", - "integrity": "sha1-TkkvjQTftviQA1B/btvy1QHnxvQ=", + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.4.tgz", + "integrity": "sha512-CzbClwlXAuiRQAlUyfqPgvPoNKTckTPGfwZV4ZdAhVcP2lh9KUxJg2b5GkE7XbjKQ3YJnQ9z6D9ntLAlB+tP8g==", "dev": true, "requires": { "websocket-driver": ">=0.5.1" @@ -5514,15 +22309,33 @@ } }, "file-loader": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.0.0.tgz", - "integrity": "sha512-/aMOAYEFXDdjG0wytpTL5YQLfZnnTmLNjn+AIrJ/6HVnTfDqLsVKUUwkDf4I4kgex36BvjuXEn/TX9B/1ESyqQ==", + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/file-loader/-/file-loader-6.2.0.tgz", + "integrity": "sha512-qo3glqyTa61Ytg4u73GultjHGjdRyig3tG6lPtyX/jOEJvHif9uB0/OCI2Kif6ctF3caQTW2G5gym21oAsI4pw==", "dev": true, "requires": { "loader-utils": "^2.0.0", - "schema-utils": "^2.6.5" + "schema-utils": "^3.0.0" + }, + "dependencies": { + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, + "file-saver": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/file-saver/-/file-saver-2.0.5.tgz", + "integrity": "sha512-P9bmyZ3h/PRG+Nzga+rbdI4OEpNDzAVyy74uVO9ATgzLK6VtAsYybF/+TOCvrc0MO793d6+42lLyZTw7/ArVzA==" + }, "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", @@ -5534,7 +22347,6 @@ "version": "7.0.1", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz", "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==", - "dev": true, "requires": { "to-regex-range": "^5.0.1" } @@ -5582,25 +22394,6 @@ "pkg-dir": "^4.1.0" }, "dependencies": { - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, "make-dir": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-3.1.0.tgz", @@ -5610,30 +22403,6 @@ "semver": "^6.0.0" } }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true - }, - "pkg-dir": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", - "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", - "dev": true, - "requires": { - "find-up": "^4.0.0" - } - }, "semver": { "version": "6.3.0", "resolved": "https://registry.npmjs.org/semver/-/semver-6.3.0.tgz", @@ -5643,12 +22412,13 @@ } }, "find-up": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-3.0.0.tgz", - "integrity": "sha512-1yD6RmLI1XBfxugvORwlck6f75tYL+iR0jqwsOrOxMZyGYqUuDhJ0l4AXdO1iX/FTs9cBAMEk1gWSEx1kSbylg==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", + "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", "dev": true, "requires": { - "locate-path": "^3.0.0" + "locate-path": "^5.0.0", + "path-exists": "^4.0.0" } }, "flatted": { @@ -5697,9 +22467,15 @@ } }, "forwarded": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", - "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=", + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz", + "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==", + "dev": true + }, + "fraction.js": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-4.1.1.tgz", + "integrity": "sha512-MHOhvvxHTfRFpF1geTK9czMIZ6xclsEor2wkIGYYq+PxcQqT7vStJqjhe6S1TenZrMZzo+wlqOufBDVepUEgPg==", "dev": true }, "fragment-cache": { @@ -5731,7 +22507,6 @@ "version": "4.0.2", "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-4.0.2.tgz", "integrity": "sha1-+RcExT0bRh+JNFKwwwfZmXZHq2s=", - "dev": true, "requires": { "graceful-fs": "^4.1.2", "jsonfile": "^4.0.0", @@ -5768,20 +22543,50 @@ "version": "2.1.3", "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz", "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==", - "dev": true, "optional": true }, "function-bind": { "version": "1.1.1", "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.1.tgz", - "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==", - "dev": true + "integrity": "sha512-yIovAzMX49sF8Yl58fSCWJ5svSLuaibPxXQJFLmBObTuCr0Mf1KiPopGM9NiFjiYBCbfaa2Fh6breQ6ANVTI0A==" }, - "genfun": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/genfun/-/genfun-5.0.0.tgz", - "integrity": "sha512-KGDOARWVga7+rnB3z9Sd2Letx515owfk0hSxHGuqjANb1M+x2bGZGqHLiozPsYMdM2OubeMni/Hpwmjq6qIUhA==", - "dev": true + "gauge": { + "version": "2.7.4", + "resolved": "https://registry.npmjs.org/gauge/-/gauge-2.7.4.tgz", + "integrity": "sha1-LANAXHU4w51+s3sxcCLjJfsBi/c=", + "dev": 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" + }, + "dependencies": { + "is-fullwidth-code-point": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", + "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", + "dev": true, + "requires": { + "number-is-nan": "^1.0.0" + } + }, + "string-width": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", + "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", + "dev": true, + "requires": { + "code-point-at": "^1.0.0", + "is-fullwidth-code-point": "^1.0.0", + "strip-ansi": "^3.0.0" + } + } + } }, "gensync": { "version": "1.0.0-beta.1", @@ -5793,6 +22598,17 @@ "resolved": "https://registry.npmjs.org/get-caller-file/-/get-caller-file-2.0.5.tgz", "integrity": "sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==" }, + "get-intrinsic": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.1.1.tgz", + "integrity": "sha512-kWZrnVM42QCiEA2Ig1bG8zjoIMOgxWwYCEeNdwY6Tv/cOSeGpcoX4pXHfKUxNKVoArnrEr2e9srnAxxGIraS9Q==", + "dev": true, + "requires": { + "function-bind": "^1.1.1", + "has": "^1.0.3", + "has-symbols": "^1.0.1" + } + }, "get-stream": { "version": "4.1.0", "resolved": "https://registry.npmjs.org/get-stream/-/get-stream-4.1.0.tgz", @@ -5832,10 +22648,9 @@ } }, "glob-parent": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz", - "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==", - "dev": true, + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz", + "integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==", "requires": { "is-glob": "^4.0.1" } @@ -5846,9 +22661,9 @@ "integrity": "sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA==" }, "globby": { - "version": "11.0.1", - "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.1.tgz", - "integrity": "sha512-iH9RmgwCmUJHi2z5o2l3eTtGBtXek1OYlHrbcxOYugyHLmAsZrPj43OtHThd62Buh/Vv6VyCBD2bdyWcGNQqoQ==", + "version": "11.0.3", + "resolved": "https://registry.npmjs.org/globby/-/globby-11.0.3.tgz", + "integrity": "sha512-ffdmosjA807y7+lA1NM0jELARVmYul/715xiILEjo3hBLPTcirgQNnXECn5g3mtR8TOLCVbkfua1Hpen25/Xcg==", "dev": true, "requires": { "array-union": "^2.1.0", @@ -5862,8 +22677,7 @@ "graceful-fs": { "version": "4.2.4", "resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.4.tgz", - "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==", - "dev": true + "integrity": "sha512-WjKPNJF79dtJAVniUlGGWHYGz2jWxT6VhN/4m1NdkbZ2nOsEF+cI1Edgql5zCRhs/VsQYRvrXctxktVXZUkixw==" }, "handle-thing": { "version": "2.0.1", @@ -5891,7 +22705,6 @@ "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" } @@ -5905,38 +22718,21 @@ "ansi-regex": "^2.0.0" } }, - "has-binary2": { - "version": "1.0.3", - "resolved": "https://registry.npmjs.org/has-binary2/-/has-binary2-1.0.3.tgz", - "integrity": "sha512-G1LWKhDSvhGeAQ8mPVQlqNcOB2sJdwATtZKl2pDKKHfpf/rYj24lkinxf69blJbnsvtqqNU+L3SL50vzZhXOnw==", - "dev": true, - "requires": { - "isarray": "2.0.1" - }, - "dependencies": { - "isarray": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", - "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", - "dev": true - } - } - }, - "has-cors": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/has-cors/-/has-cors-1.1.0.tgz", - "integrity": "sha1-XkdHk/fqmEPRu5nCPu9J/xJv/zk=", - "dev": true - }, "has-flag": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz", "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=" }, "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==", + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.0.2.tgz", + "integrity": "sha512-chXa79rL/UC2KlX17jo3vRGz0azaWEx5tGqZg5pO3NUyEJVB17dMruQlzCCOfUvElghKcm5194+BCRvi2Rv/Gw==", + "dev": true + }, + "has-unicode": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/has-unicode/-/has-unicode-2.0.1.tgz", + "integrity": "sha1-4Ob+aijPUROIVeCG0Wkedx3iqLk=", "dev": true }, "has-value": { @@ -6049,9 +22845,9 @@ } }, "hosted-git-info": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-3.0.5.tgz", - "integrity": "sha512-i4dpK6xj9BIpVOTboXIlKG9+8HMKggcrMX7WA24xZtKwX0TPelq/rbaS5rCKeNX8sJXZJGdSxpnEGtta+wismQ==", + "version": "3.0.8", + "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-3.0.8.tgz", + "integrity": "sha512-aXpmwoOhRBrw6X3j0h5RloK4x1OzsxMPyxqIHyNfSe2pypkVTZFpEiRoSipPEPlMrh0HW/XsjkJ5WgnCirpNUw==", "dev": true, "requires": { "lru-cache": "^6.0.0" @@ -6081,16 +22877,10 @@ "integrity": "sha1-wc56MWjIxmFAM6S194d/OyJfnDg=", "dev": true }, - "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-entities": { - "version": "1.3.1", - "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-1.3.1.tgz", - "integrity": "sha512-rhE/4Z3hIhzHAUKbW8jVcCyuT5oJCXXqhN/6mXXVCpzTmvJnoH2HL/bt3EZ6p55jbFJBeAe1ZNpL5BugLujxNA==", + "version": "1.4.0", + "resolved": "https://registry.npmjs.org/html-entities/-/html-entities-1.4.0.tgz", + "integrity": "sha512-8nxjcBcd8wovbeKx7h3wTji4e6+rhaVuPNpMqwWgnHh+N9ToqsCs6XztWRBPQ+UtzsoMAdKZtUENoVzU/EMtZA==", "dev": true }, "html-escaper": { @@ -6100,9 +22890,9 @@ "dev": true }, "http-cache-semantics": { - "version": "3.8.1", - "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-3.8.1.tgz", - "integrity": "sha512-5ai2iksyV8ZXmnZhHH4rWPoxxistEexSi5936zIQ1bnNTW5VnA85B6P/VpXiRM017IgRvb2kKo1a//y+0wSp3w==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz", + "integrity": "sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ==", "dev": true }, "http-deceiver": { @@ -6132,6 +22922,12 @@ } } }, + "http-parser-js": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/http-parser-js/-/http-parser-js-0.5.3.tgz", + "integrity": "sha512-t7hjvef/5HEK7RWTdUzVUhl8zkEu+LlaE0IYzdMuvbSDipxBRpOn4Uhw8ZyECEa808iVT8XCjzo6xmYt4CiLZg==", + "dev": true + }, "http-proxy": { "version": "1.18.1", "resolved": "https://registry.npmjs.org/http-proxy/-/http-proxy-1.18.1.tgz", @@ -6144,29 +22940,24 @@ } }, "http-proxy-agent": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-2.1.0.tgz", - "integrity": "sha512-qwHbBLV7WviBl0rQsOzH6o5lwyOIvwp/BdFnvVxXORldu5TmjFfjzBcWUWS5kWAZhmv+JtiDhSuQCp4sBfbIgg==", + "version": "4.0.1", + "resolved": "https://registry.npmjs.org/http-proxy-agent/-/http-proxy-agent-4.0.1.tgz", + "integrity": "sha512-k0zdNgqWTGA6aeIRVpvfVob4fL52dTfaehylg0Y4UvSySvOq/Y+BOyPrgpUrA7HylqvU8vIZGsRuXmspskV0Tg==", "dev": true, "requires": { - "agent-base": "4", - "debug": "3.1.0" + "@tootallnate/once": "1", + "agent-base": "6", + "debug": "4" }, "dependencies": { - "debug": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", - "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", + "agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", "dev": true, "requires": { - "ms": "2.0.0" + "debug": "4" } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true } } }, @@ -6234,6 +23025,12 @@ } } }, + "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 + }, "is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", @@ -6335,22 +23132,20 @@ } }, "iconv-lite": { - "version": "0.6.2", - "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.2.tgz", - "integrity": "sha512-2y91h5OpQlolefMPmUlivelittSWy0rP+oYVpn6A7GwVHNE8AWzoYOBNmlwks3LobaJxgHCYZAnyNo2GgpNRNQ==", + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", "dev": true, "requires": { - "safer-buffer": ">= 2.1.2 < 3.0.0" + "safer-buffer": ">= 2.1.2 < 3" } }, "icss-utils": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-4.1.1.tgz", - "integrity": "sha512-4aFq7wvWyMHKgxsH8QQtGpvbASCf+eM3wPRLI6R+MgAnTCZ6STYsRvttLvRWK0Nfif5piF394St3HeJDaljGPA==", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/icss-utils/-/icss-utils-5.1.0.tgz", + "integrity": "sha512-soFhflCVWLfRNOPU3iv5Z9VUdT44xFRbzjLsEzSr5AQmgqPMTHdU3PMT1Cf1ssx8fLNJDA1juftYl+PUcv3MqA==", "dev": true, - "requires": { - "postcss": "^7.0.14" - } + "requires": {} }, "ieee754": { "version": "1.1.13", @@ -6371,9 +23166,9 @@ "dev": true }, "ignore-walk": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.3.tgz", - "integrity": "sha512-m7o6xuOaT1aqheYHKf8W6J5pYH85ZI9w077erOzLje3JsB1gkafkAhHHY19dqjulgIZHFm32Cp5uNZgcQqdJKw==", + "version": "3.0.4", + "resolved": "https://registry.npmjs.org/ignore-walk/-/ignore-walk-3.0.4.tgz", + "integrity": "sha512-PY6Ii8o1jMRA1z4F2hRkH/xN59ox43DavKvD3oDpfurRlOJyAHpifIwpbdv1n4jt4ov0jSpw3kQ4GhJnpBL6WQ==", "dev": true, "requires": { "minimatch": "^3.0.4" @@ -6392,32 +23187,14 @@ "integrity": "sha1-nbHb0Pr43m++D13V5Wu2BigN5ps=", "dev": true }, - "import-cwd": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/import-cwd/-/import-cwd-2.1.0.tgz", - "integrity": "sha1-qmzzbnInYShcs3HsZRn1PiQ1sKk=", - "dev": true, - "requires": { - "import-from": "^2.1.0" - } - }, "import-fresh": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-2.0.0.tgz", - "integrity": "sha1-2BNVwVYS04bGH53dOSLUMEgipUY=", + "version": "3.3.0", + "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", + "integrity": "sha512-veYYhQa+D1QBKznvhUHxb8faxlrwUnxseDAbAp457E0wLNio2bOSKnjYDhMj+YiAq61xrMGhQk9iXVk5FzgQMw==", "dev": true, "requires": { - "caller-path": "^2.0.0", - "resolve-from": "^3.0.0" - } - }, - "import-from": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/import-from/-/import-from-2.1.0.tgz", - "integrity": "sha1-M1238qev/VOqpHHUuAId7ja387E=", - "dev": true, - "requires": { - "resolve-from": "^3.0.0" + "parent-module": "^1.0.0", + "resolve-from": "^4.0.0" } }, "import-local": { @@ -6428,6 +23205,51 @@ "requires": { "pkg-dir": "^3.0.0", "resolve-cwd": "^2.0.0" + }, + "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-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" + } + }, + "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 + }, + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + } + } } }, "imurmurhash": { @@ -6442,18 +23264,6 @@ "integrity": "sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg==", "dev": true }, - "indexes-of": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/indexes-of/-/indexes-of-1.0.1.tgz", - "integrity": "sha1-8w9xbI4r00bHtn0985FVZqfAVgc=", - "dev": true - }, - "indexof": { - "version": "0.0.1", - "resolved": "https://registry.npmjs.org/indexof/-/indexof-0.0.1.tgz", - "integrity": "sha1-gtwzbSMrkGIXnQWrMpOmYFn9Q10=", - "dev": true - }, "infer-owner": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/infer-owner/-/infer-owner-1.0.4.tgz", @@ -6475,9 +23285,9 @@ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==" }, "ini": { - "version": "1.3.5", - "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.5.tgz", - "integrity": "sha512-RZY5huIKCMRWDUqZlEi72f/lmXKMvuszcMBduliQ3nnWbx9X/ZBQO7DijMEYS9EhHBb2qacRUMtC7svLwe0lcw==", + "version": "1.3.8", + "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz", + "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==", "dev": true }, "inquirer": { @@ -6600,15 +23410,6 @@ "ipaddr.js": "^1.9.0" } }, - "invariant": { - "version": "2.2.4", - "resolved": "https://registry.npmjs.org/invariant/-/invariant-2.2.4.tgz", - "integrity": "sha512-phJfQVBuaJM5raOpJjSfkiD6BpbCE4Ns//LaXl6wGYtUBY83nWS6Rf9tXm2e8VaK60JEjYldbPif/A2B1C2gNA==", - "dev": true, - "requires": { - "loose-envify": "^1.0.0" - } - }, "ip": { "version": "1.1.5", "resolved": "https://registry.npmjs.org/ip/-/ip-1.1.5.tgz", @@ -6628,36 +23429,28 @@ "dev": true }, "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=", + "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 }, "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=", + "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": "^3.0.2" - }, - "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" - } - } + "kind-of": "^6.0.0" } }, "is-arguments": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.0.4.tgz", - "integrity": "sha512-xPh0Rmt8NE65sNzvyUmWgI1tz3mKq74lGA0mL8LYZcoIzKOzDh6HmrYm3d18k60nHerC8A9Km8kYu87zfSFnLA==", - "dev": true + "version": "1.1.0", + "resolved": "https://registry.npmjs.org/is-arguments/-/is-arguments-1.1.0.tgz", + "integrity": "sha512-1Ij4lOMPl/xB5kBDn7I+b2ttPMKa8szhEIrXDuXQD/oe3HJLTLhqhgGspwgyGd6MOywBUqVvYicF72lkgDnIHg==", + "dev": true, + "requires": { + "call-bind": "^1.0.0" + } }, "is-arrayish": { "version": "0.2.1", @@ -6669,7 +23462,6 @@ "version": "2.1.0", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz", "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==", - "dev": true, "requires": { "binary-extensions": "^2.0.0" } @@ -6680,12 +23472,6 @@ "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", "dev": true }, - "is-callable": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/is-callable/-/is-callable-1.2.2.tgz", - "integrity": "sha512-dnMqspv5nU3LoewK2N/y7KLtxtakvTuaCsU9FU50/QDmdbHNy/4/JuRtMHqRU22o3q+W89YQndQEeCVwK+3qrA==", - "dev": true - }, "is-color-stop": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/is-color-stop/-/is-color-stop-1.1.0.tgz", @@ -6698,76 +23484,69 @@ "hsla-regex": "^1.0.0", "rgb-regex": "^1.0.1", "rgba-regex": "^1.0.0" - } - }, - "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=", - "dev": true, - "requires": { - "kind-of": "^3.0.2" }, "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-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 - }, - "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": { - "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==", + "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 } } }, - "is-directory": { - "version": "0.3.1", - "resolved": "https://registry.npmjs.org/is-directory/-/is-directory-0.3.1.tgz", - "integrity": "sha1-YTObbyR1/Hcv2cnYP1yFddwVSuE=", + "is-core-module": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.4.0.tgz", + "integrity": "sha512-6A2fkfq1rfeQZjxrZJGerpLCTHRNEBiSgnu0+obeJpEPZRUooHgsizvzv0ZjJwOz3iWIHdJtVWJ/tmPr3D21/A==", + "requires": { + "has": "^1.0.3" + } + }, + "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-date-object": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/is-date-object/-/is-date-object-1.0.4.tgz", + "integrity": "sha512-/b4ZVsG7Z5XVtIxs/h9W8nvfLgSAyKYdtGWQLbqy6jA1icmgjf8WCoTKgeS4wy5tYaPePouzFMANbnj94c2Z+A==", "dev": true }, + "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-docker": { - "version": "2.1.1", - "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.1.1.tgz", - "integrity": "sha512-ZOoqiXfEwtGknTiuDEy8pN2CfE3TxMHprvNer1mXiqwkOT77Rw3YVrUQ52EqAOU3QAWDQ+bQdx7HJzrv7LS2Hw==", + "version": "2.2.1", + "resolved": "https://registry.npmjs.org/is-docker/-/is-docker-2.2.1.tgz", + "integrity": "sha512-F+i2BKsFrH66iaUFc0woD8sLy8getkwTwtOBjvs56Cx4CgJDeKQeqfz8wAYiSb8JOprWhHH5p77PbmYCvvUuXQ==", "dev": true }, "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 + "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-extglob": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz", - "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=", - "dev": true + "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=" }, "is-fullwidth-code-point": { "version": "2.0.0", @@ -6779,7 +23558,6 @@ "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": { "is-extglob": "^2.1.1" } @@ -6790,23 +23568,16 @@ "integrity": "sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w==", "dev": true }, - "is-negative-zero": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/is-negative-zero/-/is-negative-zero-2.0.0.tgz", - "integrity": "sha1-lVOxIbD6wohp2p7UWeIMdUN4hGE=", + "is-lambda": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/is-lambda/-/is-lambda-1.0.1.tgz", + "integrity": "sha1-PZh3iZ5qU+/AFgUEzeFfgubwYdU=", "dev": true }, "is-number": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz", - "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==", - "dev": true - }, - "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 + "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==" }, "is-path-cwd": { "version": "2.2.0", @@ -6832,12 +23603,6 @@ "path-is-inside": "^1.0.2" } }, - "is-plain-obj": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-1.1.0.tgz", - "integrity": "sha1-caUMhCnfync8kqOQpKA7OfzVHT4=", - "dev": true - }, "is-plain-object": { "version": "2.0.4", "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-2.0.4.tgz", @@ -6848,12 +23613,13 @@ } }, "is-regex": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.1.tgz", - "integrity": "sha512-1+QkEcxiLlB7VEyFtyBg94e08OAsvq7FUBgApTq/w2ymCLyKJgDPsybBENVtA7XCQEgEXxKPonG+mvYRxh/LIg==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/is-regex/-/is-regex-1.1.3.tgz", + "integrity": "sha512-qSVXFz28HM7y+IWX6vLCsexdlvzT1PJNFSBuaQLQ5o0IEw8UDYW6/2+eCMVyIsbM8CNLX2a/QWmSpyxYEHY7CQ==", "dev": true, "requires": { - "has-symbols": "^1.0.1" + "call-bind": "^1.0.2", + "has-symbols": "^1.0.2" } }, "is-resolvable": { @@ -6868,30 +23634,24 @@ "integrity": "sha1-EtSj3U5o4Lec6428hBc66A2RykQ=", "dev": true }, - "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": { - "html-comment-regex": "^1.1.0" - } - }, - "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": { - "has-symbols": "^1.0.1" - } - }, "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-unicode-supported": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", + "integrity": "sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==", + "dev": true + }, + "is-what": { + "version": "3.14.1", + "resolved": "https://registry.npmjs.org/is-what/-/is-what-3.14.1.tgz", + "integrity": "sha512-sNxgpk9793nzSs7bA6JQJGeIuRBQhAaNGG77kzYQgMkrID+lS6SlK07K5LaptscDlSaIgH+GPFzf+d75FVxozA==", + "dev": true + }, "is-windows": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", @@ -6914,9 +23674,9 @@ "dev": true }, "isbinaryfile": { - "version": "4.0.6", - "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.6.tgz", - "integrity": "sha512-ORrEy+SNVqUhrCaal4hA4fBzhggQQ+BaLntyPOdoEiwlKZW9BZiJXjg3RMiruE4tPEI3pyVPpySHQF/dKWperg==", + "version": "4.0.8", + "resolved": "https://registry.npmjs.org/isbinaryfile/-/isbinaryfile-4.0.8.tgz", + "integrity": "sha512-53h6XFniq77YdW+spoRrebh0mnmTxRPTlcuIArO57lmMdq4uBKFKaeTjnb92oYWrSn/LVL+LT+Hap2tFQj8V+w==", "dev": true }, "isexe": { @@ -7093,9 +23853,9 @@ "dev": true }, "jest-worker": { - "version": "26.3.0", - "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.3.0.tgz", - "integrity": "sha512-Vmpn2F6IASefL+DVBhPzI2J9/GJUsqzomdeN+P+dK8/jKxbh8R3BtFnx3FIta7wYlPU62cpJMJQo4kuOowcMnw==", + "version": "26.6.2", + "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.6.2.tgz", + "integrity": "sha512-KWYVV1c4i+jbMpaBC+U++4Va0cp8OisU185o73T1vo99hqi7w8tSJfUXYswwqqrjzwxa6KpRK54WhPvwf5w6PQ==", "dev": true, "requires": { "@types/node": "*", @@ -7191,16 +23951,15 @@ } }, "jsonc-parser": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-2.3.0.tgz", - "integrity": "sha512-b0EBt8SWFNnixVdvoR2ZtEGa9ZqLhbJnOjezn+WP+8kspFm+PFYDN8Z4Bc7pRlDjvuVcADSUkroIuTWWn/YiIA==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/jsonc-parser/-/jsonc-parser-3.0.0.tgz", + "integrity": "sha512-fQzRfAbIBnR0IQvftw9FJveWiHp72Fg20giDrHz6TdfB12UH/uue0D3hm57UB5KgAVuniLMCaS8P1IMj9NR7cA==", "dev": true }, "jsonfile": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/jsonfile/-/jsonfile-4.0.0.tgz", "integrity": "sha1-h3Gq4HmbZAdrdmQPygWPnBDjPss=", - "dev": true, "requires": { "graceful-fs": "^4.1.6" } @@ -7236,35 +23995,34 @@ } }, "karma": { - "version": "5.0.9", - "resolved": "https://registry.npmjs.org/karma/-/karma-5.0.9.tgz", - "integrity": "sha512-dUA5z7Lo7G4FRSe1ZAXqOINEEWxmCjDBbfRBmU/wYlSMwxUQJP/tEEP90yJt3Uqo03s9rCgVnxtlfq+uDhxSPg==", + "version": "6.3.3", + "resolved": "https://registry.npmjs.org/karma/-/karma-6.3.3.tgz", + "integrity": "sha512-JRAujkKWaOtO2LmyPH7K2XXRhrxuFAn9loIL9+iiah6vjz+ZLkqdKsySV9clRITGhj10t9baIfbCl6CJ5hu9gQ==", "dev": true, "requires": { "body-parser": "^1.19.0", "braces": "^3.0.2", - "chokidar": "^3.0.0", + "chokidar": "^3.4.2", "colors": "^1.4.0", "connect": "^3.7.0", "di": "^0.0.1", "dom-serialize": "^2.2.1", - "flatted": "^2.0.2", "glob": "^7.1.6", "graceful-fs": "^4.2.4", "http-proxy": "^1.18.1", "isbinaryfile": "^4.0.6", - "lodash": "^4.17.15", + "lodash": "^4.17.19", "log4js": "^6.2.1", "mime": "^2.4.5", "minimatch": "^3.0.4", "qjobs": "^1.2.0", "range-parser": "^1.2.1", "rimraf": "^3.0.2", - "socket.io": "^2.3.0", + "socket.io": "^3.1.0", "source-map": "^0.6.1", "tmp": "0.2.1", - "ua-parser-js": "0.7.21", - "yargs": "^15.3.1" + "ua-parser-js": "^0.7.23", + "yargs": "^16.1.1" }, "dependencies": { "ansi-regex": { @@ -7282,21 +24040,15 @@ "color-convert": "^2.0.1" } }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, "cliui": { - "version": "6.0.0", - "resolved": "https://registry.npmjs.org/cliui/-/cliui-6.0.0.tgz", - "integrity": "sha512-t6wbgtoCXvAzst7QgXxJYqPt0usEfbgQdftEPbLL/cvv6HPE5VgvqCuAIDR0NgU52ds6rFwqrgakNLrHEjCbrQ==", + "version": "7.0.4", + "resolved": "https://registry.npmjs.org/cliui/-/cliui-7.0.4.tgz", + "integrity": "sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==", "dev": true, "requires": { "string-width": "^4.2.0", "strip-ansi": "^6.0.0", - "wrap-ansi": "^6.2.0" + "wrap-ansi": "^7.0.0" } }, "color-convert": { @@ -7320,50 +24072,16 @@ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", "dev": true }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, "is-fullwidth-code-point": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz", "integrity": "sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==", "dev": true }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, "mime": { - "version": "2.4.6", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.6.tgz", - "integrity": "sha512-RZKhC3EmpBchfTGBVb8fb+RL2cWyw/32lshnsETttkBAyAUXSGHxbEJWWRXc751DrIxG1q04b8QwMbAwkRPpUA==", - "dev": true - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.5.2.tgz", + "integrity": "sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg==", "dev": true }, "source-map": { @@ -7373,9 +24091,9 @@ "dev": true }, "string-width": { - "version": "4.2.0", - "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.0.tgz", - "integrity": "sha512-zUz5JD+tgqtuDjMhwIg5uFVV3dtqZ9yQJlZVfq4I01/K5Paj5UHj7VyrQOJvzawSVlKpObApbfD0Ed6yJc+1eg==", + "version": "4.2.2", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-4.2.2.tgz", + "integrity": "sha512-XBJbT3N4JhVumXE0eoLU9DCjcaF92KLNqTmFCnG1pf8duUxFGwtP6AD6nkjw9a3IdiRtL3E2w3JDiE/xi3vOeA==", "dev": true, "requires": { "emoji-regex": "^8.0.0", @@ -7402,9 +24120,9 @@ } }, "wrap-ansi": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-6.2.0.tgz", - "integrity": "sha512-r6lPcBGxZXlIcymEu7InxDMhdW0KDxpLgoFLcguasxCaJ/SOIZwINatK9KY/tf+ZrlywOKU0UDj3ATXUBfxJXA==", + "version": "7.0.0", + "resolved": "https://registry.npmjs.org/wrap-ansi/-/wrap-ansi-7.0.0.tgz", + "integrity": "sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==", "dev": true, "requires": { "ansi-styles": "^4.0.0", @@ -7412,34 +24130,32 @@ "strip-ansi": "^6.0.0" } }, + "y18n": { + "version": "5.0.8", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-5.0.8.tgz", + "integrity": "sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==", + "dev": true + }, "yargs": { - "version": "15.4.1", - "resolved": "https://registry.npmjs.org/yargs/-/yargs-15.4.1.tgz", - "integrity": "sha512-aePbxDmcYW++PaqBsJ+HYUFwCdv4LVvdnhBy78E57PIor8/OVvhMrADFFEDh8DHDFRv/O9i3lPhsENjO7QX0+A==", + "version": "16.2.0", + "resolved": "https://registry.npmjs.org/yargs/-/yargs-16.2.0.tgz", + "integrity": "sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==", "dev": true, "requires": { - "cliui": "^6.0.0", - "decamelize": "^1.2.0", - "find-up": "^4.1.0", - "get-caller-file": "^2.0.1", + "cliui": "^7.0.2", + "escalade": "^3.1.1", + "get-caller-file": "^2.0.5", "require-directory": "^2.1.1", - "require-main-filename": "^2.0.0", - "set-blocking": "^2.0.0", "string-width": "^4.2.0", - "which-module": "^2.0.0", - "y18n": "^4.0.0", - "yargs-parser": "^18.1.2" + "y18n": "^5.0.5", + "yargs-parser": "^20.2.2" } }, "yargs-parser": { - "version": "18.1.3", - "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-18.1.3.tgz", - "integrity": "sha512-o50j0JeToy/4K6OZcaQmW6lyXXKhq7csREXcDwk2omFPJEwUNOVtJKvmDr9EI1fAJZUyZcRF7kxGBWmRXudrCQ==", - "dev": true, - "requires": { - "camelcase": "^5.0.0", - "decamelize": "^1.2.0" - } + "version": "20.2.7", + "resolved": "https://registry.npmjs.org/yargs-parser/-/yargs-parser-20.2.7.tgz", + "integrity": "sha512-FiNkvbeHzB/syOjIUxFDCnhSfzAL8R5vs40MgLFBorXACCOAEaWu0gRZl14vG8MR9AOJIZbmkjhusqBYZ3HTHw==", + "dev": true } } }, @@ -7508,17 +24224,19 @@ "dev": true }, "less": { - "version": "3.12.2", - "resolved": "https://registry.npmjs.org/less/-/less-3.12.2.tgz", - "integrity": "sha512-+1V2PCMFkL+OIj2/HrtrvZw0BC0sYLMICJfbQjuj/K8CEnlrFX6R5cKKgzzttsZDHyxQNL1jqMREjKN3ja/E3Q==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/less/-/less-4.1.1.tgz", + "integrity": "sha512-w09o8tZFPThBscl5d0Ggp3RcrKIouBoQscnOMgFH3n5V3kN/CXGHNfCkRPtxJk6nKryDXaV9aHLK55RXuH4sAw==", "dev": true, "requires": { + "copy-anything": "^2.0.1", "errno": "^0.1.1", "graceful-fs": "^4.1.2", "image-size": "~0.5.0", "make-dir": "^2.1.0", "mime": "^1.4.1", - "native-request": "^1.0.5", + "needle": "^2.5.2", + "parse-node-version": "^1.0.1", "source-map": "~0.6.0", "tslib": "^1.10.0" }, @@ -7539,40 +24257,55 @@ } }, "less-loader": { - "version": "6.2.0", - "resolved": "https://registry.npmjs.org/less-loader/-/less-loader-6.2.0.tgz", - "integrity": "sha512-Cl5h95/Pz/PWub/tCBgT1oNMFeH1WTD33piG80jn5jr12T4XbxZcjThwNXDQ7AG649WEynuIzO4b0+2Tn9Qolg==", + "version": "7.3.0", + "resolved": "https://registry.npmjs.org/less-loader/-/less-loader-7.3.0.tgz", + "integrity": "sha512-Mi8915g7NMaLlgi77mgTTQvK022xKRQBIVDSyfl3ErTuBhmZBQab0mjeJjNNqGbdR+qrfTleKXqbGI4uEFavxg==", "dev": true, "requires": { - "clone": "^2.1.2", - "less": "^3.11.3", + "klona": "^2.0.4", "loader-utils": "^2.0.0", - "schema-utils": "^2.7.0" - } - }, - "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 - }, - "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": { - "leven": "^3.1.0" + "schema-utils": "^3.0.0" + }, + "dependencies": { + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, "license-webpack-plugin": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/license-webpack-plugin/-/license-webpack-plugin-2.3.0.tgz", - "integrity": "sha512-JK/DXrtN6UeYQSgkg5q1+pgJ8aiKPL9tnz9Wzw+Ikkf+8mJxG56x6t8O+OH/tAeF/5NREnelTEMyFtbJNkjH4w==", + "version": "2.3.11", + "resolved": "https://registry.npmjs.org/license-webpack-plugin/-/license-webpack-plugin-2.3.11.tgz", + "integrity": "sha512-0iVGoX5vx0WDy8dmwTTpOOMYiGqILyUbDeVMFH52AjgBlS58lHwOlFMSoqg5nY8Kxl6+FRKyUZY/UdlQaOyqDw==", "dev": true, "requires": { "@types/webpack-sources": "^0.1.5", "webpack-sources": "^1.2.0" + }, + "dependencies": { + "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 + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + } } }, "lie": { @@ -7584,6 +24317,12 @@ "immediate": "~3.0.5" } }, + "lines-and-columns": { + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.1.6.tgz", + "integrity": "sha1-HADHQ7QzzQpOgHWPe2SldEDZ/wA=", + "dev": true + }, "loader-runner": { "version": "2.4.0", "resolved": "https://registry.npmjs.org/loader-runner/-/loader-runner-2.4.0.tgz", @@ -7602,25 +24341,18 @@ } }, "locate-path": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-3.0.0.tgz", - "integrity": "sha512-7AO748wWnIhNqAuaty2ZWHkQHRSNfPVIsPIfwEOWO22AmaoVrWavlOcMR5nzTLNYvp36X220/maaRsrec1G65A==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", + "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", "dev": true, "requires": { - "p-locate": "^3.0.0", - "path-exists": "^3.0.0" + "p-locate": "^4.1.0" } }, "lodash": { - "version": "4.17.20", - "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.20.tgz", - "integrity": "sha512-PlhdFcillOINfeV7Ni6oF1TAEayyZBoZ8bcshTHqOYJYlrqzRK5hagpagky5o4HfCzzd1TRkXPMFq6cKk9rGmA==" - }, - "lodash.clonedeep": { - "version": "4.5.0", - "resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz", - "integrity": "sha1-4j8/nE+Pvd6HJSnBBxhXoIblzO8=", - "dev": true + "version": "4.17.21", + "resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz", + "integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==" }, "lodash.memoize": { "version": "4.1.2", @@ -7628,12 +24360,6 @@ "integrity": "sha1-vMbEmkKihA7Zl/Mj6tpezRguC/4=", "dev": true }, - "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": { "version": "4.5.0", "resolved": "https://registry.npmjs.org/lodash.uniq/-/lodash.uniq-4.5.0.tgz", @@ -7641,12 +24367,13 @@ "dev": true }, "log-symbols": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.0.0.tgz", - "integrity": "sha512-FN8JBzLx6CzeMrB0tg6pqlGU1wCrXW+ZXGH481kfsBqer0hToTIiHdjH4Mq8xJUbvATujKCvaREGWpGUionraA==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/log-symbols/-/log-symbols-4.1.0.tgz", + "integrity": "sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==", "dev": true, "requires": { - "chalk": "^4.0.0" + "chalk": "^4.1.0", + "is-unicode-supported": "^0.1.0" }, "dependencies": { "ansi-styles": { @@ -7659,9 +24386,9 @@ } }, "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", "dev": true, "requires": { "ansi-styles": "^4.1.0", @@ -7714,20 +24441,11 @@ } }, "loglevel": { - "version": "1.7.0", - "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.0.tgz", - "integrity": "sha512-i2sY04nal5jDcagM3FMfG++T69GEEM8CYuOfeOIvmXzOIcwE9a/CJPR0MFM97pYMj/u10lzz7/zd7+qwhrBTqQ==", + "version": "1.7.1", + "resolved": "https://registry.npmjs.org/loglevel/-/loglevel-1.7.1.tgz", + "integrity": "sha512-Hesni4s5UkWkwCGJMQGAh71PaLUmKFM60dHvq0zi/vDhhrzuk+4GgNbTXJ12YYQJn6ZKBDNIjYcuQGKudvqrIw==", "dev": true }, - "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": { - "js-tokens": "^3.0.0 || ^4.0.0" - } - }, "lru-cache": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-6.0.0.tgz", @@ -7741,7 +24459,6 @@ "version": "0.25.7", "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz", "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==", - "dev": true, "requires": { "sourcemap-codec": "^1.4.4" } @@ -7771,85 +24488,62 @@ "dev": true }, "make-fetch-happen": { - "version": "5.0.2", - "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-5.0.2.tgz", - "integrity": "sha512-07JHC0r1ykIoruKO8ifMXu+xEU8qOXDFETylktdug6vJDACnP+HKevOu3PXyNPzFyTSlz8vrBYlBO1JZRe8Cag==", + "version": "8.0.14", + "resolved": "https://registry.npmjs.org/make-fetch-happen/-/make-fetch-happen-8.0.14.tgz", + "integrity": "sha512-EsS89h6l4vbfJEtBZnENTOFk8mCRpY5ru36Xe5bcX1KYIli2mkSHqoFsp5O1wMDvTJJzxe/4THpCTtygjeeGWQ==", "dev": true, "requires": { - "agentkeepalive": "^3.4.1", - "cacache": "^12.0.0", - "http-cache-semantics": "^3.8.1", - "http-proxy-agent": "^2.1.0", - "https-proxy-agent": "^2.2.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "node-fetch-npm": "^2.0.2", - "promise-retry": "^1.1.1", - "socks-proxy-agent": "^4.0.0", - "ssri": "^6.0.0" + "agentkeepalive": "^4.1.3", + "cacache": "^15.0.5", + "http-cache-semantics": "^4.1.0", + "http-proxy-agent": "^4.0.1", + "https-proxy-agent": "^5.0.0", + "is-lambda": "^1.0.1", + "lru-cache": "^6.0.0", + "minipass": "^3.1.3", + "minipass-collect": "^1.0.2", + "minipass-fetch": "^1.3.2", + "minipass-flush": "^1.0.5", + "minipass-pipeline": "^1.2.4", + "promise-retry": "^2.0.1", + "socks-proxy-agent": "^5.0.0", + "ssri": "^8.0.0" }, "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", + "agent-base": { + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", "dev": true, "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" + "debug": "4" } }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", + "err-code": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/err-code/-/err-code-2.0.3.tgz", + "integrity": "sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA==", "dev": true }, - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", + "https-proxy-agent": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/https-proxy-agent/-/https-proxy-agent-5.0.0.tgz", + "integrity": "sha512-EkYm5BcKUGiduxzSt3Eppko+PiNWNEpa4ySk9vTC6wDsQJW9rHSa+UhGNJoRYp7bz6Ht1eaRIa6QaJqO5rCFbA==", "dev": true, "requires": { - "yallist": "^3.0.2" + "agent-base": "6", + "debug": "4" } }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", + "promise-retry": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/promise-retry/-/promise-retry-2.0.1.tgz", + "integrity": "sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g==", "dev": true, "requires": { - "glob": "^7.1.3" + "err-code": "^2.0.2", + "retry": "^0.12.0" } - }, - "ssri": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.1.tgz", - "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true } } }, @@ -7868,6 +24562,11 @@ "object-visit": "^1.0.0" } }, + "material-colors": { + "version": "1.2.6", + "resolved": "https://registry.npmjs.org/material-colors/-/material-colors-1.2.6.tgz", + "integrity": "sha512-6qE4B9deFBIa9YSpOc9O0Sgc43zTeVYbgDT5veRKSlB2+ZuHNoVVxA1L/ckMUayV9Ay9y7Z/SZCLcGteW9i7bg==" + }, "md5.js": { "version": "1.3.5", "resolved": "https://registry.npmjs.org/md5.js/-/md5.js-1.3.5.tgz", @@ -7880,9 +24579,9 @@ } }, "mdn-data": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.4.tgz", - "integrity": "sha512-iV3XNKw06j5Q7mi6h+9vbx23Tv7JkjEVgKHW4pimwyDGWm0OIQntJJ+u1C6mg6mK1EaTv42XQ7w76yuzH7M2cA==", + "version": "2.0.14", + "resolved": "https://registry.npmjs.org/mdn-data/-/mdn-data-2.0.14.tgz", + "integrity": "sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==", "dev": true }, "media-typer": { @@ -7892,9 +24591,9 @@ "dev": true }, "memory-fs": { - "version": "0.5.0", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", - "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", + "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", "dev": true, "requires": { "errno": "^0.1.3", @@ -7943,13 +24642,13 @@ "dev": true }, "micromatch": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.2.tgz", - "integrity": "sha512-y7FpHSbMUMoyPbYUSzO6PaZ6FyRnQOpHuKwbo1G+Knck95XVU4QAiKdGEnj5wwoS7PlOgthX/09u5iFJ+aYf5Q==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.4.tgz", + "integrity": "sha512-pRmzw/XUcwXGpD9aI9q/0XOwLNygjETJ8y0ao0wdqprrzDa4YnxLcz7fQRZr8voh8V10kGhABbNcHVk5wHgWwg==", "dev": true, "requires": { "braces": "^3.0.1", - "picomatch": "^2.0.5" + "picomatch": "^2.2.3" } }, "miller-rabin": { @@ -7963,9 +24662,9 @@ }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -7998,58 +24697,41 @@ "dev": true }, "mini-css-extract-plugin": { - "version": "0.10.0", - "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-0.10.0.tgz", - "integrity": "sha512-QgKgJBjaJhxVPwrLNqqwNS0AGkuQQ31Hp4xGXEK/P7wehEg6qmNtReHKai3zRXqY60wGVWLYcOMJK2b98aGc3A==", + "version": "1.3.5", + "resolved": "https://registry.npmjs.org/mini-css-extract-plugin/-/mini-css-extract-plugin-1.3.5.tgz", + "integrity": "sha512-tvmzcwqJJXau4OQE5vT72pRT18o2zF+tQJp8CWchqvfQnTlflkzS+dANYcRdyPRWUWRkfmeNTKltx0NZI/b5dQ==", "dev": true, "requires": { - "loader-utils": "^1.1.0", - "normalize-url": "1.9.1", - "schema-utils": "^1.0.0", + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0", "webpack-sources": "^1.1.0" }, "dependencies": { - "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": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, - "normalize-url": { - "version": "1.9.1", - "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-1.9.1.tgz", - "integrity": "sha1-LMDWazHqIwNkWENuNiDYWVTGbDw=", - "dev": true, - "requires": { - "object-assign": "^4.0.1", - "prepend-http": "^1.0.0", - "query-string": "^4.1.0", - "sort-keys": "^1.0.0" - } - }, "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", "dev": true, "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, + "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 + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" } } } @@ -8097,6 +24779,18 @@ "minipass": "^3.0.0" } }, + "minipass-fetch": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/minipass-fetch/-/minipass-fetch-1.3.3.tgz", + "integrity": "sha512-akCrLDWfbdAWkMLBxJEeWTdNsjML+dt5YgOI4gJ53vuO0vrmYQkUPxa6j6V65s9CcePIr2SSWqjT2EcrNseryQ==", + "dev": true, + "requires": { + "encoding": "^0.1.12", + "minipass": "^3.1.0", + "minipass-sized": "^1.0.3", + "minizlib": "^2.0.0" + } + }, "minipass-flush": { "version": "1.0.5", "resolved": "https://registry.npmjs.org/minipass-flush/-/minipass-flush-1.0.5.tgz", @@ -8106,6 +24800,16 @@ "minipass": "^3.0.0" } }, + "minipass-json-stream": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/minipass-json-stream/-/minipass-json-stream-1.0.1.tgz", + "integrity": "sha512-ODqY18UZt/I8k+b7rl2AENgbWE8IDYam+undIJONvigAz8KR5GWblsFTEfQs0WODsjbSXWlm+JHEv8Gr6Tfdbg==", + "dev": true, + "requires": { + "jsonparse": "^1.3.1", + "minipass": "^3.0.0" + } + }, "minipass-pipeline": { "version": "1.2.4", "resolved": "https://registry.npmjs.org/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz", @@ -8115,6 +24819,15 @@ "minipass": "^3.0.0" } }, + "minipass-sized": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/minipass-sized/-/minipass-sized-1.0.3.tgz", + "integrity": "sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g==", + "dev": true, + "requires": { + "minipass": "^3.0.0" + } + }, "minizlib": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-2.1.2.tgz", @@ -8151,17 +24864,6 @@ "requires": { "for-in": "^1.0.2", "is-extendable": "^1.0.1" - }, - "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" - } - } } }, "mkdirp": { @@ -8173,11 +24875,6 @@ "minimist": "^1.2.5" } }, - "moment": { - "version": "2.18.1", - "resolved": "https://registry.npmjs.org/moment/-/moment-2.18.1.tgz", - "integrity": "sha1-w2GT3Tzhwu7SrbfIAtu8d6gbHA8=" - }, "move-concurrently": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/move-concurrently/-/move-concurrently-1.0.1.tgz", @@ -8237,6 +24934,12 @@ "dev": true, "optional": true }, + "nanoid": { + "version": "3.1.23", + "resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.1.23.tgz", + "integrity": "sha512-FiB0kzdP0FFVGDKlRLEQ1BgDzU87dy5NnzjeW9YZNt+/c3+q82EQDUwniSAUxp/F0gFNI1ZhKU1FqYsMuqZVnw==", + "dev": true + }, "nanomatch": { "version": "1.2.13", "resolved": "https://registry.npmjs.org/nanomatch/-/nanomatch-1.2.13.tgz", @@ -8256,12 +24959,29 @@ "to-regex": "^3.0.1" } }, - "native-request": { - "version": "1.0.8", - "resolved": "https://registry.npmjs.org/native-request/-/native-request-1.0.8.tgz", - "integrity": "sha512-vU2JojJVelUGp6jRcLwToPoWGxSx23z/0iX+I77J3Ht17rf2INGjrhOoQnjVo60nQd8wVsgzKkPfRXBiVdD2ag==", + "needle": { + "version": "2.6.0", + "resolved": "https://registry.npmjs.org/needle/-/needle-2.6.0.tgz", + "integrity": "sha512-KKYdza4heMsEfSWD7VPUIz3zX2XDwOyX2d+geb4vrERZMT5RMU6ujjaD+I5Yr54uZxQ2w6XRTAhHBbSCyovZBg==", "dev": true, - "optional": true + "optional": true, + "requires": { + "debug": "^3.2.6", + "iconv-lite": "^0.4.4", + "sax": "^1.2.4" + }, + "dependencies": { + "debug": { + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", + "dev": true, + "optional": true, + "requires": { + "ms": "^2.1.1" + } + } + } }, "negotiator": { "version": "0.6.2", @@ -8275,20 +24995,6 @@ "integrity": "sha512-Yd3UES5mWCSqR+qNT93S3UoYUkqAZ9lLg8a7g9rimsWmYGK8cVToA4/sF3RrshdyV3sAGMXVUmpMYOw+dLpOuw==", "dev": true }, - "next-tick": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.0.0.tgz", - "integrity": "sha1-yobR/ogoFpsBICCOPchCS524NCw=", - "dev": true - }, - "ng-bootstrap": { - "version": "1.6.3", - "resolved": "https://registry.npmjs.org/ng-bootstrap/-/ng-bootstrap-1.6.3.tgz", - "integrity": "sha1-1B/UIVTAWTQiy4PEc6OCiqdSW/U=", - "requires": { - "moment": "2.18.1" - } - }, "ng2-pdf-viewer": { "version": "6.3.2", "resolved": "https://registry.npmjs.org/ng2-pdf-viewer/-/ng2-pdf-viewer-6.3.2.tgz", @@ -8306,6 +25012,22 @@ } } }, + "ngx-bootstrap": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/ngx-bootstrap/-/ngx-bootstrap-6.2.0.tgz", + "integrity": "sha512-5WKHo6/ltkenw4UyXZwED8rODCgp2RGbWurzYzZsF/gH1JO5SN7TJ+AL6kXYk6XM42sDA2WhN9Db+ZPNjiyHnA==", + "requires": {} + }, + "ngx-color": { + "version": "6.2.0", + "resolved": "https://registry.npmjs.org/ngx-color/-/ngx-color-6.2.0.tgz", + "integrity": "sha512-n04tcMnCpOgmI24egST94YwHmnSoAxK8O1T2t3nGrTwWbvw5XBRJvImNFnoNrriBXzc4Gx4hFehH5MU8CZxp1w==", + "requires": { + "@ctrl/tinycolor": "^3.1.6", + "material-colors": "^1.2.6", + "tslib": "^2.0.0" + } + }, "ngx-cookie-service": { "version": "10.1.1", "resolved": "https://registry.npmjs.org/ngx-cookie-service/-/ngx-cookie-service-10.1.1.tgz", @@ -8315,9 +25037,9 @@ } }, "ngx-file-drop": { - "version": "10.0.0", - "resolved": "https://registry.npmjs.org/ngx-file-drop/-/ngx-file-drop-10.0.0.tgz", - "integrity": "sha512-izV90rNuXGeTCoodvD35sCC/D/bDIuFFdj1UnTMf8n3PA5O5v+7L/PJ8d9IAXIhjAg3fY0lIIqbP8RFexMbNeA==", + "version": "11.1.0", + "resolved": "https://registry.npmjs.org/ngx-file-drop/-/ngx-file-drop-11.1.0.tgz", + "integrity": "sha512-jRqnI58jh9xVYmbYY315TIaSfpzOXRwj8JQOjNALwi/r1a9SVkxp3C069tXo5Kkxi/elsMePN0Be4EmgLf4EVw==", "requires": { "tslib": "^2.0.0" } @@ -8337,23 +25059,41 @@ "integrity": "sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ==", "dev": true }, - "node-fetch-npm": { - "version": "2.0.4", - "resolved": "https://registry.npmjs.org/node-fetch-npm/-/node-fetch-npm-2.0.4.tgz", - "integrity": "sha512-iOuIQDWDyjhv9qSDrj9aq/klt6F9z1p2otB3AV7v3zBDcL/x+OfGsvGQZZCcMZbUf4Ujw1xGNQkjvGnVT22cKg==", - "dev": true, - "requires": { - "encoding": "^0.1.11", - "json-parse-better-errors": "^1.0.0", - "safe-buffer": "^5.1.1" - } - }, "node-forge": { "version": "0.10.0", "resolved": "https://registry.npmjs.org/node-forge/-/node-forge-0.10.0.tgz", "integrity": "sha512-PPmu8eEeG9saEUvI97fm4OYxXVB6bFvyNTyiUOBichBpFG8A1Ljw3bY62+5oOjDEMHRnd0Y7HQ+x7uzxOzC6JA==", "dev": true }, + "node-gyp": { + "version": "7.1.2", + "resolved": "https://registry.npmjs.org/node-gyp/-/node-gyp-7.1.2.tgz", + "integrity": "sha512-CbpcIo7C3eMu3dL1c3d0xw449fHIGALIJsRP4DDPHpyiW8vcriNY7ubh9TE4zEKfSxscY7PjeFnshE7h75ynjQ==", + "dev": true, + "requires": { + "env-paths": "^2.2.0", + "glob": "^7.1.4", + "graceful-fs": "^4.2.3", + "nopt": "^5.0.0", + "npmlog": "^4.1.2", + "request": "^2.88.2", + "rimraf": "^3.0.2", + "semver": "^7.3.2", + "tar": "^6.0.2", + "which": "^2.0.2" + }, + "dependencies": { + "which": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", + "integrity": "sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==", + "dev": true, + "requires": { + "isexe": "^2.0.0" + } + } + } + }, "node-libs-browser": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/node-libs-browser/-/node-libs-browser-2.2.1.tgz", @@ -8394,42 +25134,24 @@ } }, "node-releases": { - "version": "1.1.64", - "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.64.tgz", - "integrity": "sha512-Iec8O9166/x2HRMJyLLLWkd0sFFLrFNy+Xf+JQfSQsdBJzPcHpNl3JQ9gD4j+aJxmCa25jNsIbM4bmACtSbkSg==", + "version": "1.1.73", + "resolved": "https://registry.npmjs.org/node-releases/-/node-releases-1.1.73.tgz", + "integrity": "sha512-uW7fodD6pyW2FZNZnp/Z3hvWKeEW1Y8R1+1CnErE8cXFXzl5blBOoVB41CvMer6P6Q0S5FXDwcHgFd1Wj0U9zg==", "dev": true }, - "normalize-package-data": { - "version": "2.5.0", - "resolved": "https://registry.npmjs.org/normalize-package-data/-/normalize-package-data-2.5.0.tgz", - "integrity": "sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA==", + "nopt": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/nopt/-/nopt-5.0.0.tgz", + "integrity": "sha512-Tbj67rffqceeLpcRXrT7vKAN8CwfPeIBgM7E6iBkmKLV7bEMwpGgYLGv0jACUsECaa/vuxP0IjEont6umdMgtQ==", "dev": true, "requires": { - "hosted-git-info": "^2.1.4", - "resolve": "^1.10.0", - "semver": "2 || 3 || 4 || 5", - "validate-npm-package-license": "^3.0.1" - }, - "dependencies": { - "hosted-git-info": { - "version": "2.8.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", - "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - } + "abbrev": "1" } }, "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 + "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==" }, "normalize-range": { "version": "0.1.2", @@ -8438,15 +25160,15 @@ "dev": true }, "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==", + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/normalize-url/-/normalize-url-6.0.1.tgz", + "integrity": "sha512-VU4pzAuh7Kip71XEmO9aNREYAdMHFGTVj/i+CaTImS8x0i1d3jUZkXhqluy/PRgjPLMgsLQulYY3PJ/aSbSjpQ==", "dev": true }, "npm-bundled": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.1.tgz", - "integrity": "sha512-gqkfgGePhTpAEgUsGEgcq1rqPXA+tv/aVBlgEzfXwA1yiUJF7xtEt3CtVwOjNYQOVknDk0F20w58Fnm3EtG0fA==", + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/npm-bundled/-/npm-bundled-1.1.2.tgz", + "integrity": "sha512-x5DHup0SuyQcmL3s7Rx/YQ8sbw/Hzg0rj48eN0dV7hf5cmQq5PXIeioroH3raV1QC1yh3uTYuMThvEQF3iKgGQ==", "dev": true, "requires": { "npm-normalize-package-bin": "^1.0.1" @@ -8468,24 +25190,25 @@ "dev": true }, "npm-package-arg": { - "version": "8.0.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.0.1.tgz", - "integrity": "sha512-/h5Fm6a/exByzFSTm7jAyHbgOqErl9qSNJDQF32Si/ZzgwT2TERVxRxn3Jurw1wflgyVVAxnFR4fRHPM7y1ClQ==", + "version": "8.1.0", + "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-8.1.0.tgz", + "integrity": "sha512-/ep6QDxBkm9HvOhOg0heitSd7JHA1U7y1qhhlRlteYYAi9Pdb/ZV7FW5aHpkrpM8+P+4p/jjR8zCyKPBMBjSig==", "dev": true, "requires": { - "hosted-git-info": "^3.0.2", + "hosted-git-info": "^3.0.6", "semver": "^7.0.0", "validate-npm-package-name": "^3.0.0" } }, "npm-packlist": { - "version": "1.4.8", - "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-1.4.8.tgz", - "integrity": "sha512-5+AZgwru5IevF5ZdnFglB5wNlHG1AOOuw28WhUq8/8emhBmLv6jX5by4WJCh7lW0uSYZYS6DXqIsyZVIXRZU9A==", + "version": "2.2.2", + "resolved": "https://registry.npmjs.org/npm-packlist/-/npm-packlist-2.2.2.tgz", + "integrity": "sha512-Jt01acDvJRhJGthnUJVF/w6gumWOZxO7IkpY/lsX9//zqQgnF7OJaxgQXcerd4uQOLu7W5bkb4mChL9mdfm+Zg==", "dev": true, "requires": { - "ignore-walk": "^3.0.1", - "npm-bundled": "^1.0.1", + "glob": "^7.1.6", + "ignore-walk": "^3.0.3", + "npm-bundled": "^1.1.1", "npm-normalize-package-bin": "^1.0.1" } }, @@ -8501,65 +25224,19 @@ } }, "npm-registry-fetch": { - "version": "4.0.7", - "resolved": "https://registry.npmjs.org/npm-registry-fetch/-/npm-registry-fetch-4.0.7.tgz", - "integrity": "sha512-cny9v0+Mq6Tjz+e0erFAB+RYJ/AVGzkjnISiobqP8OWj9c9FLoZZu8/SPSKJWE17F1tk4018wfjV+ZbIbqC7fQ==", + "version": "9.0.0", + "resolved": "https://registry.npmjs.org/npm-registry-fetch/-/npm-registry-fetch-9.0.0.tgz", + "integrity": "sha512-PuFYYtnQ8IyVl6ib9d3PepeehcUeHN9IO5N/iCRhyg9tStQcqGQBRVHmfmMWPDERU3KwZoHFvbJ4FPXPspvzbA==", "dev": true, "requires": { - "JSONStream": "^1.3.4", - "bluebird": "^3.5.1", - "figgy-pudding": "^3.4.1", - "lru-cache": "^5.1.1", - "make-fetch-happen": "^5.0.0", - "npm-package-arg": "^6.1.0", - "safe-buffer": "^5.2.0" - }, - "dependencies": { - "hosted-git-info": { - "version": "2.8.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", - "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", - "dev": true - }, - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "requires": { - "yallist": "^3.0.2" - } - }, - "npm-package-arg": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", - "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", - "dev": true, - "requires": { - "hosted-git-info": "^2.7.1", - "osenv": "^0.1.5", - "semver": "^5.6.0", - "validate-npm-package-name": "^3.0.0" - } - }, - "safe-buffer": { - "version": "5.2.1", - "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz", - "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==", - "dev": true - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", - "dev": true - } + "@npmcli/ci-detect": "^1.0.0", + "lru-cache": "^6.0.0", + "make-fetch-happen": "^8.0.9", + "minipass": "^3.1.3", + "minipass-fetch": "^1.3.0", + "minipass-json-stream": "^1.0.1", + "minizlib": "^2.0.0", + "npm-package-arg": "^8.0.0" } }, "npm-run-path": { @@ -8571,19 +25248,31 @@ "path-key": "^2.0.0" } }, - "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==", + "npmlog": { + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/npmlog/-/npmlog-4.1.2.tgz", + "integrity": "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg==", "dev": true, "requires": { - "boolbase": "~1.0.0" + "are-we-there-yet": "~1.1.2", + "console-control-strings": "~1.1.0", + "gauge": "~2.7.3", + "set-blocking": "~2.0.0" } }, - "num2fraction": { - "version": "1.2.2", - "resolved": "https://registry.npmjs.org/num2fraction/-/num2fraction-1.2.2.tgz", - "integrity": "sha1-b2gragJ6Tp3fpFZM0lidHU5mnt4=", + "nth-check": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/nth-check/-/nth-check-2.0.0.tgz", + "integrity": "sha512-i4sc/Kj8htBrAiH1viZ0TgU8Y5XqCaV/FziYK6TBczxmeKm3AEFWqqF3195yKudrarqy7Zu80Ra5dobFjn9X/Q==", + "dev": true, + "requires": { + "boolbase": "^1.0.0" + } + }, + "number-is-nan": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", "dev": true }, "oauth-sign": { @@ -8598,12 +25287,6 @@ "integrity": "sha1-IQmtx5ZYh8/AXLvUQsrIv7s2CGM=", "dev": true }, - "object-component": { - "version": "0.0.3", - "resolved": "https://registry.npmjs.org/object-component/-/object-component-0.0.3.tgz", - "integrity": "sha1-8MaapQ78lbhmwYb0AKM3acsvEpE=", - "dev": true - }, "object-copy": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/object-copy/-/object-copy-0.1.0.tgz", @@ -8624,6 +25307,43 @@ "is-descriptor": "^0.1.0" } }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + } + }, + "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": { + "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 + } + } + }, "kind-of": { "version": "3.2.2", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-3.2.2.tgz", @@ -8635,20 +25355,14 @@ } } }, - "object-inspect": { - "version": "1.8.0", - "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.8.0.tgz", - "integrity": "sha512-jLdtEOB112fORuypAyl/50VRVIBIdVQOSUUGQHzJ4xBSbit81zRarz7GThkEFZy1RceYrWYcPcBFPQwHyAc1gA==", - "dev": true - }, "object-is": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/object-is/-/object-is-1.1.3.tgz", - "integrity": "sha512-teyqLvFWzLkq5B9ki8FVWA902UER2qkxmdA4nLf+wjOLAWgxzCWZNCxpDq9MvE8MmhWNr+I8w3BN49Vx36Y6Xg==", + "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": { - "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.1" + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" } }, "object-keys": { @@ -8667,48 +25381,17 @@ } }, "object.assign": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.1.tgz", - "integrity": "sha512-VT/cxmx5yaoHSOTSyrCygIDFco+RsibY2NM0a4RdEeY/4KgqezwFtK1yr3U67xYhqJSlASm2pKhLVzPj2lr4bA==", + "version": "4.1.2", + "resolved": "https://registry.npmjs.org/object.assign/-/object.assign-4.1.2.tgz", + "integrity": "sha512-ixT2L5THXsApyiUPYKmW+2EHpXXe5Ii3M+f4e+aJFAHao5amFRW6J0OO6c/LU8Be47utCx2GL89hxGB6XSmKuQ==", "dev": true, "requires": { + "call-bind": "^1.0.0", "define-properties": "^1.1.3", - "es-abstract": "^1.18.0-next.0", "has-symbols": "^1.0.1", "object-keys": "^1.1.1" } }, - "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": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, "object.pick": { "version": "1.3.0", "resolved": "https://registry.npmjs.org/object.pick/-/object.pick-1.3.0.tgz", @@ -8718,39 +25401,6 @@ "isobject": "^3.0.1" } }, - "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": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1", - "function-bind": "^1.1.1", - "has": "^1.0.3" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, "obuf": { "version": "1.1.2", "resolved": "https://registry.npmjs.org/obuf/-/obuf-1.1.2.tgz", @@ -8790,9 +25440,9 @@ } }, "open": { - "version": "7.2.0", - "resolved": "https://registry.npmjs.org/open/-/open-7.2.0.tgz", - "integrity": "sha512-4HeyhxCvBTI5uBePsAdi55C5fmqnWZ2e2MlmvWi5KW5tdH5rxoiv/aMtbeVxKZc3eWkT1GymMnLG8XC4Rq4TDQ==", + "version": "7.4.0", + "resolved": "https://registry.npmjs.org/open/-/open-7.4.0.tgz", + "integrity": "sha512-PGoBCX/lclIWlpS/R2PQuIR4NJoXh6X5AwVzE7WXnWRGvHg7+4TBCgsujUgiPpm0K1y4qvQeWnCWVTpTKZBtvA==", "dev": true, "requires": { "is-docker": "^2.0.0", @@ -8822,17 +25472,17 @@ } }, "ora": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/ora/-/ora-5.0.0.tgz", - "integrity": "sha512-s26qdWqke2kjN/wC4dy+IQPBIMWBJlSU/0JZhk30ZDBLelW25rv66yutUWARMigpGPzcXHb+Nac5pNhN/WsARw==", + "version": "5.3.0", + "resolved": "https://registry.npmjs.org/ora/-/ora-5.3.0.tgz", + "integrity": "sha512-zAKMgGXUim0Jyd6CXK9lraBnD3H5yPGBPPOkC23a2BG6hsm4Zu6OQSjQuEtV0BHDf4aKHcUFvJiGRrFuW3MG8g==", "dev": true, "requires": { + "bl": "^4.0.3", "chalk": "^4.1.0", "cli-cursor": "^3.1.0", - "cli-spinners": "^2.4.0", + "cli-spinners": "^2.5.0", "is-interactive": "^1.0.0", "log-symbols": "^4.0.0", - "mute-stream": "0.0.8", "strip-ansi": "^6.0.0", "wcwidth": "^1.0.1" }, @@ -8853,9 +25503,9 @@ } }, "chalk": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.0.tgz", - "integrity": "sha512-qwx12AxXe2Q5xQ43Ac//I6v5aXTipYrSESdOgzrN+9XjgEpyjpKuvSGaN4qE93f7TQTlerQQ8S+EQ0EyDoVL1A==", + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", "dev": true, "requires": { "ansi-styles": "^4.1.0", @@ -8918,28 +25568,12 @@ "integrity": "sha1-hUNzx/XCMVkU/Jv8a9gjj92h7Cc=", "dev": true }, - "os-homedir": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/os-homedir/-/os-homedir-1.0.2.tgz", - "integrity": "sha1-/7xJiDNuDoM94MFox+8VISGqf7M=", - "dev": true - }, "os-tmpdir": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/os-tmpdir/-/os-tmpdir-1.0.2.tgz", "integrity": "sha1-u+Z0BseaqFxc/sdm/lc0VV36EnQ=", "dev": true }, - "osenv": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/osenv/-/osenv-0.1.5.tgz", - "integrity": "sha512-0CWcCECdMVc2Rw3U5w9ZjqX6ga6ubk1xDVKxtBQPK7wis/0F2r9T6k4ydGYhecl7YUBxBVxhL5oisPsNxAPe2g==", - "dev": true, - "requires": { - "os-homedir": "^1.0.0", - "os-tmpdir": "^1.0.0" - } - }, "p-finally": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/p-finally/-/p-finally-1.0.0.tgz", @@ -8950,17 +25584,18 @@ "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==", + "version": "4.1.0", + "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", + "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", "dev": true, "requires": { - "p-limit": "^2.0.0" + "p-limit": "^2.2.0" } }, "p-map": { @@ -8984,184 +25619,40 @@ "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==" + "integrity": "sha512-R4nPAVTAU0B9D35/Gk3uJf/7XYbQcyohSKdvAxIRSNghFl4e71hVoGnBNQz9cWaXxO2I10KTC+3jMdvvoKw6dQ==", + "dev": true }, "pacote": { - "version": "9.5.12", - "resolved": "https://registry.npmjs.org/pacote/-/pacote-9.5.12.tgz", - "integrity": "sha512-BUIj/4kKbwWg4RtnBncXPJd15piFSVNpTzY0rysSr3VnMowTYgkGKcaHrbReepAkjTr8lH2CVWRi58Spg2CicQ==", + "version": "11.2.4", + "resolved": "https://registry.npmjs.org/pacote/-/pacote-11.2.4.tgz", + "integrity": "sha512-GfTeVQGJ6WyBQbQD4t3ocHbyOmTQLmWjkCKSZPmKiGFKYKNUaM5U2gbLzUW8WG1XmS9yQFnsTFA0k3o1+q4klQ==", "dev": true, "requires": { - "bluebird": "^3.5.3", - "cacache": "^12.0.2", - "chownr": "^1.1.2", - "figgy-pudding": "^3.5.1", - "get-stream": "^4.1.0", - "glob": "^7.1.3", + "@npmcli/git": "^2.0.1", + "@npmcli/installed-package-contents": "^1.0.5", + "@npmcli/promise-spawn": "^1.2.0", + "@npmcli/run-script": "^1.3.0", + "cacache": "^15.0.5", + "chownr": "^2.0.0", + "fs-minipass": "^2.1.0", "infer-owner": "^1.0.4", - "lru-cache": "^5.1.1", - "make-fetch-happen": "^5.0.0", - "minimatch": "^3.0.4", - "minipass": "^2.3.5", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "normalize-package-data": "^2.4.0", - "npm-normalize-package-bin": "^1.0.0", - "npm-package-arg": "^6.1.0", - "npm-packlist": "^1.1.12", - "npm-pick-manifest": "^3.0.0", - "npm-registry-fetch": "^4.0.0", - "osenv": "^0.1.5", - "promise-inflight": "^1.0.1", + "minipass": "^3.1.3", + "mkdirp": "^1.0.3", + "npm-package-arg": "^8.0.1", + "npm-packlist": "^2.1.4", + "npm-pick-manifest": "^6.0.0", + "npm-registry-fetch": "^9.0.0", "promise-retry": "^1.1.1", - "protoduck": "^5.0.1", - "rimraf": "^2.6.2", - "safe-buffer": "^5.1.2", - "semver": "^5.6.0", - "ssri": "^6.0.1", - "tar": "^4.4.10", - "unique-filename": "^1.1.1", - "which": "^1.3.1" + "read-package-json-fast": "^1.1.3", + "rimraf": "^3.0.2", + "ssri": "^8.0.0", + "tar": "^6.1.0" }, "dependencies": { - "cacache": { - "version": "12.0.4", - "resolved": "https://registry.npmjs.org/cacache/-/cacache-12.0.4.tgz", - "integrity": "sha512-a0tMB40oefvuInr4Cwb3GerbL9xTj1D5yg0T5xrjGCGyfvbxseIXX7BAO/u/hIXdafzOI5JC3wDwHyf24buOAQ==", - "dev": true, - "requires": { - "bluebird": "^3.5.5", - "chownr": "^1.1.1", - "figgy-pudding": "^3.5.1", - "glob": "^7.1.4", - "graceful-fs": "^4.1.15", - "infer-owner": "^1.0.3", - "lru-cache": "^5.1.1", - "mississippi": "^3.0.0", - "mkdirp": "^0.5.1", - "move-concurrently": "^1.0.1", - "promise-inflight": "^1.0.1", - "rimraf": "^2.6.3", - "ssri": "^6.0.1", - "unique-filename": "^1.1.1", - "y18n": "^4.0.0" - } - }, - "chownr": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz", - "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", - "dev": true - }, - "fs-minipass": { - "version": "1.2.7", - "resolved": "https://registry.npmjs.org/fs-minipass/-/fs-minipass-1.2.7.tgz", - "integrity": "sha512-GWSSJGFy4e9GUeCcbIkED+bgAoFyj7XF1mV8rma3QW4NIqX9Kyx79N/PF61H5udOV3aY1IaMLs6pGbH71nlCTA==", - "dev": true, - "requires": { - "minipass": "^2.6.0" - } - }, - "hosted-git-info": { - "version": "2.8.8", - "resolved": "https://registry.npmjs.org/hosted-git-info/-/hosted-git-info-2.8.8.tgz", - "integrity": "sha512-f/wzC2QaWBs7t9IYqB4T3sR1xviIViXJRJTWBlx2Gf3g0Xi5vI7Yy4koXQ1c9OYDGHN9sBy1DQ2AB8fqZBWhUg==", - "dev": true - }, - "lru-cache": { - "version": "5.1.1", - "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", - "integrity": "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==", - "dev": true, - "requires": { - "yallist": "^3.0.2" - } - }, - "minipass": { - "version": "2.9.0", - "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.9.0.tgz", - "integrity": "sha512-wxfUjg9WebH+CUDX/CdbRlh5SmfZiy/hpkxaRI16Y9W56Pa75sWgd/rvFilSgrauD9NyFymP/+JFV3KwzIsJeg==", - "dev": true, - "requires": { - "safe-buffer": "^5.1.2", - "yallist": "^3.0.0" - } - }, - "minizlib": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/minizlib/-/minizlib-1.3.3.tgz", - "integrity": "sha512-6ZYMOEnmVsdCeTJVE0W9ZD+pVnE8h9Hma/iOwwRDsdQoePpoX56/8B6z3P9VNwppJuBKNRuFDRNRqRWexT9G9Q==", - "dev": true, - "requires": { - "minipass": "^2.9.0" - } - }, - "npm-package-arg": { - "version": "6.1.1", - "resolved": "https://registry.npmjs.org/npm-package-arg/-/npm-package-arg-6.1.1.tgz", - "integrity": "sha512-qBpssaL3IOZWi5vEKUKW0cO7kzLeT+EQO9W8RsLOZf76KF9E/K9+wH0C7t06HXPpaH8WH5xF1MExLuCwbTqRUg==", - "dev": true, - "requires": { - "hosted-git-info": "^2.7.1", - "osenv": "^0.1.5", - "semver": "^5.6.0", - "validate-npm-package-name": "^3.0.0" - } - }, - "npm-pick-manifest": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/npm-pick-manifest/-/npm-pick-manifest-3.0.2.tgz", - "integrity": "sha512-wNprTNg+X5nf+tDi+hbjdHhM4bX+mKqv6XmPh7B5eG+QY9VARfQPfCEH013H5GqfNj6ee8Ij2fg8yk0mzps1Vw==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1", - "npm-package-arg": "^6.0.0", - "semver": "^5.4.1" - } - }, - "rimraf": { - "version": "2.7.1", - "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", - "integrity": "sha512-uWjbaKIK3T1OSVptzX7Nl6PvQ3qAGtKEtVRjRuazjfL3Bx5eI409VZSqgND+4UNnmzLVdPj9FqFJNPqBZFve4w==", - "dev": true, - "requires": { - "glob": "^7.1.3" - } - }, - "semver": { - "version": "5.7.1", - "resolved": "https://registry.npmjs.org/semver/-/semver-5.7.1.tgz", - "integrity": "sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ==", - "dev": true - }, - "ssri": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.1.tgz", - "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", - "dev": true, - "requires": { - "figgy-pudding": "^3.5.1" - } - }, - "tar": { - "version": "4.4.13", - "resolved": "https://registry.npmjs.org/tar/-/tar-4.4.13.tgz", - "integrity": "sha512-w2VwSrBoHa5BsSyH+KxEqeQBAllHhccyMFVHtGtdMpF4W7IRWfZjFiQceJPChOeTsSDVUpER2T8FA93pr0L+QA==", - "dev": 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" - } - }, - "yallist": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", - "integrity": "sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g==", + "mkdirp": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-1.0.4.tgz", + "integrity": "sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==", "dev": true } } @@ -9183,6 +25674,15 @@ "readable-stream": "^2.1.5" } }, + "parent-module": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parent-module/-/parent-module-1.0.1.tgz", + "integrity": "sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g==", + "dev": true, + "requires": { + "callsites": "^3.0.0" + } + }, "parse-asn1": { "version": "5.1.6", "resolved": "https://registry.npmjs.org/parse-asn1/-/parse-asn1-5.1.6.tgz", @@ -9197,21 +25697,39 @@ } }, "parse-json": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-4.0.0.tgz", - "integrity": "sha1-vjX1Qlvh9/bHRxhPmKeIy5lHfuA=", + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/parse-json/-/parse-json-5.2.0.tgz", + "integrity": "sha512-ayCKvm/phCGxOkYRSCM82iDwct8/EonSEgCSxWxD7ve6jHggsFl4fZVQBPRNgQoKiuV/odhFrGzQXZwbifC8Rg==", "dev": true, "requires": { + "@babel/code-frame": "^7.0.0", "error-ex": "^1.3.1", - "json-parse-better-errors": "^1.0.1" + "json-parse-even-better-errors": "^2.3.0", + "lines-and-columns": "^1.1.6" } }, + "parse-node-version": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/parse-node-version/-/parse-node-version-1.0.1.tgz", + "integrity": "sha512-3YHlOa/JgH6Mnpr05jP9eDG254US9ek25LyIxZlDItp2iJtwyaXQb57lBYLdT3MowkUFYEV2XXNAYIPlESvJlA==", + "dev": true + }, "parse5": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/parse5/-/parse5-6.0.1.tgz", "integrity": "sha512-Ofn/CTFzRGTTxwpNEs9PP93gXShHcTq255nzRYSKe8AkVpZY7e1fpmTfOyoIvjP5HG7Z2ZM7VS9PPhQGW2pOpw==", "dev": true }, + "parse5-html-rewriting-stream": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5-html-rewriting-stream/-/parse5-html-rewriting-stream-6.0.1.tgz", + "integrity": "sha512-vwLQzynJVEfUlURxgnf51yAJDQTtVpNyGD8tKi2Za7m+akukNHxCcUQMAa/mUGLhCeicFdpy7Tlvj8ZNKadprg==", + "dev": true, + "requires": { + "parse5": "^6.0.1", + "parse5-sax-parser": "^6.0.1" + } + }, "parse5-htmlparser2-tree-adapter": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-6.0.1.tgz", @@ -9221,22 +25739,13 @@ "parse5": "^6.0.1" } }, - "parseqs": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/parseqs/-/parseqs-0.0.5.tgz", - "integrity": "sha1-1SCKNzjkZ2bikbouoXNoSSGouJ0=", + "parse5-sax-parser": { + "version": "6.0.1", + "resolved": "https://registry.npmjs.org/parse5-sax-parser/-/parse5-sax-parser-6.0.1.tgz", + "integrity": "sha512-kXX+5S81lgESA0LsDuGjAlBybImAChYRMT+/uKCEXFBFOeEhS52qUCydGhU3qLRD8D9DVjaUo821WK7DM4iCeg==", "dev": true, "requires": { - "better-assert": "~1.0.0" - } - }, - "parseuri": { - "version": "0.0.5", - "resolved": "https://registry.npmjs.org/parseuri/-/parseuri-0.0.5.tgz", - "integrity": "sha1-gCBKUNTbt3m/3G6+J3jZDkvOMgo=", - "dev": true, - "requires": { - "better-assert": "~1.0.0" + "parse5": "^6.0.1" } }, "parseurl": { @@ -9264,9 +25773,9 @@ "dev": true }, "path-exists": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-3.0.0.tgz", - "integrity": "sha1-zg6+ql94yxiSXqfYENe1mwEP1RU=", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", + "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", "dev": true }, "path-is-absolute": { @@ -9304,9 +25813,9 @@ "dev": true }, "pbkdf2": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.1.tgz", - "integrity": "sha512-4Ejy1OPxi9f2tt1rRV7Go7zmfDQ+ZectEQz3VGUQhgq62HtIRPDyG/JtnwIxs6x3uNMwo2V7q1fMvKjb+Tnpqg==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/pbkdf2/-/pbkdf2-3.1.2.tgz", + "integrity": "sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==", "dev": true, "requires": { "create-hash": "^1.1.2", @@ -9328,10 +25837,9 @@ "dev": true }, "picomatch": { - "version": "2.2.2", - "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz", - "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==", - "dev": true + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.0.tgz", + "integrity": "sha512-lY1Q/PiJGC2zOv/z391WOTD+Z02bCgsFfvxoXXf6h7kv9o+WmsmzYqrAwY63sNgOxE4xEdq0WyUnXfKeBrSvYw==" }, "pify": { "version": "4.0.1", @@ -9355,12 +25863,12 @@ } }, "pkg-dir": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", - "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-4.2.0.tgz", + "integrity": "sha512-HRDzbaKjC+AOWVXxAU/x54COGeIv9eb+6CkDSQoNTt4XyWoIJvuPsXizxu/Fr23EiekbtZwmh1IcIG/l/a10GQ==", "dev": true, "requires": { - "find-up": "^3.0.0" + "find-up": "^4.0.0" } }, "pnp-webpack-plugin": { @@ -9384,9 +25892,9 @@ }, "dependencies": { "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { "ms": "^2.1.1" @@ -9401,14 +25909,14 @@ "dev": true }, "postcss": { - "version": "7.0.32", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.32.tgz", - "integrity": "sha512-03eXong5NLnNCD05xscnGKGDZ98CyzoqPSMjOe6SuoQY7Z2hIj0Ld1g/O/UQRuOle2aRtiIRDg9tDcTGAkLfKw==", + "version": "8.2.15", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-8.2.15.tgz", + "integrity": "sha512-2zO3b26eJD/8rb106Qu2o7Qgg52ND5HPjcyQiK2B98O388h43A448LCslC0dI2P97wCAQRJsFvwTRcXxTKds+Q==", "dev": true, "requires": { - "chalk": "^2.4.2", - "source-map": "^0.6.1", - "supports-color": "^6.1.0" + "colorette": "^1.2.2", + "nanoid": "^3.1.23", + "source-map": "^0.6.1" }, "dependencies": { "source-map": { @@ -9416,606 +25924,360 @@ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", "dev": true - }, - "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" - } } } }, "postcss-calc": { - "version": "7.0.5", - "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-7.0.5.tgz", - "integrity": "sha512-1tKHutbGtLtEZF6PT4JSihCHfIVldU72mZ8SdZHIYriIZ9fh9k9aWSppaT8rHsyI3dX+KSR+W+Ix9BMY3AODrg==", + "version": "8.0.0", + "resolved": "https://registry.npmjs.org/postcss-calc/-/postcss-calc-8.0.0.tgz", + "integrity": "sha512-5NglwDrcbiy8XXfPM11F3HeC6hoT9W7GUH/Zi5U/p7u3Irv4rHhdDcIZwG0llHXV4ftsBjpfWMXAnXNl4lnt8g==", "dev": true, "requires": { - "postcss": "^7.0.27", "postcss-selector-parser": "^6.0.2", "postcss-value-parser": "^4.0.2" } }, "postcss-colormin": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-4.0.3.tgz", - "integrity": "sha512-WyQFAdDZpExQh32j0U0feWisZ0dmOtPl44qYmJKkq9xFWY3p+4qnRzCHeNrkeRhwPHz9bQ3mo0/yVkaply0MNw==", + "version": "5.2.0", + "resolved": "https://registry.npmjs.org/postcss-colormin/-/postcss-colormin-5.2.0.tgz", + "integrity": "sha512-+HC6GfWU3upe5/mqmxuqYZ9B2Wl4lcoUUNkoaX59nEWV4EtADCMiBqui111Bu8R8IvaZTmqmxrqOAqjbHIwXPw==", "dev": true, "requires": { - "browserslist": "^4.0.0", - "color": "^3.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "browserslist": "^4.16.6", + "caniuse-api": "^3.0.0", + "colord": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-convert-values/-/postcss-convert-values-5.0.1.tgz", + "integrity": "sha512-C3zR1Do2BkKkCgC0g3sF8TS0koF2G+mN8xxayZx3f10cIRmTaAnpgpRQZjNekTZxM2ciSPoh2IWJm0VZx8NoQg==", "dev": true, "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-comments/-/postcss-discard-comments-5.0.1.tgz", + "integrity": "sha512-lgZBPTDvWrbAYY1v5GYEv8fEO/WhKOu/hmZqmCYfrpD6eyDWWzAOsl2rF29lpvziKO02Gc5GJQtlpkTmakwOWg==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } + "requires": {} }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-duplicates/-/postcss-discard-duplicates-5.0.1.tgz", + "integrity": "sha512-svx747PWHKOGpAXXQkCc4k/DsWo+6bc5LsVrAsw+OU+Ibi7klFZCyX54gjYzX4TH+f2uzXjRviLARxkMurA2bA==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } + "requires": {} }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-empty/-/postcss-discard-empty-5.0.1.tgz", + "integrity": "sha512-vfU8CxAQ6YpMxV2SvMcMIyF2LX1ZzWpy0lqHDsOdaKKLQVQGVP1pzhrI9JlsO65s66uQTfkQBKBD/A5gp9STFw==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } + "requires": {} }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-discard-overridden/-/postcss-discard-overridden-5.0.1.tgz", + "integrity": "sha512-Y28H7y93L2BpJhrdUR2SR2fnSsT+3TVx1NmVQLbcnZWwIUpJ7mfcTC6Za9M2PG6w8j7UQRfzxqn8jU2VwFxo3Q==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } + "requires": {} }, "postcss-import": { - "version": "12.0.1", - "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-12.0.1.tgz", - "integrity": "sha512-3Gti33dmCjyKBgimqGxL3vcV8w9+bsHwO5UrBawp796+jdardbcFl4RP5w/76BwNL7aGzpKstIfF9I+kdE8pTw==", + "version": "14.0.0", + "resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-14.0.0.tgz", + "integrity": "sha512-gFDDzXhqr9ELmnLHgCC3TbGfA6Dm/YMb/UN8/f7Uuq4fL7VTk2vOIj6hwINEwbokEmp123bLD7a5m+E+KIetRg==", "dev": true, "requires": { - "postcss": "^7.0.1", - "postcss-value-parser": "^3.2.3", + "postcss-value-parser": "^4.0.0", "read-cache": "^1.0.0", "resolve": "^1.1.7" - }, - "dependencies": { - "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 - } - } - }, - "postcss-load-config": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-2.1.2.tgz", - "integrity": "sha512-/rDeGV6vMUo3mwJZmeHfEDvwnTKKqQ0S7OHUi/kJvvtx3aWtyWG2/0ZWnzCt2keEclwN6Tf0DST2v9kITdOKYw==", - "dev": true, - "requires": { - "cosmiconfig": "^5.0.0", - "import-cwd": "^2.0.0" } }, "postcss-loader": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-3.0.0.tgz", - "integrity": "sha512-cLWoDEY5OwHcAjDnkyRQzAXfs2jrKjXpO/HQFcc5b5u/r7aa471wdmChmwfnv7x2u840iat/wi0lQ5nbRgSkUA==", + "version": "4.2.0", + "resolved": "https://registry.npmjs.org/postcss-loader/-/postcss-loader-4.2.0.tgz", + "integrity": "sha512-mqgScxHqbiz1yxbnNcPdKYo/6aVt+XExURmEbQlviFVWogDbM4AJ0A/B+ZBpYsJrTRxKw7HyRazg9x0Q9SWwLA==", "dev": true, "requires": { - "loader-utils": "^1.1.0", - "postcss": "^7.0.0", - "postcss-load-config": "^2.0.0", - "schema-utils": "^1.0.0" + "cosmiconfig": "^7.0.0", + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "schema-utils": "^3.0.0", + "semver": "^7.3.4" }, "dependencies": { - "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": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" - } - }, "schema-utils": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-1.0.0.tgz", - "integrity": "sha512-i27Mic4KovM/lnGsy8whRCHhc7VicJajAjTrYg11K9zfZXnYIt4k5F+kZkwjnrhKzLic/HLU4j11mjsz2G/75g==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", "dev": true, "requires": { - "ajv": "^6.1.0", - "ajv-errors": "^1.0.0", - "ajv-keywords": "^3.1.0" + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, + "semver": { + "version": "7.3.5", + "resolved": "https://registry.npmjs.org/semver/-/semver-7.3.5.tgz", + "integrity": "sha512-PoeGJYh8HK4BTO/a9Tf6ZG3veo/A7ZVsYrSA6J8ny9nb3B1VrpkuN+z9OE5wfE5p6H4LchYZsegiQgbJD94ZFQ==", + "dev": true, + "requires": { + "lru-cache": "^6.0.0" } } } }, "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==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-merge-longhand/-/postcss-merge-longhand-5.0.2.tgz", + "integrity": "sha512-BMlg9AXSI5G9TBT0Lo/H3PfUy63P84rVz3BjCFE9e9Y9RXQZD3+h3YO1kgTNsNJy7bBc1YQp8DmSnwLIW5VPcw==", "dev": true, "requires": { - "css-color-names": "0.0.4", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0", - "stylehacks": "^4.0.0" - }, - "dependencies": { - "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 - } + "css-color-names": "^1.0.1", + "postcss-value-parser": "^4.1.0", + "stylehacks": "^5.0.1" } }, "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==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-merge-rules/-/postcss-merge-rules-5.0.2.tgz", + "integrity": "sha512-5K+Md7S3GwBewfB4rjDeol6V/RZ8S+v4B66Zk2gChRqLTCC8yjnHQ601omj9TKftS19OPGqZ/XzoqpzNQQLwbg==", "dev": true, "requires": { - "browserslist": "^4.0.0", + "browserslist": "^4.16.6", "caniuse-api": "^3.0.0", - "cssnano-util-same-parent": "^4.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0", - "vendors": "^1.0.0" - }, - "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" - } - } + "cssnano-utils": "^2.0.1", + "postcss-selector-parser": "^6.0.5", + "vendors": "^1.0.3" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-font-values/-/postcss-minify-font-values-5.0.1.tgz", + "integrity": "sha512-7JS4qIsnqaxk+FXY1E8dHBDmraYFWmuL6cgt0T1SWGRO5bzJf8sUoelwa4P88LEWJZweHevAiDKxHlofuvtIoA==", "dev": true, "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-gradients/-/postcss-minify-gradients-5.0.1.tgz", + "integrity": "sha512-odOwBFAIn2wIv+XYRpoN2hUV3pPQlgbJ10XeXPq8UY2N+9ZG42xu45lTn/g9zZ+d70NKSQD6EOi6UiCMu3FN7g==", "dev": true, "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "is-color-stop": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "is-color-stop": "^1.1.0", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-minify-params/-/postcss-minify-params-5.0.1.tgz", + "integrity": "sha512-4RUC4k2A/Q9mGco1Z8ODc7h+A0z7L7X2ypO1B6V8057eVK6mZ6xwz6QN64nHuHLbqbclkX1wyzRnIrdZehTEHw==", "dev": true, "requires": { - "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", + "alphanum-sort": "^1.0.2", + "browserslist": "^4.16.0", + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0", "uniqs": "^2.0.0" - }, - "dependencies": { - "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 - } } }, "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==", + "version": "5.1.0", + "resolved": "https://registry.npmjs.org/postcss-minify-selectors/-/postcss-minify-selectors-5.1.0.tgz", + "integrity": "sha512-NzGBXDa7aPsAcijXZeagnJBKBPMYLaJJzB8CQh6ncvyl2sIndLVWfbcDi0SBjRWk5VqEjXvf8tYwzoKf4Z07og==", "dev": true, "requires": { - "alphanum-sort": "^1.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0" - }, - "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" - } - } + "alphanum-sort": "^1.0.2", + "postcss-selector-parser": "^6.0.5" } }, "postcss-modules-extract-imports": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-2.0.0.tgz", - "integrity": "sha512-LaYLDNS4SG8Q5WAWqIJgdHPJrDDr/Lv775rMBFUbgjTz6j34lUznACHcdRWroPvXANP2Vj7yNK57vp9eFqzLWQ==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-extract-imports/-/postcss-modules-extract-imports-3.0.0.tgz", + "integrity": "sha512-bdHleFnP3kZ4NYDhuGlVK+CMrQ/pqUm8bx/oGL93K6gVwiclvX5x0n76fYMKuIGKzlABOy13zsvqjb0f92TEXw==", "dev": true, - "requires": { - "postcss": "^7.0.5" - } + "requires": {} }, "postcss-modules-local-by-default": { - "version": "3.0.3", - "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-3.0.3.tgz", - "integrity": "sha512-e3xDq+LotiGesympRlKNgaJ0PCzoUIdpH0dj47iWAui/kyTgh3CiAr1qP54uodmJhl6p9rN6BoNcdEDVJx9RDw==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-local-by-default/-/postcss-modules-local-by-default-4.0.0.tgz", + "integrity": "sha512-sT7ihtmGSF9yhm6ggikHdV0hlziDTX7oFoXtuVWeDd3hHObNkcHRo9V3yg7vCAY7cONyxJC/XXCmmiHHcvX7bQ==", "dev": true, "requires": { - "icss-utils": "^4.1.1", - "postcss": "^7.0.32", + "icss-utils": "^5.0.0", "postcss-selector-parser": "^6.0.2", "postcss-value-parser": "^4.1.0" } }, "postcss-modules-scope": { - "version": "2.2.0", - "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-2.2.0.tgz", - "integrity": "sha512-YyEgsTMRpNd+HmyC7H/mh3y+MeFWevy7V1evVhJWewmMbjDHIbZbOXICC2y+m1xI1UVfIT1HMW/O04Hxyu9oXQ==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-scope/-/postcss-modules-scope-3.0.0.tgz", + "integrity": "sha512-hncihwFA2yPath8oZ15PZqvWGkWf+XUfQgUGamS4LqoP1anQLOsOJw0vr7J7IwLpoY9fatA2qiGUGmuZL0Iqlg==", "dev": true, "requires": { - "postcss": "^7.0.6", - "postcss-selector-parser": "^6.0.0" + "postcss-selector-parser": "^6.0.4" } }, "postcss-modules-values": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-3.0.0.tgz", - "integrity": "sha512-1//E5jCBrZ9DmRX+zCtmQtRSV6PV42Ix7Bzj9GbwJceduuf7IqP8MgeTXuRDHOWj2m0VzZD5+roFWDuU8RQjcg==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/postcss-modules-values/-/postcss-modules-values-4.0.0.tgz", + "integrity": "sha512-RDxHkAiEGI78gS2ofyvCsu7iycRv7oqw5xMWn9iMoR0N/7mf9D50ecQqUo5BZ9Zh2vH4bCUR/ktCqbB9m8vJjQ==", "dev": true, "requires": { - "icss-utils": "^4.0.0", - "postcss": "^7.0.6" + "icss-utils": "^5.0.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-charset/-/postcss-normalize-charset-5.0.1.tgz", + "integrity": "sha512-6J40l6LNYnBdPSk+BHZ8SF+HAkS4q2twe5jnocgd+xWpz/mx/5Sa32m3W1AA8uE8XaXN+eg8trIlfu8V9x61eg==", "dev": true, - "requires": { - "postcss": "^7.0.0" - } + "requires": {} }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-display-values/-/postcss-normalize-display-values-5.0.1.tgz", + "integrity": "sha512-uupdvWk88kLDXi5HEyI9IaAJTE3/Djbcrqq8YgjvAVuzgVuqIk3SuJWUisT2gaJbZm1H9g5k2w1xXilM3x8DjQ==", "dev": true, "requires": { - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-positions/-/postcss-normalize-positions-5.0.1.tgz", + "integrity": "sha512-rvzWAJai5xej9yWqlCb1OWLd9JjW2Ex2BCPzUJrbaXmtKtgfL8dBMOOMTX6TnvQMtjk3ei1Lswcs78qKO1Skrg==", "dev": true, "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-repeat-style/-/postcss-normalize-repeat-style-5.0.1.tgz", + "integrity": "sha512-syZ2itq0HTQjj4QtXZOeefomckiV5TaUO6ReIEabCh3wgDs4Mr01pkif0MeVwKyU/LHEkPJnpwFKRxqWA/7O3w==", "dev": true, "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-string/-/postcss-normalize-string-5.0.1.tgz", + "integrity": "sha512-Ic8GaQ3jPMVl1OEn2U//2pm93AXUcF3wz+OriskdZ1AOuYV25OdgS7w9Xu2LO5cGyhHCgn8dMXh9bO7vi3i9pA==", "dev": true, "requires": { - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-timing-functions/-/postcss-normalize-timing-functions-5.0.1.tgz", + "integrity": "sha512-cPcBdVN5OsWCNEo5hiXfLUnXfTGtSFiBU9SK8k7ii8UD7OLuznzgNRYkLZow11BkQiiqMcgPyh4ZqXEEUrtQ1Q==", "dev": true, "requires": { - "cssnano-util-get-match": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-unicode/-/postcss-normalize-unicode-5.0.1.tgz", + "integrity": "sha512-kAtYD6V3pK0beqrU90gpCQB7g6AOfP/2KIPCVBKJM2EheVsBQmx/Iof+9zR9NFKLAx4Pr9mDhogB27pmn354nA==", "dev": true, "requires": { - "browserslist": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "browserslist": "^4.16.0", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-normalize-url/-/postcss-normalize-url-5.0.2.tgz", + "integrity": "sha512-k4jLTPUxREQ5bpajFQZpx8bCF2UrlqOTzP9kEqcEnOfwsRshWs2+oAFIHfDQB8GO2PaUaSE0NlTAYtbluZTlHQ==", "dev": true, "requires": { - "is-absolute-url": "^2.0.0", - "normalize-url": "^3.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "is-absolute-url": "^3.0.3", + "normalize-url": "^6.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-normalize-whitespace/-/postcss-normalize-whitespace-5.0.1.tgz", + "integrity": "sha512-iPklmI5SBnRvwceb/XH568yyzK0qRVuAG+a1HFUsFRf11lEJTiQQa03a4RSCQvLKdcpX7XsI1Gen9LuLoqwiqA==", "dev": true, "requires": { - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-ordered-values/-/postcss-ordered-values-5.0.2.tgz", + "integrity": "sha512-8AFYDSOYWebJYLyJi3fyjl6CqMEG/UVworjiyK1r573I56kb3e879sCJLGvR3merj+fAdPpVplXKQZv+ey6CgQ==", "dev": true, "requires": { - "cssnano-util-get-arguments": "^4.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-reduce-initial/-/postcss-reduce-initial-5.0.1.tgz", + "integrity": "sha512-zlCZPKLLTMAqA3ZWH57HlbCjkD55LX9dsRyxlls+wfuRfqCi5mSlZVan0heX5cHr154Dq9AfbH70LyhrSAezJw==", "dev": true, "requires": { - "browserslist": "^4.0.0", - "caniuse-api": "^3.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0" + "browserslist": "^4.16.0", + "caniuse-api": "^3.0.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-reduce-transforms/-/postcss-reduce-transforms-5.0.1.tgz", + "integrity": "sha512-a//FjoPeFkRuAguPscTVmRQUODP+f3ke2HqFNgGPwdYnpeC29RZdCBvGRGTsKpMURb/I3p6jdKoBQ2zI+9Q7kA==", "dev": true, "requires": { - "cssnano-util-get-match": "^4.0.0", - "has": "^1.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0" - }, - "dependencies": { - "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 - } + "cssnano-utils": "^2.0.1", + "postcss-value-parser": "^4.1.0" } }, "postcss-selector-parser": { - "version": "6.0.4", - "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.4.tgz", - "integrity": "sha512-gjMeXBempyInaBqpp8gODmwZ52WaYsVOsfr4L4lDQ7n3ncD6mEyySiDtgzCT+NYC0mmeOLvtsF8iaEf0YT6dBw==", + "version": "6.0.6", + "resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.0.6.tgz", + "integrity": "sha512-9LXrvaaX3+mcv5xkg5kFwqSzSH1JIObIx51PrndZwlmznwXRfxMddDvo9gve3gVR8ZTKgoFDdWkbRFmEhT4PMg==", "dev": true, "requires": { "cssesc": "^3.0.0", - "indexes-of": "^1.0.1", - "uniq": "^1.0.1", "util-deprecate": "^1.0.2" } }, "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==", + "version": "5.0.2", + "resolved": "https://registry.npmjs.org/postcss-svgo/-/postcss-svgo-5.0.2.tgz", + "integrity": "sha512-YzQuFLZu3U3aheizD+B1joQ94vzPfE6BNUcSYuceNxlVnKKsOtdo6hL9/zyC168Q8EwfLSgaDSalsUGa9f2C0A==", "dev": true, "requires": { - "is-svg": "^3.0.0", - "postcss": "^7.0.0", - "postcss-value-parser": "^3.0.0", - "svgo": "^1.0.0" - }, - "dependencies": { - "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 - } + "postcss-value-parser": "^4.1.0", + "svgo": "^2.3.0" } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/postcss-unique-selectors/-/postcss-unique-selectors-5.0.1.tgz", + "integrity": "sha512-gwi1NhHV4FMmPn+qwBNuot1sG1t2OmacLQ/AX29lzyggnjd+MnVD5uqQmpXO3J17KGL2WAxQruj1qTd3H0gG/w==", "dev": true, "requires": { - "alphanum-sort": "^1.0.0", - "postcss": "^7.0.0", + "alphanum-sort": "^1.0.2", + "postcss-selector-parser": "^6.0.5", "uniqs": "^2.0.0" } }, @@ -10025,10 +26287,10 @@ "integrity": "sha512-97DXOFbQJhk71ne5/Mt6cOu6yxsSfM0QGQyl0L25Gca4yGWEGJaig7l7gbCX623VqTBNGLRLaVUCnNkcedlRSQ==", "dev": true }, - "prepend-http": { - "version": "1.0.4", - "resolved": "https://registry.npmjs.org/prepend-http/-/prepend-http-1.0.4.tgz", - "integrity": "sha1-1PRWKwzjaW5BrFLQ4ALlemNdxtw=", + "pretty-bytes": { + "version": "5.6.0", + "resolved": "https://registry.npmjs.org/pretty-bytes/-/pretty-bytes-5.6.0.tgz", + "integrity": "sha512-FFw039TmrBqFK8ma/7OL3sDz/VytdtJr044/QUJtH0wK9lb9jLq9tJyIxUwtQJHwar2BqtiA4iCWSwo9JLkzFg==", "dev": true }, "process": { @@ -10067,15 +26329,6 @@ } } }, - "protoduck": { - "version": "5.0.1", - "resolved": "https://registry.npmjs.org/protoduck/-/protoduck-5.0.1.tgz", - "integrity": "sha512-WxoCeDCoCBY55BMvj4cAEjdVUFGRWed9ZxPlqTKYyw1nDDTQ4pqmnIMAGfJlg7Dx35uB/M+PHJPTmGOvaCaPTg==", - "dev": true, - "requires": { - "genfun": "^5.0.0" - } - }, "protractor": { "version": "7.0.0", "resolved": "https://registry.npmjs.org/protractor/-/protractor-7.0.0.tgz", @@ -10126,12 +26379,6 @@ "array-uniq": "^1.0.1" } }, - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, "chalk": { "version": "1.1.3", "resolved": "https://registry.npmjs.org/chalk/-/chalk-1.1.3.tgz", @@ -10203,16 +26450,6 @@ "integrity": "sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==", "dev": true }, - "find-up": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/find-up/-/find-up-4.1.0.tgz", - "integrity": "sha512-PpOwAdQ/YlXQ2vj8a3h8IipDuYRi3wceVQQGYWxNINccq40Anw7BlsEXCMbt1Zt+OLA6Fq9suIpIWD0OsnISlw==", - "dev": true, - "requires": { - "locate-path": "^5.0.0", - "path-exists": "^4.0.0" - } - }, "globby": { "version": "5.0.0", "resolved": "https://registry.npmjs.org/globby/-/globby-5.0.0.tgz", @@ -10257,30 +26494,6 @@ "path-is-inside": "^1.0.1" } }, - "locate-path": { - "version": "5.0.0", - "resolved": "https://registry.npmjs.org/locate-path/-/locate-path-5.0.0.tgz", - "integrity": "sha512-t7hw9pI+WvuwNJXwk5zVHpyhIqzg2qTlklJOf0mVxGSbe3Fp2VieZcduNYjaLDoy6p9uGpQEGWG87WpMKlNq8g==", - "dev": true, - "requires": { - "p-locate": "^4.1.0" - } - }, - "p-locate": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/p-locate/-/p-locate-4.1.0.tgz", - "integrity": "sha512-R79ZZ/0wAxKGu3oYMlz8jy/kbhsNrS7SKZ7PxEHBgJ5+F2mtFW2fK2cOtBh1cHYkQsbzFV7I+EoRKe6Yt0oK7A==", - "dev": true, - "requires": { - "p-limit": "^2.2.0" - } - }, - "path-exists": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/path-exists/-/path-exists-4.0.0.tgz", - "integrity": "sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==", - "dev": true - }, "pify": { "version": "2.3.0", "resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz", @@ -10433,12 +26646,12 @@ } }, "proxy-addr": { - "version": "2.0.6", - "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", - "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", + "version": "2.0.7", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz", + "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==", "dev": true, "requires": { - "forwarded": "~0.1.2", + "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, @@ -10469,9 +26682,9 @@ }, "dependencies": { "bn.js": { - "version": "4.11.9", - "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.11.9.tgz", - "integrity": "sha512-E6QoYqCKZfgatHTdHzs1RRKP7ip4vvm+EyRUeE2RF0NblwVvb0p6jSVeNTOFxPn26QXN2o6SMfNxKp6kU8zQaw==", + "version": "4.12.0", + "resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.0.tgz", + "integrity": "sha512-c98Bf3tPniI+scsdk237ku1Dc3ujXQTSgyiPUDEOe7tRkhrqridvh8klBv0HCEso1OLOYcHuCv/cS6DNxKH+ZA==", "dev": true } } @@ -10515,12 +26728,6 @@ "integrity": "sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A==", "dev": true }, - "q": { - "version": "1.5.1", - "resolved": "https://registry.npmjs.org/q/-/q-1.5.1.tgz", - "integrity": "sha1-fjL3W0E4EpHQRhHxvxQQmsAGUdc=", - "dev": true - }, "qjobs": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/qjobs/-/qjobs-1.2.0.tgz", @@ -10533,16 +26740,6 @@ "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==", "dev": true }, - "query-string": { - "version": "4.3.4", - "resolved": "https://registry.npmjs.org/query-string/-/query-string-4.3.4.tgz", - "integrity": "sha1-u7aTucqRXCMlFbIosaArYJBD2+s=", - "dev": true, - "requires": { - "object-assign": "^4.1.0", - "strict-uri-encode": "^1.0.0" - } - }, "querystring": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/querystring/-/querystring-0.2.0.tgz", @@ -10561,6 +26758,12 @@ "integrity": "sha512-FIqgj2EUvTa7R50u0rGsyTftzjYmv/a3hO345bZNrqabNqjtgiDMgmo4mkUjd+nzU5oF3dClKqFIPUKybUyqoQ==", "dev": true }, + "queue-microtask": { + "version": "1.2.3", + "resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz", + "integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==", + "dev": true + }, "randombytes": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz", @@ -10603,26 +26806,30 @@ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==", "dev": true - }, - "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": { - "safer-buffer": ">= 2.1.2 < 3" - } } } }, "raw-loader": { - "version": "4.0.1", - "resolved": "https://registry.npmjs.org/raw-loader/-/raw-loader-4.0.1.tgz", - "integrity": "sha512-baolhQBSi3iNh1cglJjA0mYzga+wePk7vdEX//1dTFd+v4TsQlQE0jitJSNF1OIP82rdYulH7otaVmdlDaJ64A==", + "version": "4.0.2", + "resolved": "https://registry.npmjs.org/raw-loader/-/raw-loader-4.0.2.tgz", + "integrity": "sha512-ZnScIV3ag9A4wPX/ZayxL/jZH+euYb6FcUinPcgiQW0+UBtEv0O6Q3lGd3cqJ+GHH+rksEv3Pj99oxJ3u3VIKA==", "dev": true, "requires": { "loader-utils": "^2.0.0", - "schema-utils": "^2.6.5" + "schema-utils": "^3.0.0" + }, + "dependencies": { + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, "read-cache": { @@ -10642,27 +26849,14 @@ } } }, - "read-package-json": { - "version": "2.1.2", - "resolved": "https://registry.npmjs.org/read-package-json/-/read-package-json-2.1.2.tgz", - "integrity": "sha512-D1KmuLQr6ZSJS0tW8hf3WGpRlwszJOXZ3E8Yd/DNRaM5d+1wVRZdHlpGBLAuovjr28LbWvjpWkBHMxpRGGjzNA==", + "read-package-json-fast": { + "version": "1.2.2", + "resolved": "https://registry.npmjs.org/read-package-json-fast/-/read-package-json-fast-1.2.2.tgz", + "integrity": "sha512-39DbPJjkltEzfXJXB6D8/Ir3GFOU2YbSKa2HaB/Y3nKrc/zY+0XrALpID6/13ezWyzqvOHrBbR4t4cjQuTdBVQ==", "dev": true, "requires": { - "glob": "^7.1.1", "json-parse-even-better-errors": "^2.3.0", - "normalize-package-data": "^2.0.0", - "npm-normalize-package-bin": "^1.0.0" - } - }, - "read-package-tree": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/read-package-tree/-/read-package-tree-5.3.1.tgz", - "integrity": "sha512-mLUDsD5JVtlZxjSlPPx1RETkNjjvQYuweKwNVt1Sn8kP5Jh44pvYuUHCp6xSVDZWbNxVxG5lyZJ921aJH61sTw==", - "dev": true, - "requires": { - "read-package-json": "^2.0.0", - "readdir-scoped-modules": "^1.0.0", - "util-promisify": "^2.1.0" + "npm-normalize-package-bin": "^1.0.1" } }, "readable-stream": { @@ -10680,23 +26874,10 @@ "util-deprecate": "~1.0.1" } }, - "readdir-scoped-modules": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/readdir-scoped-modules/-/readdir-scoped-modules-1.1.0.tgz", - "integrity": "sha512-asaikDeqAQg7JifRsZn1NJZXo9E+VwlyCfbkZhwyISinqk5zNS6266HS5kah6P0SaQKGF6SkNnZVHUzHFYxYDw==", - "dev": true, - "requires": { - "debuglog": "^1.0.1", - "dezalgo": "^1.0.0", - "graceful-fs": "^4.1.2", - "once": "^1.3.0" - } - }, "readdirp": { "version": "3.4.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz", "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==", - "dev": true, "requires": { "picomatch": "^2.2.1" } @@ -10704,13 +26885,12 @@ "reflect-metadata": { "version": "0.1.13", "resolved": "https://registry.npmjs.org/reflect-metadata/-/reflect-metadata-0.1.13.tgz", - "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==", - "dev": true + "integrity": "sha512-Ts1Y/anZELhSsjMcU605fU9RE4Oi3p5ORujwbIKXfWa+0Zxs510Qrmrce5/Jowq3cHSZSJqBjypxmHarc+vEWg==" }, "regenerate": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.1.tgz", - "integrity": "sha512-j2+C8+NtXQgEKWk49MMP5P/u2GhnahTtVkRIHr5R5lVRlbKvmQ+oS+A5aLKWp2ma5VkT8sh6v+v4hbH0YHR66A==", + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/regenerate/-/regenerate-1.4.2.tgz", + "integrity": "sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A==", "dev": true }, "regenerate-unicode-properties": { @@ -10754,34 +26934,13 @@ "dev": true }, "regexp.prototype.flags": { - "version": "1.3.0", - "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.3.0.tgz", - "integrity": "sha512-2+Q0C5g951OlYlJz6yu5/M33IcsESLlLfsyIaLJaG4FA2r4yP8MvVMJUUP/fVBkSpbbbZlS5gynbEWLipiiXiQ==", + "version": "1.3.1", + "resolved": "https://registry.npmjs.org/regexp.prototype.flags/-/regexp.prototype.flags-1.3.1.tgz", + "integrity": "sha512-JiBdRBq91WlY7uRJ0ds7R+dU02i6LKi8r3BuQhNXn+kmeLN+EfHhfjqMRis1zJxnlu88hq/4dx0P2OP3APRTOA==", "dev": true, "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.0-next.1" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } + "call-bind": "^1.0.2", + "define-properties": "^1.1.3" } }, "regexpu-core": { @@ -10805,9 +26964,9 @@ "dev": true }, "regjsparser": { - "version": "0.6.4", - "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.4.tgz", - "integrity": "sha512-64O87/dPDgfk8/RQqC4gkZoGyyWFIEUTTh80CU6CWuK5vkCGyekIx+oKcEIYtP/RAxSQltCZHCNu/mdd7fqlJw==", + "version": "0.6.9", + "resolved": "https://registry.npmjs.org/regjsparser/-/regjsparser-0.6.9.tgz", + "integrity": "sha512-ZqbNRz1SNjLAiYuwY0zoXW8Ne675IX5q+YHioAGbCw4X96Mjl2+dcX9B2ciaeyYjViDAfvIjFpQjJgLttTEERQ==", "dev": true, "requires": { "jsesc": "~0.5.0" @@ -10828,9 +26987,9 @@ "dev": true }, "repeat-element": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.3.tgz", - "integrity": "sha512-ahGq0ZnV5m5XtZLMb+vP76kcAM5nkLqk0lpqAuojSKGgQtn4eRi4ZZGm2olo2zKFH+sMsWaqOCW1dqAnOru72g==", + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/repeat-element/-/repeat-element-1.1.4.tgz", + "integrity": "sha512-LFiNfRcSu7KK3evMyYOuCzv3L10TW7yC1G2/+StMjK8Y6Vqd2MG7r/Qjw4ghtuCOjFvlnms/iMmLqpvW/ES/WQ==", "dev": true }, "repeat-string": { @@ -10889,7 +27048,8 @@ "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==" + "integrity": "sha512-NKN5kMDylKuldxYLSUfrbo5Tuzh4hd+2E8NPPX02mZtn1VuREQToYe/ZdlJy+J3uCpfaiGF05e7B8W0iXbQHmg==", + "dev": true }, "requires-port": { "version": "1.0.0", @@ -10898,10 +27058,11 @@ "dev": true }, "resolve": { - "version": "1.17.0", - "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz", - "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==", + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.19.0.tgz", + "integrity": "sha512-rArEXAgsBG4UgRGcynxWIWKFvh/XZCcS8UJdHhwy91zwAvCZIbcs+vAbflgBnNjYMs/i/i+/Ux6IZhML1yPvxg==", "requires": { + "is-core-module": "^2.1.0", "path-parse": "^1.0.6" } }, @@ -10912,12 +27073,20 @@ "dev": true, "requires": { "resolve-from": "^3.0.0" + }, + "dependencies": { + "resolve-from": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", + "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "dev": true + } } }, "resolve-from": { - "version": "3.0.0", - "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-3.0.0.tgz", - "integrity": "sha1-six699nWiBvItuZTM17rywoYh0g=", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-from/-/resolve-from-4.0.0.tgz", + "integrity": "sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g==", "dev": true }, "resolve-url": { @@ -10927,59 +27096,22 @@ "dev": true }, "resolve-url-loader": { - "version": "3.1.2", - "resolved": "https://registry.npmjs.org/resolve-url-loader/-/resolve-url-loader-3.1.2.tgz", - "integrity": "sha512-QEb4A76c8Mi7I3xNKXlRKQSlLBwjUV/ULFMP+G7n3/7tJZ8MG5wsZ3ucxP1Jz8Vevn6fnJsxDx9cIls+utGzPQ==", + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/resolve-url-loader/-/resolve-url-loader-4.0.0.tgz", + "integrity": "sha512-05VEMczVREcbtT7Bz+C+96eUO5HDNvdthIiMB34t7FcF8ehcu4wC0sSgPUubs3XW2Q3CNLJk/BJrCU9wVRymiA==", "dev": true, "requires": { - "adjust-sourcemap-loader": "3.0.0", - "camelcase": "5.3.1", - "compose-function": "3.0.3", - "convert-source-map": "1.7.0", - "es6-iterator": "2.0.3", - "loader-utils": "1.2.3", - "postcss": "7.0.21", - "rework": "1.0.1", - "rework-visit": "1.0.0", + "adjust-sourcemap-loader": "^4.0.0", + "convert-source-map": "^1.7.0", + "loader-utils": "^2.0.0", + "postcss": "^7.0.35", "source-map": "0.6.1" }, "dependencies": { - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - }, - "emojis-list": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/emojis-list/-/emojis-list-2.1.0.tgz", - "integrity": "sha1-TapNnbAPmBmIDHn6RXrlsJof04k=", - "dev": true - }, - "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": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.2.3", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.2.3.tgz", - "integrity": "sha512-fkpz8ejdnEMG3s37wGL07iSBDg99O9D5yflE9RGNH3hRdx9SOwYfnGYdZOUIZitN8E+E2vkq3MUMYMvPYl5ZZA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^2.0.0", - "json5": "^1.0.1" - } - }, "postcss": { - "version": "7.0.21", - "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.21.tgz", - "integrity": "sha512-uIFtJElxJo29QC753JzhidoAhvp/e/Exezkdhfmt8AymWT6/5B7W1WmponYWkHk2eg6sONyTch0A3nkMPun3SQ==", + "version": "7.0.36", + "resolved": "https://registry.npmjs.org/postcss/-/postcss-7.0.36.tgz", + "integrity": "sha512-BebJSIUMwJHRH0HAQoxN4u1CN86glsrwsW0q7T+/m44eXOUAxSNdHRkNZPYz5vVUbg17hFgOQDE7fZk7li3pZw==", "dev": true, "requires": { "chalk": "^2.4.2", @@ -11032,34 +27164,10 @@ "integrity": "sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw==", "dev": true }, - "rework": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/rework/-/rework-1.0.1.tgz", - "integrity": "sha1-MIBqhBNCtUUQqkEQhQzUhTQUSqc=", - "dev": true, - "requires": { - "convert-source-map": "^0.3.3", - "css": "^2.0.0" - }, - "dependencies": { - "convert-source-map": { - "version": "0.3.5", - "resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-0.3.5.tgz", - "integrity": "sha1-8dgClQr33SYxof6+BZZVDIarMZA=", - "dev": true - } - } - }, - "rework-visit": { - "version": "1.0.0", - "resolved": "https://registry.npmjs.org/rework-visit/-/rework-visit-1.0.0.tgz", - "integrity": "sha1-mUWygD8hni96ygCtuLyfZA+ELJo=", - "dev": true - }, "rfdc": { - "version": "1.1.4", - "resolved": "https://registry.npmjs.org/rfdc/-/rfdc-1.1.4.tgz", - "integrity": "sha512-5C9HXdzK8EAqN7JDif30jqsBzavB7wLpaubisuQIGHWf2gUXSpzy6ArX/+Da8RjFpagWsCn+pIgxTMAmKw9Zug==", + "version": "1.3.0", + "resolved": "https://registry.npmjs.org/rfdc/-/rfdc-1.3.0.tgz", + "integrity": "sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA==", "dev": true }, "rgb-regex": { @@ -11094,12 +27202,21 @@ } }, "rollup": { - "version": "2.26.5", - "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.26.5.tgz", - "integrity": "sha512-rCyFG3ZtQdnn9YwfuAVH0l/Om34BdO5lwCA0W6Hq+bNB21dVEBbCRxhaHOmu1G7OBFDWytbzAC104u7rxHwGjA==", + "version": "2.38.4", + "resolved": "https://registry.npmjs.org/rollup/-/rollup-2.38.4.tgz", + "integrity": "sha512-B0LcJhjiwKkTl79aGVF/u5KdzsH8IylVfV56Ut6c9ouWLJcUK17T83aZBetNYSnZtXf2OHD4+2PbmRW+Fp5ulg==", "dev": true, "requires": { - "fsevents": "~2.1.2" + "fsevents": "~2.3.1" + }, + "dependencies": { + "fsevents": { + "version": "2.3.2", + "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.2.tgz", + "integrity": "sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA==", + "dev": true, + "optional": true + } } }, "run-async": { @@ -11109,10 +27226,13 @@ "dev": true }, "run-parallel": { - "version": "1.1.9", - "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.1.9.tgz", - "integrity": "sha512-DEqnSRTDw/Tc3FXf49zedI638Z9onwUotBMiUFKmrO2sdFKIbXamXGQ3Axd4qgphxKB4kw/qP1w5kTxnfU1B9Q==", - "dev": true + "version": "1.2.0", + "resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz", + "integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==", + "dev": true, + "requires": { + "queue-microtask": "^1.2.2" + } }, "run-queue": { "version": "1.0.3", @@ -11159,25 +27279,38 @@ "dev": true }, "sass": { - "version": "1.26.10", - "resolved": "https://registry.npmjs.org/sass/-/sass-1.26.10.tgz", - "integrity": "sha512-bzN0uvmzfsTvjz0qwccN1sPm2HxxpNI/Xa+7PlUEMS+nQvbyuEK7Y0qFqxlPHhiNHb1Ze8WQJtU31olMObkAMw==", + "version": "1.32.6", + "resolved": "https://registry.npmjs.org/sass/-/sass-1.32.6.tgz", + "integrity": "sha512-1bcDHDcSqeFtMr0JXI3xc/CXX6c4p0wHHivJdru8W7waM7a1WjKMm4m/Z5sY7CbVw4Whi2Chpcw6DFfSWwGLzQ==", "dev": true, "requires": { "chokidar": ">=2.0.0 <4.0.0" } }, "sass-loader": { - "version": "10.0.1", - "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-10.0.1.tgz", - "integrity": "sha512-b2PSldKVTS3JcFPHSrEXh3BeAfR7XknGiGCAO5aHruR3Pf3kqLP3Gb2ypXLglRrAzgZkloNxLZ7GXEGDX0hBUQ==", + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/sass-loader/-/sass-loader-10.1.1.tgz", + "integrity": "sha512-W6gVDXAd5hR/WHsPicvZdjAWHBcEJ44UahgxcIE196fW2ong0ZHMPO1kZuI5q0VlvMQZh32gpv69PLWQm70qrw==", "dev": true, "requires": { - "klona": "^2.0.3", + "klona": "^2.0.4", "loader-utils": "^2.0.0", "neo-async": "^2.6.2", - "schema-utils": "^2.7.0", + "schema-utils": "^3.0.0", "semver": "^7.3.2" + }, + "dependencies": { + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, "saucelabs": { @@ -11245,9 +27378,9 @@ } }, "selfsigned": { - "version": "1.10.8", - "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-1.10.8.tgz", - "integrity": "sha512-2P4PtieJeEwVgTU9QEcwIRDQ/mXJLX8/+I3ur+Pg16nS8oNbrGxEso9NyYWy8NAmXiNl4dlAp5MwoNeCWzON4w==", + "version": "1.10.11", + "resolved": "https://registry.npmjs.org/selfsigned/-/selfsigned-1.10.11.tgz", + "integrity": "sha512-aVmbPOfViZqOZPgRBT0+3u4yZFHpmnIghLMlAcb5/xhp5ZtB/RVnKhz5vl2M32CLXAqR4kha9zfhNg0Lf/sxKA==", "dev": true, "requires": { "node-forge": "^0.10.0" @@ -11340,9 +27473,9 @@ } }, "serialize-javascript": { - "version": "4.0.0", - "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", - "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-5.0.1.tgz", + "integrity": "sha512-SaaNal9imEO737H2c05Og0/8LUXG7EnsZyMa8MzkmuHoELfT6txuj0cMqRj6zfPKnmQ1yasR4PCJc8x+M4JSPA==", "dev": true, "requires": { "randombytes": "^2.1.0" @@ -11419,7 +27552,8 @@ "set-blocking": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/set-blocking/-/set-blocking-2.0.0.tgz", - "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=" + "integrity": "sha1-BF+XgtARrppoA93TgrJDkrPYkPc=", + "dev": true }, "set-immediate-shim": { "version": "1.0.1", @@ -11447,6 +27581,12 @@ "requires": { "is-extendable": "^0.1.0" } + }, + "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 } } }, @@ -11472,6 +27612,15 @@ "safe-buffer": "^5.0.1" } }, + "shallow-clone": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/shallow-clone/-/shallow-clone-3.0.1.tgz", + "integrity": "sha512-/6KqX+GVUdqPuPPd2LxDDxzX6CAbjJehAAOKlNpqqUpAqPM6HeL8f+o3a+JsyGjn2lv0WY8UsTgUJjU9Ok55NA==", + "dev": true, + "requires": { + "kind-of": "^6.0.2" + } + }, "shebang-command": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-1.2.0.tgz", @@ -11493,23 +27642,6 @@ "integrity": "sha512-VUJ49FC8U1OxwZLxIbTTrDvLnf/6TDgxZcK8wxR8zs13xpx7xbG60ndBlhNrFi2EMuFRoeDoJO7wthSLq42EjA==", "dev": true }, - "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 - } - } - }, "slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -11565,6 +27697,69 @@ "is-extendable": "^0.1.0" } }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-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": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "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 + }, + "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 + }, "ms": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", @@ -11576,6 +27771,19 @@ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.5.7.tgz", "integrity": "sha1-igOdLRAh0i0eoUyA2OpGi6LvP8w=", "dev": true + }, + "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==", + "dev": true, + "requires": { + "atob": "^2.1.2", + "decode-uri-component": "^0.2.0", + "resolve-url": "^0.2.1", + "source-map-url": "^0.4.0", + "urix": "^0.1.0" + } } } }, @@ -11598,35 +27806,6 @@ "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" - } } } }, @@ -11651,164 +27830,48 @@ } }, "socket.io": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-2.3.0.tgz", - "integrity": "sha512-2A892lrj0GcgR/9Qk81EaY2gYhCBxurV0PfmmESO6p27QPrUK1J3zdns+5QPqvUYK2q657nSj0guoIil9+7eFg==", + "version": "3.1.2", + "resolved": "https://registry.npmjs.org/socket.io/-/socket.io-3.1.2.tgz", + "integrity": "sha512-JubKZnTQ4Z8G4IZWtaAZSiRP3I/inpy8c/Bsx2jrwGrTbKeVU5xd6qkKMHpChYeM3dWZSO0QACiGK+obhBNwYw==", "dev": true, "requires": { - "debug": "~4.1.0", - "engine.io": "~3.4.0", - "has-binary2": "~1.0.2", - "socket.io-adapter": "~1.1.0", - "socket.io-client": "2.3.0", - "socket.io-parser": "~3.4.0" - }, - "dependencies": { - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - } + "@types/cookie": "^0.4.0", + "@types/cors": "^2.8.8", + "@types/node": ">=10.0.0", + "accepts": "~1.3.4", + "base64id": "~2.0.0", + "debug": "~4.3.1", + "engine.io": "~4.1.0", + "socket.io-adapter": "~2.1.0", + "socket.io-parser": "~4.0.3" } }, "socket.io-adapter": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-1.1.2.tgz", - "integrity": "sha512-WzZRUj1kUjrTIrUKpZLEzFZ1OLj5FwLlAFQs9kuZJzJi5DKdU7FsWc36SNmA8iDOtwBQyT8FkrriRM8vXLYz8g==", + "version": "2.1.0", + "resolved": "https://registry.npmjs.org/socket.io-adapter/-/socket.io-adapter-2.1.0.tgz", + "integrity": "sha512-+vDov/aTsLjViYTwS9fPy5pEtTkrbEKsw2M+oVSoFGw6OD1IpvlV1VPhUzNbofCQ8oyMbdYJqDtGdmHQK6TdPg==", "dev": true }, - "socket.io-client": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/socket.io-client/-/socket.io-client-2.3.0.tgz", - "integrity": "sha512-cEQQf24gET3rfhxZ2jJ5xzAOo/xhZwK+mOqtGRg5IowZsMgwvHwnf/mCRapAAkadhM26y+iydgwsXGObBB5ZdA==", - "dev": true, - "requires": { - "backo2": "1.0.2", - "base64-arraybuffer": "0.1.5", - "component-bind": "1.0.0", - "component-emitter": "1.2.1", - "debug": "~4.1.0", - "engine.io-client": "~3.4.0", - "has-binary2": "~1.0.2", - "has-cors": "1.1.0", - "indexof": "0.0.1", - "object-component": "0.0.3", - "parseqs": "0.0.5", - "parseuri": "0.0.5", - "socket.io-parser": "~3.3.0", - "to-array": "0.1.4" - }, - "dependencies": { - "base64-arraybuffer": { - "version": "0.1.5", - "resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-0.1.5.tgz", - "integrity": "sha1-c5JncZI7Whl0etZmqlzUv5xunOg=", - "dev": true - }, - "component-emitter": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", - "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=", - "dev": true - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "isarray": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", - "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", - "dev": true - }, - "socket.io-parser": { - "version": "3.3.1", - "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.3.1.tgz", - "integrity": "sha512-1QLvVAe8dTz+mKmZ07Swxt+LAo4Y1ff50rlyoEx00TQmDFVQYPfcqGvIDJLGaBdhdNCecXtyKpD+EgKGcmmbuQ==", - "dev": true, - "requires": { - "component-emitter": "~1.3.0", - "debug": "~3.1.0", - "isarray": "2.0.1" - }, - "dependencies": { - "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 - }, - "debug": { - "version": "3.1.0", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.1.0.tgz", - "integrity": "sha512-OX8XqP7/1a9cqkxYw2yXss15f26NKWBpDXQd0/uK/KPqdQhxbPa994hnzjcE2VqQpDslf55723cKPUOGSmMY3g==", - "dev": true, - "requires": { - "ms": "2.0.0" - } - }, - "ms": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", - "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=", - "dev": true - } - } - } - } - }, "socket.io-parser": { - "version": "3.4.1", - "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-3.4.1.tgz", - "integrity": "sha512-11hMgzL+WCLWf1uFtHSNvliI++tcRUWdoeYuwIl+Axvwy9z2gQM+7nJyN3STj1tLj5JyIUH8/gpDGxzAlDdi0A==", + "version": "4.0.4", + "resolved": "https://registry.npmjs.org/socket.io-parser/-/socket.io-parser-4.0.4.tgz", + "integrity": "sha512-t+b0SS+IxG7Rxzda2EVvyBZbvFPBCjJoyHuE0P//7OAsN23GItzDRdWa6ALxZI/8R5ygK7jAR6t028/z+7295g==", "dev": true, "requires": { - "component-emitter": "1.2.1", - "debug": "~4.1.0", - "isarray": "2.0.1" - }, - "dependencies": { - "component-emitter": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/component-emitter/-/component-emitter-1.2.1.tgz", - "integrity": "sha1-E3kY1teCg/ffemt8WmPhQOaUJeY=", - "dev": true - }, - "debug": { - "version": "4.1.1", - "resolved": "https://registry.npmjs.org/debug/-/debug-4.1.1.tgz", - "integrity": "sha512-pYAIzeRo8J6KPEaJ0VWOh5Pzkbw/RetuzehGM7QRRX5he4fPHx2rdKMB256ehJCkX+XRQm16eZLqLNS8RSZXZw==", - "dev": true, - "requires": { - "ms": "^2.1.1" - } - }, - "isarray": { - "version": "2.0.1", - "resolved": "https://registry.npmjs.org/isarray/-/isarray-2.0.1.tgz", - "integrity": "sha1-o32U7ZzaLVmGXJ92/llu4fM4dB4=", - "dev": true - } + "@types/component-emitter": "^1.2.10", + "component-emitter": "~1.3.0", + "debug": "~4.3.1" } }, "sockjs": { - "version": "0.3.20", - "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.20.tgz", - "integrity": "sha512-SpmVOVpdq0DJc0qArhF3E5xsxvaiqGNb73XfgBpK1y3UD5gs8DSo8aCTsuT5pX8rssdc2NDIzANwP9eCAiSdTA==", + "version": "0.3.21", + "resolved": "https://registry.npmjs.org/sockjs/-/sockjs-0.3.21.tgz", + "integrity": "sha512-DhbPFGpxjc6Z3I+uX07Id5ZO2XwYsWOrYjaSeieES78cq+JaJvVe5q/m1uvjIQhXinhIeCFRH6JgXe+mvVMyXw==", "dev": true, "requires": { - "faye-websocket": "^0.10.0", + "faye-websocket": "^0.11.3", "uuid": "^3.4.0", - "websocket-driver": "0.6.5" + "websocket-driver": "^0.7.4" }, "dependencies": { "uuid": { @@ -11820,79 +27883,62 @@ } }, "sockjs-client": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/sockjs-client/-/sockjs-client-1.4.0.tgz", - "integrity": "sha512-5zaLyO8/nri5cua0VtOrFXBPK1jbL4+1cebT/mmKA1E1ZXOvJrII75bPu0l0k843G/+iAbhEqzyKr0w/eCCj7g==", + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/sockjs-client/-/sockjs-client-1.5.1.tgz", + "integrity": "sha512-VnVAb663fosipI/m6pqRXakEOw7nvd7TUgdr3PlR/8V2I95QIdwT8L4nMxhyU8SmDBHYXU1TOElaKOmKLfYzeQ==", "dev": true, "requires": { - "debug": "^3.2.5", + "debug": "^3.2.6", "eventsource": "^1.0.7", - "faye-websocket": "~0.11.1", - "inherits": "^2.0.3", - "json3": "^3.3.2", - "url-parse": "^1.4.3" + "faye-websocket": "^0.11.3", + "inherits": "^2.0.4", + "json3": "^3.3.3", + "url-parse": "^1.5.1" }, "dependencies": { "debug": { - "version": "3.2.6", - "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.6.tgz", - "integrity": "sha512-mel+jf7nrtEl5Pn1Qx46zARXKDpBbvzezse7p7LqINmdoIk8PYP5SySaxEmYv6TZ0JyEKA1hsCId6DIhgITtWQ==", + "version": "3.2.7", + "resolved": "https://registry.npmjs.org/debug/-/debug-3.2.7.tgz", + "integrity": "sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ==", "dev": true, "requires": { "ms": "^2.1.1" } - }, - "faye-websocket": { - "version": "0.11.3", - "resolved": "https://registry.npmjs.org/faye-websocket/-/faye-websocket-0.11.3.tgz", - "integrity": "sha512-D2y4bovYpzziGgbHYtGCMjlJM36vAl/y+xUyn1C+FVx8szd1E+86KwVw6XvYSzOP8iMpm1X0I4xJD+QtUb36OA==", - "dev": true, - "requires": { - "websocket-driver": ">=0.5.1" - } } } }, "socks": { - "version": "2.3.3", - "resolved": "https://registry.npmjs.org/socks/-/socks-2.3.3.tgz", - "integrity": "sha512-o5t52PCNtVdiOvzMry7wU4aOqYWL0PeCXRWBEiJow4/i/wr+wpsJQ9awEu1EonLIqsfGd5qSgDdxEOvCdmBEpA==", + "version": "2.6.1", + "resolved": "https://registry.npmjs.org/socks/-/socks-2.6.1.tgz", + "integrity": "sha512-kLQ9N5ucj8uIcxrDwjm0Jsqk06xdpBjGNQtpXy4Q8/QY2k+fY7nZH8CARy+hkbG+SGAovmzzuauCpBlb8FrnBA==", "dev": true, "requires": { - "ip": "1.1.5", + "ip": "^1.1.5", "smart-buffer": "^4.1.0" } }, "socks-proxy-agent": { - "version": "4.0.2", - "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-4.0.2.tgz", - "integrity": "sha512-NT6syHhI9LmuEMSK6Kd2V7gNv5KFZoLE7V5udWmn0de+3Mkj3UMA/AJPLyeNUVmElCurSHtUdM3ETpR3z770Wg==", + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/socks-proxy-agent/-/socks-proxy-agent-5.0.0.tgz", + "integrity": "sha512-lEpa1zsWCChxiynk+lCycKuC502RxDWLKJZoIhnxrWNjLSDGYRFflHA1/228VkRcnv9TIb8w98derGbpKxJRgA==", "dev": true, "requires": { - "agent-base": "~4.2.1", - "socks": "~2.3.2" + "agent-base": "6", + "debug": "4", + "socks": "^2.3.3" }, "dependencies": { "agent-base": { - "version": "4.2.1", - "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-4.2.1.tgz", - "integrity": "sha512-JVwXMr9nHYTUXsBFKUqhJwvlcYU/blreOEUkhNR2eXZIvwd+c+o5V4MgDPKWnMS/56awN3TRzIP+KoPn+roQtg==", + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/agent-base/-/agent-base-6.0.2.tgz", + "integrity": "sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ==", "dev": true, "requires": { - "es6-promisify": "^5.0.0" + "debug": "4" } } } }, - "sort-keys": { - "version": "1.1.2", - "resolved": "https://registry.npmjs.org/sort-keys/-/sort-keys-1.1.2.tgz", - "integrity": "sha1-RBttTTRnmPG05J6JIK37oOVD+a0=", - "dev": true, - "requires": { - "is-plain-obj": "^1.0.0" - } - }, "source-list-map": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/source-list-map/-/source-list-map-2.0.1.tgz", @@ -11906,18 +27952,39 @@ "dev": true }, "source-map-loader": { - "version": "1.0.2", - "resolved": "https://registry.npmjs.org/source-map-loader/-/source-map-loader-1.0.2.tgz", - "integrity": "sha512-oX8d6ndRjN+tVyjj6PlXSyFPhDdVAPsZA30nD3/II8g4uOv8fCz0DMn5sy8KtVbDfKQxOpGwGJnK3xIW3tauDw==", + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/source-map-loader/-/source-map-loader-1.1.3.tgz", + "integrity": "sha512-6YHeF+XzDOrT/ycFJNI53cgEsp/tHTMl37hi7uVyqFAlTXW109JazaQCkbc+jjoL2637qkH1amLi+JzrIpt5lA==", "dev": true, "requires": { - "data-urls": "^2.0.0", + "abab": "^2.0.5", "iconv-lite": "^0.6.2", "loader-utils": "^2.0.0", - "schema-utils": "^2.7.0", - "source-map": "^0.6.1" + "schema-utils": "^3.0.0", + "source-map": "^0.6.1", + "whatwg-mimetype": "^2.3.0" }, "dependencies": { + "iconv-lite": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz", + "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==", + "dev": true, + "requires": { + "safer-buffer": ">= 2.1.2 < 3.0.0" + } + }, + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", @@ -11927,16 +27994,13 @@ } }, "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==", + "version": "0.6.0", + "resolved": "https://registry.npmjs.org/source-map-resolve/-/source-map-resolve-0.6.0.tgz", + "integrity": "sha512-KXBr9d/fO/bWo97NXsPIAW1bFSBOuCnjbNTBMO7N59hsv5i9yzRDfcYwwt0l04+VqnKC+EwzvJZIP/qkuMgR/w==", "dev": true, "requires": { "atob": "^2.1.2", - "decode-uri-component": "^0.2.0", - "resolve-url": "^0.2.1", - "source-map-url": "^0.4.0", - "urix": "^0.1.0" + "decode-uri-component": "^0.2.0" } }, "source-map-support": { @@ -11958,48 +28022,15 @@ } }, "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=", + "version": "0.4.1", + "resolved": "https://registry.npmjs.org/source-map-url/-/source-map-url-0.4.1.tgz", + "integrity": "sha512-cPiFOTLUKvJFIg4SKVScy4ilPPW6rFgMgfuZJPNoDuMs3nC1HbMUycBoJw77xFIp6z1UJQJOfx6C9GMH80DiTw==", "dev": true }, "sourcemap-codec": { "version": "1.4.8", "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz", - "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==", - "dev": true - }, - "spdx-correct": { - "version": "3.1.1", - "resolved": "https://registry.npmjs.org/spdx-correct/-/spdx-correct-3.1.1.tgz", - "integrity": "sha512-cOYcUWwhCuHCXi49RhFRCyJEK3iPj1Ziz9DpViV3tbZOwXD49QzIN3MpOLJNxh2qwq2lJJZaKMVw9qNi4jTC0w==", - "dev": true, - "requires": { - "spdx-expression-parse": "^3.0.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-exceptions": { - "version": "2.3.0", - "resolved": "https://registry.npmjs.org/spdx-exceptions/-/spdx-exceptions-2.3.0.tgz", - "integrity": "sha512-/tTrYOC7PPI1nUAgx34hUpqXuyJG+DTHJTnIULG4rDygi4xu/tfgmq1e1cIRwRzwZgo4NLySi+ricLkZkw4i5A==", - "dev": true - }, - "spdx-expression-parse": { - "version": "3.0.1", - "resolved": "https://registry.npmjs.org/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz", - "integrity": "sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q==", - "dev": true, - "requires": { - "spdx-exceptions": "^2.1.0", - "spdx-license-ids": "^3.0.0" - } - }, - "spdx-license-ids": { - "version": "3.0.6", - "resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.6.tgz", - "integrity": "sha512-+orQK83kyMva3WyPf59k1+Y525csj5JejicWut55zeTWANuN17qSiSLUXWtzHeNWORSvT7GLDJ/E/XiIWoXBTw==", - "dev": true + "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==" }, "spdy": { "version": "4.0.2", @@ -12042,12 +28073,63 @@ } }, "speed-measure-webpack-plugin": { - "version": "1.3.3", - "resolved": "https://registry.npmjs.org/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.3.3.tgz", - "integrity": "sha512-2ljD4Ch/rz2zG3HsLsnPfp23osuPBS0qPuz9sGpkNXTN1Ic4M+W9xB8l8rS8ob2cO4b1L+WTJw/0AJwWYVgcxQ==", + "version": "1.4.2", + "resolved": "https://registry.npmjs.org/speed-measure-webpack-plugin/-/speed-measure-webpack-plugin-1.4.2.tgz", + "integrity": "sha512-AtVzD0bnIy2/B0fWqJpJgmhcrfWFhBlduzSo0uwplr/QvB33ZNZj2NEth3NONgdnZJqicK0W0mSxnLSbsVCDbw==", "dev": true, "requires": { - "chalk": "^2.0.1" + "chalk": "^4.1.0" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } } }, "split-string": { @@ -12083,9 +28165,9 @@ } }, "ssri": { - "version": "8.0.0", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.0.tgz", - "integrity": "sha512-aq/pz989nxVYwn16Tsbj1TqFpD5LLrQxHf5zaHuieFV+R0Bbr4y8qUsOA45hXT/N4/9UNXTarBjnjVmjSOVaAA==", + "version": "8.0.1", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-8.0.1.tgz", + "integrity": "sha512-97qShzy1AiyxvPNIkLWoGua7xoQzzPjQ0HAH4B0rWKo7SZ6USuPcrUiAFrws0UH8RrbWmgq3LMTObhPIHbbBeQ==", "dev": true, "requires": { "minipass": "^3.1.1" @@ -12115,6 +28197,63 @@ "requires": { "is-descriptor": "^0.1.0" } + }, + "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=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-data-descriptor": { + "version": "0.1.4", + "resolved": "https://registry.npmjs.org/is-data-descriptor/-/is-data-descriptor-0.1.4.tgz", + "integrity": "sha1-C17mSDiOLIYCgueT8YVv7D8wG1Y=", + "dev": true, + "requires": { + "kind-of": "^3.0.2" + }, + "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-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": { + "is-accessor-descriptor": "^0.1.6", + "is-data-descriptor": "^0.1.4", + "kind-of": "^5.0.0" + } + }, + "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 } } }, @@ -12193,11 +28332,14 @@ } } }, - "strict-uri-encode": { - "version": "1.1.0", - "resolved": "https://registry.npmjs.org/strict-uri-encode/-/strict-uri-encode-1.1.0.tgz", - "integrity": "sha1-J5siXfHVgrH1TmWt3UNS4Y+qBxM=", - "dev": true + "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" + } }, "string-width": { "version": "3.1.0", @@ -12227,77 +28369,6 @@ } } }, - "string.prototype.trimend": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimend/-/string.prototype.trimend-1.0.1.tgz", - "integrity": "sha512-LRPxFUaTtpqYsTeNKaFOw3R4bxIzWOnbQ837QfBylo8jIxtcbK/A/sMV7Q+OAV/vWo+7s25pOE10KYSjaSO06g==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, - "string.prototype.trimstart": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/string.prototype.trimstart/-/string.prototype.trimstart-1.0.1.tgz", - "integrity": "sha512-XxZn+QpvrBI1FOcg6dIpxUPgWCPuNXvMD72aaRaUQv1eD4e/Qy8i/hFTe0BUmD60p/QA6bh1avmuPTfNjqVWRw==", - "dev": true, - "requires": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.5" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, - "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": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", @@ -12314,37 +28385,36 @@ "dev": true }, "style-loader": { - "version": "1.2.1", - "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-1.2.1.tgz", - "integrity": "sha512-ByHSTQvHLkWE9Ir5+lGbVOXhxX10fbprhLvdg96wedFZb4NDekDPxVKv5Fwmio+QcMlkkNfuK+5W1peQ5CUhZg==", + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/style-loader/-/style-loader-2.0.0.tgz", + "integrity": "sha512-Z0gYUJmzZ6ZdRUqpg1r8GsaFKypE+3xAzuFeMuoHgjc9KZv3wMyCRjQIWEbhoFSq7+7yoHXySDJyyWQaPajeiQ==", "dev": true, "requires": { "loader-utils": "^2.0.0", - "schema-utils": "^2.6.6" + "schema-utils": "^3.0.0" + }, + "dependencies": { + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" + } + } } }, "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==", + "version": "5.0.1", + "resolved": "https://registry.npmjs.org/stylehacks/-/stylehacks-5.0.1.tgz", + "integrity": "sha512-Es0rVnHIqbWzveU1b24kbw92HsebBepxfcqe5iix7t9j0PQqhs0IxXVXv0pY2Bxa08CgMkzD6OWql7kbGOuEdA==", "dev": true, "requires": { - "browserslist": "^4.0.0", - "postcss": "^7.0.0", - "postcss-selector-parser": "^3.0.0" - }, - "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" - } - } + "browserslist": "^4.16.0", + "postcss-selector-parser": "^6.0.4" } }, "stylus": { @@ -12393,34 +28463,27 @@ } }, "stylus-loader": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/stylus-loader/-/stylus-loader-3.0.2.tgz", - "integrity": "sha512-+VomPdZ6a0razP+zinir61yZgpw2NfljeSsdUF5kJuEzlo3khXhY19Fn6l8QQz1GRJGtMCo8nG5C04ePyV7SUA==", + "version": "4.3.3", + "resolved": "https://registry.npmjs.org/stylus-loader/-/stylus-loader-4.3.3.tgz", + "integrity": "sha512-PpWB5PnCXUzW4WMYhCvNzAHJBjIBPMXwsdfkkKuA9W7k8OQFMl/19/AQvaWsxz2IptxUlCseyJ6TY/eEKJ4+UQ==", "dev": true, "requires": { - "loader-utils": "^1.0.2", - "lodash.clonedeep": "^4.5.0", - "when": "~3.6.x" + "fast-glob": "^3.2.4", + "klona": "^2.0.4", + "loader-utils": "^2.0.0", + "normalize-path": "^3.0.0", + "schema-utils": "^3.0.0" }, "dependencies": { - "json5": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/json5/-/json5-1.0.1.tgz", - "integrity": "sha512-aKS4WQjPenRxiQsC93MNfjx+nbF4PAdYzmd/1JIj8HYzqfbu86beTuNgXDzPknWk0n0uARlyewZo4s++ES36Ow==", + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", "dev": true, "requires": { - "minimist": "^1.2.0" - } - }, - "loader-utils": { - "version": "1.4.0", - "resolved": "https://registry.npmjs.org/loader-utils/-/loader-utils-1.4.0.tgz", - "integrity": "sha512-qH0WSMBtn/oHuwjy/NucEgbx5dbxxnxup9s4PVXJUDHZBQY+s0NWA9rJf53RBnQZxfch7euUui7hpoAPvALZdA==", - "dev": true, - "requires": { - "big.js": "^5.2.2", - "emojis-list": "^3.0.0", - "json5": "^1.0.1" + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" } } } @@ -12434,42 +28497,93 @@ } }, "svgo": { - "version": "1.3.2", - "resolved": "https://registry.npmjs.org/svgo/-/svgo-1.3.2.tgz", - "integrity": "sha512-yhy/sQYxR5BkC98CY7o31VGsg014AKLEPxdfhora76l36hD9Rdy5NZA/Ocn6yayNPgSamYdtX2rFJdcv07AYVw==", + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/svgo/-/svgo-2.3.0.tgz", + "integrity": "sha512-fz4IKjNO6HDPgIQxu4IxwtubtbSfGEAJUq/IXyTPIkGhWck/faiiwfkvsB8LnBkKLvSoyNNIY6d13lZprJMc9Q==", "dev": true, "requires": { - "chalk": "^2.4.1", - "coa": "^2.0.2", - "css-select": "^2.0.0", - "css-select-base-adapter": "^0.1.1", - "css-tree": "1.0.0-alpha.37", - "csso": "^4.0.2", - "js-yaml": "^3.13.1", - "mkdirp": "~0.5.1", - "object.values": "^1.1.0", - "sax": "~1.2.4", - "stable": "^0.1.8", - "unquote": "~1.1.1", - "util.promisify": "~1.0.0" + "@trysound/sax": "0.1.1", + "chalk": "^4.1.0", + "commander": "^7.1.0", + "css-select": "^3.1.2", + "css-tree": "^1.1.2", + "csso": "^4.2.0", + "stable": "^0.1.8" + }, + "dependencies": { + "ansi-styles": { + "version": "4.3.0", + "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-4.3.0.tgz", + "integrity": "sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==", + "dev": true, + "requires": { + "color-convert": "^2.0.1" + } + }, + "chalk": { + "version": "4.1.1", + "resolved": "https://registry.npmjs.org/chalk/-/chalk-4.1.1.tgz", + "integrity": "sha512-diHzdDKxcU+bAsUboHLPEDQiw0qEe0qd7SYUn3HgcFlWgbDcfLGswOHYeGrHKzG9z6UYf01d9VFMfZxPM1xZSg==", + "dev": true, + "requires": { + "ansi-styles": "^4.1.0", + "supports-color": "^7.1.0" + } + }, + "color-convert": { + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz", + "integrity": "sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==", + "dev": true, + "requires": { + "color-name": "~1.1.4" + } + }, + "color-name": { + "version": "1.1.4", + "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz", + "integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==", + "dev": true + }, + "commander": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz", + "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==", + "dev": true + }, + "has-flag": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz", + "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==", + "dev": true + }, + "supports-color": { + "version": "7.2.0", + "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz", + "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==", + "dev": true, + "requires": { + "has-flag": "^4.0.0" + } + } } }, "symbol-observable": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-1.2.0.tgz", - "integrity": "sha512-e900nM8RRtGhlV36KGEU9k65K3mPb1WV70OdjfxlG2EAuM1noi/E/BaW/uMhL7bPEssK8QV57vN3esixjUvcXQ==", + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/symbol-observable/-/symbol-observable-3.0.0.tgz", + "integrity": "sha512-6tDOXSHiVjuCaasQSWTmHUWn4PuG7qa3+1WT031yTc/swT7+rLiw3GOrFxaH1E3lLP09dH3bVuVDf2gK5rxG3Q==", "dev": true }, "tapable": { - "version": "1.1.3", - "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", - "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.0.tgz", + "integrity": "sha512-FBk4IesMV1rBxX2tfiK8RAmogtWn53puLOQlvO8XuwlgxcYbP4mVPS9Ph4aeamSyyVjOl24aYWAuc8U5kCVwMw==", "dev": true }, "tar": { - "version": "6.0.5", - "resolved": "https://registry.npmjs.org/tar/-/tar-6.0.5.tgz", - "integrity": "sha512-0b4HOimQHj9nXNEAA7zWwMM91Zhhba3pspja6sQbgTpynOJf+bkjBnfybNYzbpLbnwXnbyB4LOREvlyXLkCHSg==", + "version": "6.1.0", + "resolved": "https://registry.npmjs.org/tar/-/tar-6.1.0.tgz", + "integrity": "sha512-DUCttfhsnLCjwoDoFcI+B2iJgYa93vBnDUATYEeRx6sntCTdN01VnqsIuTlALXla/LWooNg0yEGeB+Y8WdFxGA==", "dev": true, "requires": { "chownr": "^2.0.0", @@ -12489,48 +28603,51 @@ } }, "terser": { - "version": "5.3.0", - "resolved": "https://registry.npmjs.org/terser/-/terser-5.3.0.tgz", - "integrity": "sha512-XTT3D3AwxC54KywJijmY2mxZ8nJiEjBHVYzq8l9OaYuRFWeQNBwvipuzzYEP4e+/AVcd1hqG/CqgsdIRyT45Fg==", + "version": "5.5.1", + "resolved": "https://registry.npmjs.org/terser/-/terser-5.5.1.tgz", + "integrity": "sha512-6VGWZNVP2KTUcltUQJ25TtNjx/XgdDsBDKGt8nN0MpydU36LmbPPcMBd2kmtZNNGVVDLg44k7GKeHHj+4zPIBQ==", "dev": true, "requires": { "commander": "^2.20.0", - "source-map": "~0.6.1", - "source-map-support": "~0.5.12" - }, - "dependencies": { - "source-map": { - "version": "0.6.1", - "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", - "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", - "dev": true - } + "source-map": "~0.7.2", + "source-map-support": "~0.5.19" } }, "terser-webpack-plugin": { - "version": "4.1.0", - "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-4.1.0.tgz", - "integrity": "sha512-0ZWDPIP8BtEDZdChbufcXUigOYk6dOX/P/X0hWxqDDcVAQLb8Yy/0FAaemSfax3PAA67+DJR778oz8qVbmy4hA==", + "version": "4.2.3", + "resolved": "https://registry.npmjs.org/terser-webpack-plugin/-/terser-webpack-plugin-4.2.3.tgz", + "integrity": "sha512-jTgXh40RnvOrLQNgIkwEKnQ8rmHjHK4u+6UBEi+W+FPmvb+uo+chJXntKe7/3lW5mNysgSWD60KyesnhW8D6MQ==", "dev": true, "requires": { "cacache": "^15.0.5", "find-cache-dir": "^3.3.1", - "jest-worker": "^26.3.0", + "jest-worker": "^26.5.0", "p-limit": "^3.0.2", - "schema-utils": "^2.6.6", - "serialize-javascript": "^4.0.0", + "schema-utils": "^3.0.0", + "serialize-javascript": "^5.0.1", "source-map": "^0.6.1", - "terser": "^5.0.0", + "terser": "^5.3.4", "webpack-sources": "^1.4.3" }, "dependencies": { "p-limit": { - "version": "3.0.2", - "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.0.2.tgz", - "integrity": "sha512-iwqZSOoWIW+Ew4kAGUlN16J4M7OB3ysMLSZtnhmqx7njIHFPlxWBX8xo3lVTyFVq6mI/lL9qt2IsN1sHwaxJkg==", + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/p-limit/-/p-limit-3.1.0.tgz", + "integrity": "sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==", "dev": true, "requires": { - "p-try": "^2.0.0" + "yocto-queue": "^0.1.0" + } + }, + "schema-utils": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/schema-utils/-/schema-utils-3.0.0.tgz", + "integrity": "sha512-6D82/xSzO094ajanoOSbe4YvXWMfn2A//8Y1+MUqFAJul5Bs+yn36xbK9OtNDcRVSBJ9jjeoXftM6CfztsjOAA==", + "dev": true, + "requires": { + "@types/json-schema": "^7.0.6", + "ajv": "^6.12.5", + "ajv-keywords": "^3.5.2" } }, "source-map": { @@ -12538,9 +28655,25 @@ "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==", "dev": true + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } } } }, + "text-table": { + "version": "0.2.0", + "resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz", + "integrity": "sha1-f17oI66AUgfACvLfSoTsP8+lcLQ=", + "dev": true + }, "through": { "version": "2.3.8", "resolved": "https://registry.npmjs.org/through/-/through-2.3.8.tgz", @@ -12564,9 +28697,9 @@ "dev": true }, "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==", + "version": "2.0.12", + "resolved": "https://registry.npmjs.org/timers-browserify/-/timers-browserify-2.0.12.tgz", + "integrity": "sha512-9phl76Cqm6FhSX9Xe1ZUAMLtm1BLkKj2Qd5ApyWkXzsMRaA7dgr81kf4wJmQf/hAvg8EEyJxDo3du/0KlhPiKQ==", "dev": true, "requires": { "setimmediate": "^1.0.4" @@ -12587,12 +28720,6 @@ "os-tmpdir": "~1.0.2" } }, - "to-array": { - "version": "0.1.4", - "resolved": "https://registry.npmjs.org/to-array/-/to-array-0.1.4.tgz", - "integrity": "sha1-F+bBH3PdTz10zaek/zI46a2b+JA=", - "dev": true - }, "to-arraybuffer": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/to-arraybuffer/-/to-arraybuffer-1.0.1.tgz", @@ -12640,7 +28767,6 @@ "version": "5.0.1", "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz", "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==", - "dev": true, "requires": { "is-number": "^7.0.0" } @@ -12661,15 +28787,6 @@ "punycode": "^2.1.1" } }, - "tr46": { - "version": "2.0.2", - "resolved": "https://registry.npmjs.org/tr46/-/tr46-2.0.2.tgz", - "integrity": "sha512-3n1qG+/5kg+jrbTzwAykB5yRYtQCTqOGKq5U5PE3b0a1/mzo6snDhjGS0zJVJunO0NrT3Dg1MLy5TjWP/UJppg==", - "dev": true, - "requires": { - "punycode": "^2.1.1" - } - }, "tree-kill": { "version": "1.2.2", "resolved": "https://registry.npmjs.org/tree-kill/-/tree-kill-1.2.2.tgz", @@ -12773,12 +28890,6 @@ "integrity": "sha1-WuaBd/GS1EViadEIr6k/+HQ/T2Q=", "dev": true }, - "type": { - "version": "1.2.0", - "resolved": "https://registry.npmjs.org/type/-/type-1.2.0.tgz", - "integrity": "sha512-+5nt5AAniqsCnu2cEQQdpzCAh33kVx8n0VoFidKpB1dVVLAN/F+bgVOqOJqOnEnrhp222clB5p3vUlD+1QAnfg==", - "dev": true - }, "type-fest": { "version": "0.11.0", "resolved": "https://registry.npmjs.org/type-fest/-/type-fest-0.11.0.tgz", @@ -12802,15 +28913,14 @@ "dev": true }, "typescript": { - "version": "4.0.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.0.3.tgz", - "integrity": "sha512-tEu6DGxGgRJPb/mVPIZ48e69xCn2yRmCgYmDugAVwmJ6o+0u1RI18eO7E7WBTLYLaEVVOhwQmcdhQHweux/WPg==", - "dev": true + "version": "4.1.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-4.1.5.tgz", + "integrity": "sha512-6OSu9PTIzmn9TCDiovULTnET6BgXtDYL4Gg4szY+cGsc3JP1dQL8qvE8kShTRx1NIw4Q9IBHlwODjkjWEtMUyA==" }, "ua-parser-js": { - "version": "0.7.21", - "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.21.tgz", - "integrity": "sha512-+O8/qh/Qj8CgC6eYBVBykMrNtp5Gebn4dlGD/kKXVkJNDwyrAwSIqwz8CDf+tsAIWVycKcku6gIXJ0qwx/ZXaQ==", + "version": "0.7.28", + "resolved": "https://registry.npmjs.org/ua-parser-js/-/ua-parser-js-0.7.28.tgz", + "integrity": "sha512-6Gurc1n//gjp9eQNXjD9O3M/sMwVtN5S8Lv9bvOYBfKfDNiIIhqiyi01vMBO45u4zkDE420w/e0se7Vs+sIg+g==", "dev": true }, "unicode-canonical-property-names-ecmascript": { @@ -12851,14 +28961,16 @@ "get-value": "^2.0.6", "is-extendable": "^0.1.1", "set-value": "^2.0.1" + }, + "dependencies": { + "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 + } } }, - "uniq": { - "version": "1.0.1", - "resolved": "https://registry.npmjs.org/uniq/-/uniq-1.0.1.tgz", - "integrity": "sha1-sxxa6CVIRKOoKBVBzisEuGWnNP8=", - "dev": true - }, "uniqs": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/uniqs/-/uniqs-2.0.0.tgz", @@ -12905,8 +29017,7 @@ "universalify": { "version": "0.1.2", "resolved": "https://registry.npmjs.org/universalify/-/universalify-0.1.2.tgz", - "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==", - "dev": true + "integrity": "sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg==" }, "unpipe": { "version": "1.0.0", @@ -12914,12 +29025,6 @@ "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=", "dev": true }, - "unquote": { - "version": "1.1.1", - "resolved": "https://registry.npmjs.org/unquote/-/unquote-1.1.1.tgz", - "integrity": "sha1-j97XMk7G6IoP+LkF58CYzcCG1UQ=", - "dev": true - }, "unset-value": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/unset-value/-/unset-value-1.0.0.tgz", @@ -13000,9 +29105,9 @@ } }, "url-parse": { - "version": "1.4.7", - "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.4.7.tgz", - "integrity": "sha512-d3uaVyzDB9tQoSXFvuSUNFibTd9zxd2bkVrDRvF5TmvWWQwqE4lgYJ5m+x1DbecWkw+LK4RNl2CU1hHuOKPVlg==", + "version": "1.5.1", + "resolved": "https://registry.npmjs.org/url-parse/-/url-parse-1.5.1.tgz", + "integrity": "sha512-HOfCOUJt7iSYzEx/UqgtwKRMC6EU91NFhsCHMv9oM03VJcVo2Qrp8T8kI9D7amFf1cu+/3CEhgb3rF9zL7k85Q==", "dev": true, "requires": { "querystringify": "^2.1.1", @@ -13038,48 +29143,6 @@ "integrity": "sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8=", "dev": true }, - "util-promisify": { - "version": "2.1.0", - "resolved": "https://registry.npmjs.org/util-promisify/-/util-promisify-2.1.0.tgz", - "integrity": "sha1-PCI2R2xNMsX/PEcAKt18E7moKlM=", - "dev": true, - "requires": { - "object.getownpropertydescriptors": "^2.0.3" - } - }, - "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": { - "define-properties": "^1.1.3", - "es-abstract": "^1.17.2", - "has-symbols": "^1.0.1", - "object.getownpropertydescriptors": "^2.1.0" - }, - "dependencies": { - "es-abstract": { - "version": "1.17.7", - "resolved": "https://registry.npmjs.org/es-abstract/-/es-abstract-1.17.7.tgz", - "integrity": "sha512-VBl/gnfcJ7OercKA9MVaegWsBHFjV492syMudcnQZvt/Dw8ezpcOHYZXa/J96O8vx+g4x65YKhxOwDUh63aS5g==", - "dev": true, - "requires": { - "es-to-primitive": "^1.2.1", - "function-bind": "^1.1.1", - "has": "^1.0.3", - "has-symbols": "^1.0.1", - "is-callable": "^1.2.2", - "is-regex": "^1.1.1", - "object-inspect": "^1.8.0", - "object-keys": "^1.1.1", - "object.assign": "^4.1.1", - "string.prototype.trimend": "^1.0.1", - "string.prototype.trimstart": "^1.0.1" - } - } - } - }, "utils-merge": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", @@ -13087,19 +29150,9 @@ "dev": true }, "uuid": { - "version": "8.3.1", - "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.1.tgz", - "integrity": "sha512-FOmRr+FmWEIG8uhZv6C2bTgEVXsHk08kE7mPlrBbEe+c3r9pjceVPgupIfNIhc4yx55H69OXANrUaSuu9eInKg==" - }, - "validate-npm-package-license": { - "version": "3.0.4", - "resolved": "https://registry.npmjs.org/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz", - "integrity": "sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==", - "dev": true, - "requires": { - "spdx-correct": "^3.0.0", - "spdx-expression-parse": "^3.0.0" - } + "version": "8.3.2", + "resolved": "https://registry.npmjs.org/uuid/-/uuid-8.3.2.tgz", + "integrity": "sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==" }, "validate-npm-package-name": { "version": "3.0.0", @@ -13146,21 +29199,21 @@ "dev": true }, "watchpack": { - "version": "1.7.4", - "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.7.4.tgz", - "integrity": "sha512-aWAgTW4MoSJzZPAicljkO1hsi1oKj/RRq/OJQh2PKI2UKL04c2Bs+MBOB+BBABHTXJpf9mCwHN7ANCvYsvY2sg==", + "version": "1.7.5", + "resolved": "https://registry.npmjs.org/watchpack/-/watchpack-1.7.5.tgz", + "integrity": "sha512-9P3MWk6SrKjHsGkLT2KHXdQ/9SNkyoJbabxnKOoJepsvJjJG8uYTR3yTPxPQvNDI3w4Nz1xnE0TLHK4RIVe/MQ==", "dev": true, "requires": { "chokidar": "^3.4.1", "graceful-fs": "^4.1.2", "neo-async": "^2.5.0", - "watchpack-chokidar2": "^2.0.0" + "watchpack-chokidar2": "^2.0.1" } }, "watchpack-chokidar2": { - "version": "2.0.0", - "resolved": "https://registry.npmjs.org/watchpack-chokidar2/-/watchpack-chokidar2-2.0.0.tgz", - "integrity": "sha512-9TyfOyN/zLUbA288wZ8IsMZ+6cbzvsNyEzSBp6e/zkifi6xxbl8SmQ/CxQq32k8NNqrdVEVUVSEf56L4rQ/ZxA==", + "version": "2.0.1", + "resolved": "https://registry.npmjs.org/watchpack-chokidar2/-/watchpack-chokidar2-2.0.1.tgz", + "integrity": "sha512-nCFfBIPKr5Sh61s4LPpy1Wtfi0HE8isJ3d2Yb5/Ppw2P2B/3eVSEBjKfN0fmHJSK14+31KwMKmcrzs2GM4P0Ww==", "dev": true, "optional": true, "requires": { @@ -13318,6 +29371,13 @@ "binary-extensions": "^1.0.0" } }, + "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, + "optional": true + }, "is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", @@ -13415,16 +29475,10 @@ "selenium-webdriver": "^3.0.1" } }, - "webidl-conversions": { - "version": "6.1.0", - "resolved": "https://registry.npmjs.org/webidl-conversions/-/webidl-conversions-6.1.0.tgz", - "integrity": "sha512-qBIvFLGiBpLjfwmYAaHPXsn+ho5xZnGvyGvsarywGNc8VyQJUMHJ8OBKGGrPER0okBeMDaan4mNBlgBROxuI8w==", - "dev": true - }, "webpack": { - "version": "4.44.1", - "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.44.1.tgz", - "integrity": "sha512-4UOGAohv/VGUNQJstzEywwNxqX417FnjZgZJpJQegddzPmTvph37eBIRbRTfdySXzVtJXLJfbMN3mMYhM6GdmQ==", + "version": "4.44.2", + "resolved": "https://registry.npmjs.org/webpack/-/webpack-4.44.2.tgz", + "integrity": "sha512-6KJVGlCxYdISyurpQ0IPTklv+DULv05rs2hseIXer6D7KrUicRDLFb4IUM1S6LUAKypPM/nSiVSuv8jHu1m3/Q==", "dev": true, "requires": { "@webassemblyjs/ast": "1.9.0", @@ -13510,6 +29564,29 @@ "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==", "dev": true }, + "enhanced-resolve": { + "version": "4.5.0", + "resolved": "https://registry.npmjs.org/enhanced-resolve/-/enhanced-resolve-4.5.0.tgz", + "integrity": "sha512-Nv9m36S/vxpsI+Hc4/ZGRs0n9mXqSWGGq49zxb/cJfPAQMbUtttJAlNPS4AQzaBdw/pKskw5bMbekT/Y7W/Wlg==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "memory-fs": "^0.5.0", + "tapable": "^1.0.0" + }, + "dependencies": { + "memory-fs": { + "version": "0.5.0", + "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.5.0.tgz", + "integrity": "sha512-jA0rdU5KoQMC0e6ppoNRtpp6vjFq6+NY7r8hywnC7V+1Xj/MtHwGIbB1QaK/dunyjWteJzmkpd7ooeWg10T7GA==", + "dev": true, + "requires": { + "errno": "^0.1.3", + "readable-stream": "^2.0.1" + } + } + } + }, "fill-range": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-4.0.0.tgz", @@ -13544,6 +29621,21 @@ "pkg-dir": "^3.0.0" } }, + "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" + } + }, + "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 + }, "is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", @@ -13590,6 +29682,16 @@ "json5": "^1.0.1" } }, + "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" + } + }, "lru-cache": { "version": "5.1.1", "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz", @@ -13599,16 +29701,6 @@ "yallist": "^3.0.2" } }, - "memory-fs": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, "micromatch": { "version": "3.1.10", "resolved": "https://registry.npmjs.org/micromatch/-/micromatch-3.1.10.tgz", @@ -13630,6 +29722,30 @@ "to-regex": "^3.0.2" } }, + "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" + } + }, + "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 + }, + "pkg-dir": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/pkg-dir/-/pkg-dir-3.0.0.tgz", + "integrity": "sha512-/E57AYkoeQ25qkxMj5PBOVgF8Kiu/h7cYS30Z5+R7WaiCCBfLq58ZI/dSeaEKb9WVJV5n/03QwrN3IeWIFllvw==", + "dev": true, + "requires": { + "find-up": "^3.0.0" + } + }, "rimraf": { "version": "2.7.1", "resolved": "https://registry.npmjs.org/rimraf/-/rimraf-2.7.1.tgz", @@ -13650,6 +29766,15 @@ "ajv-keywords": "^3.1.0" } }, + "serialize-javascript": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz", + "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==", + "dev": true, + "requires": { + "randombytes": "^2.1.0" + } + }, "source-map": { "version": "0.6.1", "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz", @@ -13657,14 +29782,20 @@ "dev": true }, "ssri": { - "version": "6.0.1", - "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.1.tgz", - "integrity": "sha512-3Wge10hNcT1Kur4PDFwEieXSCMCJs/7WvSACcrMYrNp+b8kDL1/0wJch5Ni2WrtwEa2IO8OsVfeKIciKCDx/QA==", + "version": "6.0.2", + "resolved": "https://registry.npmjs.org/ssri/-/ssri-6.0.2.tgz", + "integrity": "sha512-cepbSq/neFK7xB6A50KHN0xHDotYzq58wWCa5LeWqnPrHG8GzfEjO/4O8kpmcGW+oaxkvhEJCWgbgNk4/ZV93Q==", "dev": true, "requires": { "figgy-pudding": "^3.5.1" } }, + "tapable": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/tapable/-/tapable-1.1.3.tgz", + "integrity": "sha512-4WK/bYZmj8xLr+HUCODHGF1ZFzsYffasLUgEiMBY4fgtltdO6B4WJtlSbPaDTLpYTcGVwM2qLnFTICEcNxs3kA==", + "dev": true + }, "terser": { "version": "4.8.0", "resolved": "https://registry.npmjs.org/terser/-/terser-4.8.0.tgz", @@ -13703,6 +29834,16 @@ "repeat-string": "^1.6.1" } }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + }, "yallist": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.1.1.tgz", @@ -13724,28 +29865,18 @@ "webpack-log": "^2.0.0" }, "dependencies": { - "memory-fs": { - "version": "0.4.1", - "resolved": "https://registry.npmjs.org/memory-fs/-/memory-fs-0.4.1.tgz", - "integrity": "sha1-OpoguEYlI+RHz7x+i7gO1me/xVI=", - "dev": true, - "requires": { - "errno": "^0.1.3", - "readable-stream": "^2.0.1" - } - }, "mime": { - "version": "2.4.6", - "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.6.tgz", - "integrity": "sha512-RZKhC3EmpBchfTGBVb8fb+RL2cWyw/32lshnsETttkBAyAUXSGHxbEJWWRXc751DrIxG1q04b8QwMbAwkRPpUA==", + "version": "2.5.2", + "resolved": "https://registry.npmjs.org/mime/-/mime-2.5.2.tgz", + "integrity": "sha512-tqkh47FzKeCPD2PUiPB6pkbMzsCasjxAfC62/Wap5qrUWcb+sFasXUC5I3gYM5iBM8v/Qpn4UK0x+j0iHyFPDg==", "dev": true } } }, "webpack-dev-server": { - "version": "3.11.0", - "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.11.0.tgz", - "integrity": "sha512-PUxZ+oSTxogFQgkTtFndEtJIPNmml7ExwufBZ9L2/Xyyd5PnOL5UreWe5ZT7IU25DSdykL9p1MLQzmLh2ljSeg==", + "version": "3.11.2", + "resolved": "https://registry.npmjs.org/webpack-dev-server/-/webpack-dev-server-3.11.2.tgz", + "integrity": "sha512-A80BkuHRQfCiNtGBS1EMf2ChTUs0x+B3wGDFmOeT4rmJOHhHTCH2naNxIHhmkr0/UillP4U3yeIyv1pNp+QDLQ==", "dev": true, "requires": { "ansi-html": "0.0.7", @@ -13768,11 +29899,11 @@ "p-retry": "^3.0.1", "portfinder": "^1.0.26", "schema-utils": "^1.0.0", - "selfsigned": "^1.10.7", + "selfsigned": "^1.10.8", "semver": "^6.3.0", "serve-index": "^1.9.1", - "sockjs": "0.3.20", - "sockjs-client": "1.4.0", + "sockjs": "^0.3.21", + "sockjs-client": "^1.5.0", "spdy": "^4.0.2", "strip-ansi": "^3.0.1", "supports-color": "^6.1.0", @@ -13914,12 +30045,6 @@ } } }, - "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 - }, "is-binary-path": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-1.0.1.tgz", @@ -13929,6 +30054,12 @@ "binary-extensions": "^1.0.0" } }, + "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 + }, "is-number": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/is-number/-/is-number-3.0.0.tgz", @@ -14029,6 +30160,12 @@ "uuid": "^3.3.2" }, "dependencies": { + "ansi-colors": { + "version": "3.2.4", + "resolved": "https://registry.npmjs.org/ansi-colors/-/ansi-colors-3.2.4.tgz", + "integrity": "sha512-hHUXGagefjN2iRrID63xckIvotOXOojhQKWIPUZ4mNUZ9nLZW+7FMNoE1lOkEhNWYsx/7ysGIuJYCiMAA9FnrA==", + "dev": true + }, "uuid": { "version": "3.4.0", "resolved": "https://registry.npmjs.org/uuid/-/uuid-3.4.0.tgz", @@ -14038,22 +30175,23 @@ } }, "webpack-merge": { - "version": "4.2.2", - "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-4.2.2.tgz", - "integrity": "sha512-TUE1UGoTX2Cd42j3krGYqObZbOD+xF7u28WB7tfUordytSjbWTIjK/8V0amkBfTYN4/pB/GIDlJZZ657BGG19g==", + "version": "5.7.3", + "resolved": "https://registry.npmjs.org/webpack-merge/-/webpack-merge-5.7.3.tgz", + "integrity": "sha512-6/JUQv0ELQ1igjGDzHkXbVDRxkfA57Zw7PfiupdLFJYrgFqY5ZP8xxbpp2lU3EPwYx89ht5Z/aDkD40hFCm5AA==", "dev": true, "requires": { - "lodash": "^4.17.15" + "clone-deep": "^4.0.1", + "wildcard": "^2.0.0" } }, "webpack-sources": { - "version": "1.4.3", - "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", - "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "version": "2.2.0", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-2.2.0.tgz", + "integrity": "sha512-bQsA24JLwcnWGArOKUxYKhX3Mz/nK1Xf6hxullKERyktjNMC4x8koOeaDNTA2fEJ09BdWLbM/iTW0ithREUP0w==", "dev": true, "requires": { - "source-list-map": "^2.0.0", - "source-map": "~0.6.1" + "source-list-map": "^2.0.1", + "source-map": "^0.6.1" }, "dependencies": { "source-map": { @@ -14065,20 +30203,40 @@ } }, "webpack-subresource-integrity": { - "version": "1.4.1", - "resolved": "https://registry.npmjs.org/webpack-subresource-integrity/-/webpack-subresource-integrity-1.4.1.tgz", - "integrity": "sha512-XMLFInbGbB1HV7K4vHWANzc1CN0t/c4bBvnlvGxGwV45yE/S/feAXIm8dJsCkzqWtSKnmaEgTp/meyeThxG4Iw==", + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/webpack-subresource-integrity/-/webpack-subresource-integrity-1.5.2.tgz", + "integrity": "sha512-GBWYBoyalbo5YClwWop9qe6Zclp8CIXYGIz12OPclJhIrSplDxs1Ls1JDMH8xBPPrg1T6ISaTW9Y6zOrwEiAzw==", "dev": true, "requires": { "webpack-sources": "^1.3.0" + }, + "dependencies": { + "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 + }, + "webpack-sources": { + "version": "1.4.3", + "resolved": "https://registry.npmjs.org/webpack-sources/-/webpack-sources-1.4.3.tgz", + "integrity": "sha512-lgTS3Xhv1lCOKo7SA5TjKXMjpSM4sBjNV5+q2bqesbSPs5FjGmU6jjtBSkX9b4qW87vDIsCIlUPOEhbZrMdjeQ==", + "dev": true, + "requires": { + "source-list-map": "^2.0.0", + "source-map": "~0.6.1" + } + } } }, "websocket-driver": { - "version": "0.6.5", - "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.6.5.tgz", - "integrity": "sha1-XLJVbOuF9Dc8bYI4qmkchFThOjY=", + "version": "0.7.4", + "resolved": "https://registry.npmjs.org/websocket-driver/-/websocket-driver-0.7.4.tgz", + "integrity": "sha512-b17KeDIQVjvb0ssuSDF2cYXSg2iztliJ4B9WdsuB6J952qCPKmnVq4DyW5motImXHDC1cBT/1UezrJVsKw5zjg==", "dev": true, "requires": { + "http-parser-js": ">=0.5.1", + "safe-buffer": ">=5.1.0", "websocket-extensions": ">=0.1.1" } }, @@ -14094,23 +30252,6 @@ "integrity": "sha512-M4yMwr6mAnQz76TbJm914+gPpB/nCwvZbJU28cUD6dR004SAxDLOOSUaB1JDRqLtaOV/vi0IC5lEAGFgrjGv/g==", "dev": true }, - "whatwg-url": { - "version": "8.4.0", - "resolved": "https://registry.npmjs.org/whatwg-url/-/whatwg-url-8.4.0.tgz", - "integrity": "sha512-vwTUFf6V4zhcPkWp/4CQPr1TW9Ml6SF4lVyaIMBdJw5i6qUUJ1QWM4Z6YYVkfka0OUIzVo/0aNtGVGk256IKWw==", - "dev": true, - "requires": { - "lodash.sortby": "^4.7.0", - "tr46": "^2.0.2", - "webidl-conversions": "^6.1.0" - } - }, - "when": { - "version": "3.6.4", - "resolved": "https://registry.npmjs.org/when/-/when-3.6.4.tgz", - "integrity": "sha1-RztRfsFZ4rhQBUl6E5g/CVQS404=", - "dev": true - }, "which": { "version": "1.3.1", "resolved": "https://registry.npmjs.org/which/-/which-1.3.1.tgz", @@ -14123,7 +30264,50 @@ "which-module": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/which-module/-/which-module-2.0.0.tgz", - "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=" + "integrity": "sha1-2e8H3Od7mQK4o6j6SzHD4/fm6Ho=", + "dev": true + }, + "wide-align": { + "version": "1.1.3", + "resolved": "https://registry.npmjs.org/wide-align/-/wide-align-1.1.3.tgz", + "integrity": "sha512-QGkOQc8XL6Bt5PwnsExKBPuMKBxnGxWWW3fU55Xt4feHozMUhdUMaBCk290qpm/wG5u/RSKzwdAC4i51YigihA==", + "dev": true, + "requires": { + "string-width": "^1.0.2 || 2" + }, + "dependencies": { + "ansi-regex": { + "version": "3.0.0", + "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-3.0.0.tgz", + "integrity": "sha1-7QMXwyIGT3lGbAKWa922Bas32Zg=", + "dev": true + }, + "string-width": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/string-width/-/string-width-2.1.1.tgz", + "integrity": "sha512-nOqH59deCq9SRHlxq1Aw85Jnt4w6KvLKqWVik6oA9ZklXLNIOlqg4F2yrT1MVaTjAqvVwdfeZ7w7aCvJD7ugkw==", + "dev": true, + "requires": { + "is-fullwidth-code-point": "^2.0.0", + "strip-ansi": "^4.0.0" + } + }, + "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": { + "ansi-regex": "^3.0.0" + } + } + } + }, + "wildcard": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/wildcard/-/wildcard-2.0.0.tgz", + "integrity": "sha512-JcKqAHLPxcdb9KM49dufGXn2x3ssnfjbcaQdLlfZsL9rH9wgDQjUtDxbo8NE0F6SFvydeu1VhZe7hZuHsB2/pw==", + "dev": true }, "worker-farm": { "version": "1.7.0", @@ -14199,9 +30383,9 @@ "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, "ws": { - "version": "6.2.1", - "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz", - "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==", + "version": "6.2.2", + "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.2.tgz", + "integrity": "sha512-zmhltoSR8u1cnDsD43TX59mzoMZsLKqUweyYBAIvTngR3shc0W6aOZylZmq/7hqyVxPdi+5Ud2QInblgyE72fw==", "dev": true, "requires": { "async-limiter": "~1.0.0" @@ -14223,12 +30407,6 @@ "integrity": "sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA==", "dev": true }, - "xmlhttprequest-ssl": { - "version": "1.5.5", - "resolved": "https://registry.npmjs.org/xmlhttprequest-ssl/-/xmlhttprequest-ssl-1.5.5.tgz", - "integrity": "sha1-wodrBhaKrcQOV9l+gRkayPQ5iz4=", - "dev": true - }, "xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", @@ -14236,9 +30414,10 @@ "dev": true }, "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==" + "version": "4.0.3", + "resolved": "https://registry.npmjs.org/y18n/-/y18n-4.0.3.tgz", + "integrity": "sha512-JKhqTOwSrqNA1NY5lSztJ1GrBiUodLMmIZuLiDaMRJ+itFd+ABVE8XBjOvIWL+rSqNDC74LCSFmlb/U4UZ4hJQ==", + "dev": true }, "yallist": { "version": "4.0.0", @@ -14246,6 +30425,12 @@ "integrity": "sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A==", "dev": true }, + "yaml": { + "version": "1.10.2", + "resolved": "https://registry.npmjs.org/yaml/-/yaml-1.10.2.tgz", + "integrity": "sha512-r3vXyErRCYJ7wg28yvBY5VSoAF8ZvlcW9/BwUzEtUsjvX/DKs24dIkuwjtuprwJJHsbyUbLApepYTR1BN4uHrg==", + "dev": true + }, "yargs": { "version": "13.3.2", "resolved": "https://registry.npmjs.org/yargs/-/yargs-13.3.2.tgz", @@ -14262,6 +30447,42 @@ "which-module": "^2.0.0", "y18n": "^4.0.0", "yargs-parser": "^13.1.2" + }, + "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-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" + } + }, + "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 + } } }, "yargs-parser": { @@ -14272,32 +30493,27 @@ "requires": { "camelcase": "^5.0.0", "decamelize": "^1.2.0" - }, - "dependencies": { - "camelcase": { - "version": "5.3.1", - "resolved": "https://registry.npmjs.org/camelcase/-/camelcase-5.3.1.tgz", - "integrity": "sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==", - "dev": true - } } }, - "yeast": { - "version": "0.1.2", - "resolved": "https://registry.npmjs.org/yeast/-/yeast-0.1.2.tgz", - "integrity": "sha1-AI4G2AlDIMNy28L47XagymyKxBk=", - "dev": true - }, "yn": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz", "integrity": "sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==", "dev": true }, + "yocto-queue": { + "version": "0.1.0", + "resolved": "https://registry.npmjs.org/yocto-queue/-/yocto-queue-0.1.0.tgz", + "integrity": "sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==", + "dev": true + }, "zone.js": { - "version": "0.10.3", - "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.10.3.tgz", - "integrity": "sha512-LXVLVEq0NNOqK/fLJo3d0kfzd4sxwn2/h67/02pjCjfKDxgx1i9QqpvtHD8CrBnSSwMw5+dy11O7FRX5mkO7Cg==" + "version": "0.11.4", + "resolved": "https://registry.npmjs.org/zone.js/-/zone.js-0.11.4.tgz", + "integrity": "sha512-DDh2Ab+A/B+9mJyajPjHFPWfYU1H+pdun4wnnk0OcQTNjem1XQSZ2CDW+rfZEUDjv5M19SBqAkjZi0x5wuB5Qw==", + "requires": { + "tslib": "^2.0.0" + } } } } diff --git a/src-ui/package.json b/src-ui/package.json index 6293f2672..89880c418 100644 --- a/src-ui/package.json +++ b/src-ui/package.json @@ -11,38 +11,41 @@ }, "private": true, "dependencies": { - "@angular/animations": "~10.1.5", - "@angular/common": "~10.1.5", - "@angular/compiler": "~10.1.5", - "@angular/core": "~10.1.5", - "@angular/forms": "~10.1.5", - "@angular/localize": "~10.1.5", - "@angular/platform-browser": "~10.1.5", - "@angular/platform-browser-dynamic": "~10.1.5", - "@angular/router": "~10.1.5", - "@ng-bootstrap/ng-bootstrap": "^8.0.0", + "@angular/animations": "~11.2.14", + "@angular/common": "~11.2.14", + "@angular/compiler": "~11.2.14", + "@angular/core": "~11.2.14", + "@angular/forms": "~11.2.14", + "@angular/localize": "~11.2.14", + "@angular/platform-browser": "~11.2.14", + "@angular/platform-browser-dynamic": "~11.2.14", + "@angular/router": "~11.2.14", + "@ng-bootstrap/ng-bootstrap": "^9.1.2", + "@ng-select/ng-select": "^7.0.0", "bootstrap": "^4.5.0", - "ng-bootstrap": "^1.6.3", + "file-saver": "^2.0.5", "ng2-pdf-viewer": "^6.3.2", + "ngx-bootstrap": "^6.2.0", + "ngx-color": "^6.2.0", "ngx-cookie-service": "^10.1.1", - "ngx-file-drop": "^10.0.0", + "ngx-file-drop": "^11.1.0", "ngx-infinite-scroll": "^9.1.0", "rxjs": "~6.6.0", "tslib": "^2.0.0", "uuid": "^8.3.1", - "zone.js": "~0.10.2" + "zone.js": "~0.11.4" }, "devDependencies": { - "@angular-devkit/build-angular": "^0.1002.0", - "@angular/cli": "~10.1.5", - "@angular/compiler-cli": "~10.1.5", - "@types/jasmine": "~3.5.0", + "@angular-devkit/build-angular": "~0.1102.13", + "@angular/cli": "~11.2.14", + "@angular/compiler-cli": "~11.2.14", + "@types/jasmine": "~3.6.0", "@types/jasminewd2": "~2.0.3", "@types/node": "^12.11.1", "codelyzer": "^6.0.0", "jasmine-core": "~3.6.0", "jasmine-spec-reporter": "~5.0.0", - "karma": "~5.0.0", + "karma": "~6.3.3", "karma-chrome-launcher": "~3.1.0", "karma-coverage-istanbul-reporter": "~3.0.2", "karma-jasmine": "~4.0.0", @@ -50,6 +53,6 @@ "protractor": "~7.0.0", "ts-node": "~8.3.0", "tslint": "~6.1.0", - "typescript": "~4.0.2" + "typescript": "~4.1.5" } } diff --git a/src-ui/src/app/app-routing.module.ts b/src-ui/src/app/app-routing.module.ts index 27f0629b4..031a4bb0b 100644 --- a/src-ui/src/app/app-routing.module.ts +++ b/src-ui/src/app/app-routing.module.ts @@ -10,7 +10,7 @@ import { LogsComponent } from './components/manage/logs/logs.component'; import { SettingsComponent } from './components/manage/settings/settings.component'; import { TagListComponent } from './components/manage/tag-list/tag-list.component'; import { NotFoundComponent } from './components/not-found/not-found.component'; -import { SearchComponent } from './components/search/search.component'; +import {DocumentAsnComponent} from "./components/document-asn/document-asn.component"; const routes: Routes = [ {path: '', redirectTo: 'dashboard', pathMatch: 'full'}, @@ -18,22 +18,22 @@ const routes: Routes = [ {path: 'dashboard', component: DashboardComponent }, {path: 'documents', component: DocumentListComponent }, {path: 'view/:id', component: DocumentListComponent }, - {path: 'search', component: SearchComponent }, {path: 'documents/:id', component: DocumentDetailComponent }, - + {path: 'asn/:id', component: DocumentAsnComponent }, + {path: 'tags', component: TagListComponent }, {path: 'documenttypes', component: DocumentTypeListComponent }, {path: 'correspondents', component: CorrespondentListComponent }, {path: 'logs', component: LogsComponent }, {path: 'settings', component: SettingsComponent }, - ]}, + ]}, {path: '404', component: NotFoundComponent}, {path: '**', redirectTo: '/404', pathMatch: 'full'} ]; @NgModule({ - imports: [RouterModule.forRoot(routes)], + imports: [RouterModule.forRoot(routes, { relativeLinkResolution: 'legacy' })], exports: [RouterModule] }) export class AppRoutingModule { } diff --git a/src-ui/src/app/app.component.html b/src-ui/src/app/app.component.html index bcf2bba31..d9b7dd09b 100644 --- a/src-ui/src/app/app.component.html +++ b/src-ui/src/app/app.component.html @@ -1,3 +1,3 @@ - \ No newline at end of file + diff --git a/src-ui/src/app/app.component.ts b/src-ui/src/app/app.component.ts index 84c173a18..c5ec9604d 100644 --- a/src-ui/src/app/app.component.ts +++ b/src-ui/src/app/app.component.ts @@ -1,14 +1,70 @@ -import { Component } from '@angular/core'; +import { SettingsService, SETTINGS_KEYS } from './services/settings.service'; +import { Component, OnDestroy, OnInit } from '@angular/core'; +import { Router } from '@angular/router'; +import { Subscription } from 'rxjs'; +import { ConsumerStatusService } from './services/consumer-status.service'; +import { ToastService } from './services/toast.service'; @Component({ selector: 'app-root', templateUrl: './app.component.html', styleUrls: ['./app.component.scss'] }) -export class AppComponent { - - constructor () { +export class AppComponent implements OnInit, OnDestroy { + + newDocumentSubscription: Subscription; + successSubscription: Subscription; + failedSubscription: Subscription; + + constructor (private settings: SettingsService, private consumerStatusService: ConsumerStatusService, private toastService: ToastService, private router: Router) { + let anyWindow = (window as any) + anyWindow.pdfWorkerSrc = 'assets/js/pdf.worker.min.js'; + this.settings.updateDarkModeSettings() } + ngOnDestroy(): void { + this.consumerStatusService.disconnect() + if (this.successSubscription) { + this.successSubscription.unsubscribe() + } + if (this.failedSubscription) { + this.failedSubscription.unsubscribe() + } + if (this.newDocumentSubscription) { + this.newDocumentSubscription.unsubscribe() + } + } + + private showNotification(key) { + if (this.router.url == '/dashboard' && this.settings.get(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD)) { + return false + } + return this.settings.get(key) + } + + ngOnInit(): void { + this.consumerStatusService.connect() + + + this.successSubscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => { + if (this.showNotification(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUCCESS)) { + this.toastService.show({title: $localize`Document added`, delay: 10000, content: $localize`Document ${status.filename} was added to paperless.`, actionName: $localize`Open document`, action: () => { + this.router.navigate(['documents', status.documentId]) + }}) + } + }) + + this.failedSubscription = this.consumerStatusService.onDocumentConsumptionFailed().subscribe(status => { + if (this.showNotification(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_FAILED)) { + this.toastService.showError($localize`Could not add ${status.filename}\: ${status.message}`) + } + }) + + this.newDocumentSubscription = this.consumerStatusService.onDocumentDetected().subscribe(status => { + if (this.showNotification(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_NEW_DOCUMENT)) { + this.toastService.show({title: $localize`New document detected`, delay: 5000, content: $localize`Document ${status.filename} is being processed by paperless.`}) + } + }) + } } diff --git a/src-ui/src/app/app.module.ts b/src-ui/src/app/app.module.ts index 0ee36b478..43e5865ff 100644 --- a/src-ui/src/app/app.module.ts +++ b/src-ui/src/app/app.module.ts @@ -3,7 +3,7 @@ import { NgModule } from '@angular/core'; import { AppRoutingModule } from './app-routing.module'; import { AppComponent } from './app.component'; -import { NgbModule } from '@ng-bootstrap/ng-bootstrap'; +import { NgbDateAdapter, NgbDateParserFormatter, NgbModule } from '@ng-bootstrap/ng-bootstrap'; import { HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http'; import { DocumentListComponent } from './components/document-list/document-list.component'; import { DocumentDetailComponent } from './components/document-detail/document-detail.component'; @@ -13,29 +13,30 @@ import { DocumentTypeListComponent } from './components/manage/document-type-lis import { LogsComponent } from './components/manage/logs/logs.component'; import { SettingsComponent } from './components/manage/settings/settings.component'; import { FormsModule, ReactiveFormsModule } from '@angular/forms'; -import { DatePipe } from '@angular/common'; +import { DatePipe, registerLocaleData } from '@angular/common'; import { NotFoundComponent } from './components/not-found/not-found.component'; import { CorrespondentListComponent } from './components/manage/correspondent-list/correspondent-list.component'; -import { DeleteDialogComponent } from './components/common/delete-dialog/delete-dialog.component'; +import { ConfirmDialogComponent } from './components/common/confirm-dialog/confirm-dialog.component'; import { CorrespondentEditDialogComponent } from './components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component'; import { TagEditDialogComponent } from './components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component'; import { DocumentTypeEditDialogComponent } from './components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component'; import { TagComponent } from './components/common/tag/tag.component'; -import { SearchComponent } from './components/search/search.component'; -import { ResultHighlightComponent } from './components/search/result-highlight/result-highlight.component'; import { PageHeaderComponent } from './components/common/page-header/page-header.component'; import { AppFrameComponent } from './components/app-frame/app-frame.component'; import { ToastsComponent } from './components/common/toasts/toasts.component'; -import { FilterEditorComponent } from './components/filter-editor/filter-editor.component'; +import { FilterEditorComponent } from './components/document-list/filter-editor/filter-editor.component'; +import { FilterableDropdownComponent } from './components/common/filterable-dropdown/filterable-dropdown.component'; +import { ToggleableDropdownButtonComponent } from './components/common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; +import { DateDropdownComponent } from './components/common/date-dropdown/date-dropdown.component'; import { DocumentCardLargeComponent } from './components/document-list/document-card-large/document-card-large.component'; import { DocumentCardSmallComponent } from './components/document-list/document-card-small/document-card-small.component'; +import { BulkEditorComponent } from './components/document-list/bulk-editor/bulk-editor.component'; import { NgxFileDropModule } from 'ngx-file-drop'; import { TextComponent } from './components/common/input/text/text.component'; import { SelectComponent } from './components/common/input/select/select.component'; import { CheckComponent } from './components/common/input/check/check.component'; import { SaveViewConfigDialogComponent } from './components/document-list/save-view-config-dialog/save-view-config-dialog.component'; import { InfiniteScrollModule } from 'ngx-infinite-scroll'; -import { DateTimeComponent } from './components/common/input/date-time/date-time.component'; import { TagsComponent } from './components/common/input/tags/tags.component'; import { SortableDirective } from './directives/sortable.directive'; import { CookieService } from 'ngx-cookie-service'; @@ -48,7 +49,49 @@ import { PdfViewerModule } from 'ng2-pdf-viewer'; import { WelcomeWidgetComponent } from './components/dashboard/widgets/welcome-widget/welcome-widget.component'; import { YesNoPipe } from './pipes/yes-no.pipe'; import { FileSizePipe } from './pipes/file-size.pipe'; +import { FilterPipe } from './pipes/filter.pipe'; import { DocumentTitlePipe } from './pipes/document-title.pipe'; +import { MetadataCollapseComponent } from './components/document-detail/metadata-collapse/metadata-collapse.component'; +import { SelectDialogComponent } from './components/common/select-dialog/select-dialog.component'; +import { NgSelectModule } from '@ng-select/ng-select'; +import { NumberComponent } from './components/common/input/number/number.component'; +import { SafePipe } from './pipes/safe.pipe'; +import { CustomDatePipe } from './pipes/custom-date.pipe'; +import { DateComponent } from './components/common/input/date/date.component'; +import { ISODateTimeAdapter } from './utils/ngb-iso-date-time-adapter'; +import { LocalizedDateParserFormatter } from './utils/ngb-date-parser-formatter'; +import { ApiVersionInterceptor } from './interceptors/api-version.interceptor'; +import { ColorSliderModule } from 'ngx-color/slider'; +import { ColorComponent } from './components/common/input/color/color.component'; +import { DocumentAsnComponent } from './components/document-asn/document-asn.component'; + +import localeFr from '@angular/common/locales/fr'; +import localeNl from '@angular/common/locales/nl'; +import localeDe from '@angular/common/locales/de'; +import localePt from '@angular/common/locales/pt'; +import localeIt from '@angular/common/locales/it'; +import localeEnGb from '@angular/common/locales/en-GB'; +import localeRo from '@angular/common/locales/ro'; +import localeRu from '@angular/common/locales/ru'; +import localeEs from '@angular/common/locales/es'; +import localePl from '@angular/common/locales/pl'; +import localeSv from '@angular/common/locales/sv'; +import localeLb from '@angular/common/locales/lb'; + + +registerLocaleData(localeFr) +registerLocaleData(localeNl) +registerLocaleData(localeDe) +registerLocaleData(localePt, "pt-BR") +registerLocaleData(localePt, "pt-PT") +registerLocaleData(localeIt) +registerLocaleData(localeEnGb) +registerLocaleData(localeRo) +registerLocaleData(localeRu) +registerLocaleData(localeEs) +registerLocaleData(localePl) +registerLocaleData(localeSv) +registerLocaleData(localeLb) @NgModule({ declarations: [ @@ -63,23 +106,24 @@ import { DocumentTitlePipe } from './pipes/document-title.pipe'; SettingsComponent, NotFoundComponent, CorrespondentEditDialogComponent, - DeleteDialogComponent, + ConfirmDialogComponent, TagEditDialogComponent, DocumentTypeEditDialogComponent, TagComponent, - SearchComponent, - ResultHighlightComponent, PageHeaderComponent, AppFrameComponent, ToastsComponent, FilterEditorComponent, + FilterableDropdownComponent, + ToggleableDropdownButtonComponent, + DateDropdownComponent, DocumentCardLargeComponent, DocumentCardSmallComponent, + BulkEditorComponent, TextComponent, SelectComponent, CheckComponent, SaveViewConfigDialogComponent, - DateTimeComponent, TagsComponent, SortableDirective, SavedViewWidgetComponent, @@ -89,7 +133,16 @@ import { DocumentTitlePipe } from './pipes/document-title.pipe'; WelcomeWidgetComponent, YesNoPipe, FileSizePipe, - DocumentTitlePipe + FilterPipe, + DocumentTitlePipe, + MetadataCollapseComponent, + SelectDialogComponent, + NumberComponent, + SafePipe, + CustomDatePipe, + DateComponent, + ColorComponent, + DocumentAsnComponent ], imports: [ BrowserModule, @@ -100,7 +153,9 @@ import { DocumentTitlePipe } from './pipes/document-title.pipe'; ReactiveFormsModule, NgxFileDropModule, InfiniteScrollModule, - PdfViewerModule + PdfViewerModule, + NgSelectModule, + ColorSliderModule ], providers: [ DatePipe, @@ -108,7 +163,15 @@ import { DocumentTitlePipe } from './pipes/document-title.pipe'; provide: HTTP_INTERCEPTORS, useClass: CsrfInterceptor, multi: true - } + },{ + provide: HTTP_INTERCEPTORS, + useClass: ApiVersionInterceptor, + multi: true + }, + FilterPipe, + DocumentTitlePipe, + {provide: NgbDateAdapter, useClass: ISODateTimeAdapter}, + {provide: NgbDateParserFormatter, useClass: LocalizedDateParserFormatter} ], bootstrap: [AppComponent] }) diff --git a/src-ui/src/app/components/app-frame/app-frame.component.html b/src-ui/src/app/components/app-frame/app-frame.component.html index 3f326afdd..25d4a9dc7 100644 --- a/src-ui/src/app/components/app-frame/app-frame.component.html +++ b/src-ui/src/app/components/app-frame/app-frame.component.html @@ -1,159 +1,191 @@
diff --git a/src-ui/src/app/components/common/input/check/check.component.ts b/src-ui/src/app/components/common/input/check/check.component.ts index de0b9a0d1..d452aad87 100644 --- a/src-ui/src/app/components/common/input/check/check.component.ts +++ b/src-ui/src/app/components/common/input/check/check.component.ts @@ -15,7 +15,7 @@ import { AbstractInputComponent } from '../abstract-input'; }) export class CheckComponent extends AbstractInputComponent { - constructor() { + constructor() { super() } diff --git a/src-ui/src/app/components/common/input/color/color.component.html b/src-ui/src/app/components/common/input/color/color.component.html new file mode 100644 index 000000000..94c34dc00 --- /dev/null +++ b/src-ui/src/app/components/common/input/color/color.component.html @@ -0,0 +1,33 @@ +
+ + +
+
+     +
+ + +
+ +
+ +
+ + + +
+ +
+ + +
+ {{hint}} +
+ {{error}} +
+
diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.scss b/src-ui/src/app/components/common/input/color/color.component.scss similarity index 100% rename from src-ui/src/app/components/filter-editor/filter-editor.component.scss rename to src-ui/src/app/components/common/input/color/color.component.scss diff --git a/src-ui/src/app/components/search/search.component.spec.ts b/src-ui/src/app/components/common/input/color/color.component.spec.ts similarity index 56% rename from src-ui/src/app/components/search/search.component.spec.ts rename to src-ui/src/app/components/common/input/color/color.component.spec.ts index 918ce7071..7c5d0d270 100644 --- a/src-ui/src/app/components/search/search.component.spec.ts +++ b/src-ui/src/app/components/common/input/color/color.component.spec.ts @@ -1,20 +1,20 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { SearchComponent } from './search.component'; +import { ColorComponent } from './color.component'; -describe('SearchComponent', () => { - let component: SearchComponent; - let fixture: ComponentFixture; +describe('ColorComponent', () => { + let component: ColorComponent; + let fixture: ComponentFixture; beforeEach(async () => { await TestBed.configureTestingModule({ - declarations: [ SearchComponent ] + declarations: [ ColorComponent ] }) .compileComponents(); }); beforeEach(() => { - fixture = TestBed.createComponent(SearchComponent); + fixture = TestBed.createComponent(ColorComponent); component = fixture.componentInstance; fixture.detectChanges(); }); diff --git a/src-ui/src/app/components/common/input/color/color.component.ts b/src-ui/src/app/components/common/input/color/color.component.ts new file mode 100644 index 000000000..a7f3452f2 --- /dev/null +++ b/src-ui/src/app/components/common/input/color/color.component.ts @@ -0,0 +1,30 @@ +import { Component, forwardRef } from '@angular/core'; +import { NG_VALUE_ACCESSOR } from '@angular/forms'; +import { randomColor } from 'src/app/utils/color'; +import { AbstractInputComponent } from '../abstract-input'; + +@Component({ + providers: [{ + provide: NG_VALUE_ACCESSOR, + useExisting: forwardRef(() => ColorComponent), + multi: true + }], + selector: 'app-input-color', + templateUrl: './color.component.html', + styleUrls: ['./color.component.scss'] +}) +export class ColorComponent extends AbstractInputComponent { + + constructor() { + super() + } + + randomize() { + this.colorChanged(randomColor()) + } + + colorChanged(value) { + this.value = value + this.onChange(value) + } +} diff --git a/src-ui/src/app/components/common/input/date-time/date-time.component.html b/src-ui/src/app/components/common/input/date-time/date-time.component.html deleted file mode 100644 index 7c002db1b..000000000 --- a/src-ui/src/app/components/common/input/date-time/date-time.component.html +++ /dev/null @@ -1,13 +0,0 @@ -
-
- - -
-
- - -
-
- - - \ No newline at end of file diff --git a/src-ui/src/app/components/common/input/date-time/date-time.component.ts b/src-ui/src/app/components/common/input/date-time/date-time.component.ts deleted file mode 100644 index 6a04c5b27..000000000 --- a/src-ui/src/app/components/common/input/date-time/date-time.component.ts +++ /dev/null @@ -1,62 +0,0 @@ -import { formatDate } from '@angular/common'; -import { Component, forwardRef, Input, OnInit } from '@angular/core'; -import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; -import { AbstractInputComponent } from '../abstract-input'; - -@Component({ - providers: [{ - provide: NG_VALUE_ACCESSOR, - useExisting: forwardRef(() => DateTimeComponent), - multi: true - }], - selector: 'app-input-date-time', - templateUrl: './date-time.component.html', - styleUrls: ['./date-time.component.scss'] -}) -export class DateTimeComponent implements OnInit,ControlValueAccessor { - - constructor() { - } - - onChange = (newValue: any) => {}; - - onTouched = () => {}; - - writeValue(newValue: any): void { - this.dateValue = formatDate(newValue, 'yyyy-MM-dd', "en-US") - this.timeValue = formatDate(newValue, 'HH:mm:ss', 'en-US') - } - registerOnChange(fn: any): void { - this.onChange = fn; - } - registerOnTouched(fn: any): void { - this.onTouched = fn; - } - setDisabledState?(isDisabled: boolean): void { - this.disabled = isDisabled; - } - - @Input() - titleDate: string = "Date" - - @Input() - titleTime: string - - @Input() - disabled: boolean = false - - @Input() - hint: string - - timeValue - - dateValue - - ngOnInit(): void { - } - - dateOrTimeChanged() { - this.onChange(formatDate(this.dateValue + "T" + this.timeValue,"yyyy-MM-ddTHH:mm:ssZZZZZ", "en-us", "UTC")) - } - -} diff --git a/src-ui/src/app/components/common/input/date/date.component.html b/src-ui/src/app/components/common/input/date/date.component.html new file mode 100644 index 000000000..19f606a16 --- /dev/null +++ b/src-ui/src/app/components/common/input/date/date.component.html @@ -0,0 +1,16 @@ +
+ +
+ +
+ +
+
+
Invalid date.
+ {{hint}} +
diff --git a/src-ui/src/app/components/common/input/date/date.component.scss b/src-ui/src/app/components/common/input/date/date.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/common/input/date-time/date-time.component.spec.ts b/src-ui/src/app/components/common/input/date/date.component.spec.ts similarity index 55% rename from src-ui/src/app/components/common/input/date-time/date-time.component.spec.ts rename to src-ui/src/app/components/common/input/date/date.component.spec.ts index 0657768bd..ea92c7b30 100644 --- a/src-ui/src/app/components/common/input/date-time/date-time.component.spec.ts +++ b/src-ui/src/app/components/common/input/date/date.component.spec.ts @@ -1,20 +1,20 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { DateTimeComponent } from './date-time.component'; +import { DateComponent } from './date.component'; -describe('DateTimeComponent', () => { - let component: DateTimeComponent; - let fixture: ComponentFixture; +describe('DateComponent', () => { + let component: DateComponent; + let fixture: ComponentFixture; beforeEach(async () => { await TestBed.configureTestingModule({ - declarations: [ DateTimeComponent ] + declarations: [ DateComponent ] }) .compileComponents(); }); beforeEach(() => { - fixture = TestBed.createComponent(DateTimeComponent); + fixture = TestBed.createComponent(DateComponent); component = fixture.componentInstance; fixture.detectChanges(); }); diff --git a/src-ui/src/app/components/common/input/date/date.component.ts b/src-ui/src/app/components/common/input/date/date.component.ts new file mode 100644 index 000000000..b3b863581 --- /dev/null +++ b/src-ui/src/app/components/common/input/date/date.component.ts @@ -0,0 +1,32 @@ +import { Component, forwardRef, Input, OnInit, ViewChild } from '@angular/core'; +import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; +import { NgbDateAdapter, NgbDateParserFormatter, NgbDatepickerContent } from '@ng-bootstrap/ng-bootstrap'; +import { SettingsService } from 'src/app/services/settings.service'; +import { v4 as uuidv4 } from 'uuid'; +import { AbstractInputComponent } from '../abstract-input'; + + +@Component({ + providers: [{ + provide: NG_VALUE_ACCESSOR, + useExisting: forwardRef(() => DateComponent), + multi: true + }], + selector: 'app-input-date', + templateUrl: './date.component.html', + styleUrls: ['./date.component.scss'] +}) +export class DateComponent extends AbstractInputComponent implements OnInit { + + constructor(private settings: SettingsService) { + super() + } + + ngOnInit(): void { + super.ngOnInit() + this.placeholder = this.settings.getLocalizedDateInputFormat() + } + + placeholder: string + +} diff --git a/src-ui/src/app/components/common/input/number/number.component.html b/src-ui/src/app/components/common/input/number/number.component.html new file mode 100644 index 000000000..821226b16 --- /dev/null +++ b/src-ui/src/app/components/common/input/number/number.component.html @@ -0,0 +1,14 @@ +
+ +
+ +
+ +
+
+
+ {{error}} +
+ {{hint}} + +
diff --git a/src-ui/src/app/components/common/input/number/number.component.scss b/src-ui/src/app/components/common/input/number/number.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/common/input/number/number.component.spec.ts b/src-ui/src/app/components/common/input/number/number.component.spec.ts new file mode 100644 index 000000000..3476cbc22 --- /dev/null +++ b/src-ui/src/app/components/common/input/number/number.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { NumberComponent } from './number.component'; + +describe('NumberComponent', () => { + let component: NumberComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ NumberComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(NumberComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/common/input/number/number.component.ts b/src-ui/src/app/components/common/input/number/number.component.ts new file mode 100644 index 000000000..2d28a7801 --- /dev/null +++ b/src-ui/src/app/components/common/input/number/number.component.ts @@ -0,0 +1,39 @@ +import { Component, forwardRef } from '@angular/core'; +import { NG_VALUE_ACCESSOR } from '@angular/forms'; +import { FILTER_ASN_ISNULL } from 'src/app/data/filter-rule-type'; +import { DocumentService } from 'src/app/services/rest/document.service'; +import { AbstractInputComponent } from '../abstract-input'; + +@Component({ + providers: [{ + provide: NG_VALUE_ACCESSOR, + useExisting: forwardRef(() => NumberComponent), + multi: true + }], + selector: 'app-input-number', + templateUrl: './number.component.html', + styleUrls: ['./number.component.scss'] +}) +export class NumberComponent extends AbstractInputComponent { + + constructor(private documentService: DocumentService) { + super() + } + + nextAsn() { + if (this.value) { + return + } + this.documentService.listFiltered(1, 1, "archive_serial_number", true, [{rule_type: FILTER_ASN_ISNULL, value: "false"}]).subscribe( + results => { + if (results.count > 0) { + this.value = results.results[0].archive_serial_number + 1 + } else { + this.value = 1 + } + this.onChange(this.value) + } + ) + } + +} diff --git a/src-ui/src/app/components/common/input/select/select.component.html b/src-ui/src/app/components/common/input/select/select.component.html index 717aa7964..af4285f62 100644 --- a/src-ui/src/app/components/common/input/select/select.component.html +++ b/src-ui/src/app/components/common/input/select/select.component.html @@ -1,18 +1,38 @@ -
+
-
- -
- +
+ + +
+ +
-
{{hint}} -
\ No newline at end of file + + Suggestions:  + + {{s.name}}  + + + + +
diff --git a/src-ui/src/app/components/common/input/select/select.component.scss b/src-ui/src/app/components/common/input/select/select.component.scss index e69de29bb..8faec3bc0 100644 --- a/src-ui/src/app/components/common/input/select/select.component.scss +++ b/src-ui/src/app/components/common/input/select/select.component.scss @@ -0,0 +1 @@ +// styles for ng-select child are in styles.scss diff --git a/src-ui/src/app/components/common/input/select/select.component.ts b/src-ui/src/app/components/common/input/select/select.component.ts index e6e02ac87..921e3606c 100644 --- a/src-ui/src/app/components/common/input/select/select.component.ts +++ b/src-ui/src/app/components/common/input/select/select.component.ts @@ -1,4 +1,4 @@ -import { Component, EventEmitter, forwardRef, Input, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, forwardRef, Input, Output } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { AbstractInputComponent } from '../abstract-input'; @@ -16,6 +16,7 @@ export class SelectComponent extends AbstractInputComponent { constructor() { super() + this.addItemRef = this.addItem.bind(this) } @Input() @@ -30,11 +31,51 @@ export class SelectComponent extends AbstractInputComponent { @Input() allowNull: boolean = false + @Input() + suggestions: number[] + @Output() - createNew = new EventEmitter() - - showPlusButton(): boolean { + createNew = new EventEmitter() + + public addItemRef: (name) => void + + private _lastSearchTerm: string + + get allowCreateNew(): boolean { return this.createNew.observers.length > 0 } + getSuggestions() { + if (this.suggestions && this.items) { + return this.suggestions.filter(id => id != this.value).map(id => this.items.find(item => item.id == id)) + } else { + return [] + } + } + + addItem(name: string) { + if (name) this.createNew.next(name) + else this.createNew.next(this._lastSearchTerm) + this.clearLastSearchTerm() + } + + clickNew() { + this.createNew.next(this._lastSearchTerm) + this.clearLastSearchTerm() + } + + clearLastSearchTerm() { + this._lastSearchTerm = null + } + + onSearch($event) { + this._lastSearchTerm = $event.term + } + + onBlur() { + setTimeout(() => { + this.clearLastSearchTerm() + }, 3000); + } + } diff --git a/src-ui/src/app/components/common/input/tags/tags.component.html b/src-ui/src/app/components/common/input/tags/tags.component.html index 8029dd860..a9dd23934 100644 --- a/src-ui/src/app/components/common/input/tags/tags.component.html +++ b/src-ui/src/app/components/common/input/tags/tags.component.html @@ -1,30 +1,52 @@ -
- +
+ -
-
- -
+
+ -
- -
- -
-
+ + + + + + + + + +
+ +
+
+
-
-
{{hint}} + + Suggestions:  + + {{tag.name}}  + -
\ No newline at end of file + + + +
diff --git a/src-ui/src/app/components/common/input/tags/tags.component.scss b/src-ui/src/app/components/common/input/tags/tags.component.scss index f2635b7f2..2eaaa4f6d 100644 --- a/src-ui/src/app/components/common/input/tags/tags.component.scss +++ b/src-ui/src/app/components/common/input/tags/tags.component.scss @@ -1,10 +1,12 @@ -.tags-form-control { - height: auto; +.selected-icon { + min-width: 1em; + min-height: 1em; } +.tag-wrap { + font-size: 1rem; +} -.scrollable-menu { - height: auto; - max-height: 300px; - overflow-x: hidden; -} \ No newline at end of file +.tag-wrap-delete { + cursor: pointer; +} diff --git a/src-ui/src/app/components/common/input/tags/tags.component.ts b/src-ui/src/app/components/common/input/tags/tags.component.ts index 81bd9d470..8db444ba3 100644 --- a/src-ui/src/app/components/common/input/tags/tags.component.ts +++ b/src-ui/src/app/components/common/input/tags/tags.component.ts @@ -1,8 +1,6 @@ -import { ThrowStmt } from '@angular/compiler'; import { Component, forwardRef, Input, OnInit } from '@angular/core'; import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; -import { Observable } from 'rxjs'; import { TagEditDialogComponent } from 'src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component'; import { PaperlessTag } from 'src/app/data/paperless-tag'; import { TagService } from 'src/app/services/rest/tag.service'; @@ -19,18 +17,16 @@ import { TagService } from 'src/app/services/rest/tag.service'; }) export class TagsComponent implements OnInit, ControlValueAccessor { - constructor(private tagService: TagService, private modalService: NgbModal) { } - + constructor(private tagService: TagService, private modalService: NgbModal) { + this.createTagRef = this.createTag.bind(this) + } onChange = (newValue: number[]) => {}; - + onTouched = () => {}; writeValue(newValue: number[]): void { this.value = newValue - if (this.tags) { - this.displayValue = newValue - } } registerOnChange(fn: any): void { this.onChange = fn; @@ -45,7 +41,6 @@ export class TagsComponent implements OnInit, ControlValueAccessor { ngOnInit(): void { this.tagService.listAll().subscribe(result => { this.tags = result.results - this.displayValue = this.value }) } @@ -55,42 +50,74 @@ export class TagsComponent implements OnInit, ControlValueAccessor { @Input() hint - value: number[] + @Input() + suggestions: number[] - displayValue: number[] = [] + value: number[] tags: PaperlessTag[] + public createTagRef: (name) => void + + private _lastSearchTerm: string + getTag(id) { - return this.tags.find(tag => tag.id == id) + if (this.tags) { + return this.tags.find(tag => tag.id == id) + } else { + return null + } } removeTag(id) { - let index = this.displayValue.indexOf(id) + let index = this.value.indexOf(id) if (index > -1) { - this.displayValue.splice(index, 1) - this.onChange(this.displayValue) + let oldValue = this.value + oldValue.splice(index, 1) + this.value = [...oldValue] + this.onChange(this.value) } } - addTag(id) { - let index = this.displayValue.indexOf(id) - if (index == -1) { - this.displayValue.push(id) - this.onChange(this.displayValue) - } - } - - - createTag() { + createTag(name: string = null) { var modal = this.modalService.open(TagEditDialogComponent, {backdrop: 'static'}) modal.componentInstance.dialogMode = 'create' + if (name) modal.componentInstance.object = { name: name } + else if (this._lastSearchTerm) modal.componentInstance.object = { name: this._lastSearchTerm } modal.componentInstance.success.subscribe(newTag => { this.tagService.listAll().subscribe(tags => { this.tags = tags.results - this.addTag(newTag.id) + this.value = [...this.value, newTag.id] + this.onChange(this.value) }) }) } + getSuggestions() { + if (this.suggestions && this.tags) { + return this.suggestions.filter(id => !this.value.includes(id)).map(id => this.tags.find(tag => tag.id == id)) + } else { + return [] + } + } + + addTag(id) { + this.value = [...this.value, id] + this.onChange(this.value) + } + + clearLastSearchTerm() { + this._lastSearchTerm = null + } + + onSearch($event) { + this._lastSearchTerm = $event.term + } + + onBlur() { + setTimeout(() => { + this.clearLastSearchTerm() + }, 3000); + } + } diff --git a/src-ui/src/app/components/common/input/text/text.component.html b/src-ui/src/app/components/common/input/text/text.component.html index 3a43b052f..0c15c22a2 100644 --- a/src-ui/src/app/components/common/input/text/text.component.html +++ b/src-ui/src/app/components/common/input/text/text.component.html @@ -1,5 +1,8 @@
- + {{hint}} -
\ No newline at end of file +
+ {{error}} +
+
diff --git a/src-ui/src/app/components/common/page-header/page-header.component.html b/src-ui/src/app/components/common/page-header/page-header.component.html index fccce9b98..1174ee044 100644 --- a/src-ui/src/app/components/common/page-header/page-header.component.html +++ b/src-ui/src/app/components/common/page-header/page-header.component.html @@ -1,9 +1,9 @@ -
+

{{title}}

{{subTitle}}

-
+
diff --git a/src-ui/src/app/components/common/page-header/page-header.component.ts b/src-ui/src/app/components/common/page-header/page-header.component.ts index 93ec3bfb7..153e6bea6 100644 --- a/src-ui/src/app/components/common/page-header/page-header.component.ts +++ b/src-ui/src/app/components/common/page-header/page-header.component.ts @@ -1,21 +1,29 @@ -import { Component, Input, OnInit } from '@angular/core'; +import { Component, Input } from '@angular/core'; +import { Title } from '@angular/platform-browser'; +import { environment } from 'src/environments/environment'; @Component({ selector: 'app-page-header', templateUrl: './page-header.component.html', styleUrls: ['./page-header.component.scss'] }) -export class PageHeaderComponent implements OnInit { +export class PageHeaderComponent { - constructor() { } + constructor(private titleService: Title) { } + + _title = "" @Input() - title: string = "" + set title(title: string) { + this._title = title + this.titleService.setTitle(`${this.title} - ${environment.appTitle}`) + } + + get title() { + return this._title + } @Input() subTitle: string = "" - ngOnInit(): void { - } - } diff --git a/src-ui/src/app/components/common/select-dialog/select-dialog.component.html b/src-ui/src/app/components/common/select-dialog/select-dialog.component.html new file mode 100644 index 000000000..87758c6cf --- /dev/null +++ b/src-ui/src/app/components/common/select-dialog/select-dialog.component.html @@ -0,0 +1,15 @@ + + + diff --git a/src-ui/src/app/components/common/select-dialog/select-dialog.component.scss b/src-ui/src/app/components/common/select-dialog/select-dialog.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/common/select-dialog/select-dialog.component.spec.ts b/src-ui/src/app/components/common/select-dialog/select-dialog.component.spec.ts new file mode 100644 index 000000000..3810bcbea --- /dev/null +++ b/src-ui/src/app/components/common/select-dialog/select-dialog.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { SelectDialogComponent } from './select-dialog.component'; + +describe('SelectDialogComponent', () => { + let component: SelectDialogComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ SelectDialogComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(SelectDialogComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/common/select-dialog/select-dialog.component.ts b/src-ui/src/app/components/common/select-dialog/select-dialog.component.ts new file mode 100644 index 000000000..99bf9b91a --- /dev/null +++ b/src-ui/src/app/components/common/select-dialog/select-dialog.component.ts @@ -0,0 +1,34 @@ +import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; +import { ObjectWithId } from 'src/app/data/object-with-id'; + +@Component({ + selector: 'app-select-dialog', + templateUrl: './select-dialog.component.html', + styleUrls: ['./select-dialog.component.scss'] +}) + +export class SelectDialogComponent implements OnInit { + constructor(public activeModal: NgbActiveModal) { } + + @Output() + public selectClicked = new EventEmitter() + + @Input() + title = $localize`Select` + + @Input() + message = $localize`Please select an object` + + @Input() + objects: ObjectWithId[] = [] + + selected: number + + ngOnInit(): void { + } + + cancelClicked() { + this.activeModal.close() + } +} diff --git a/src-ui/src/app/components/common/tag/tag.component.html b/src-ui/src/app/components/common/tag/tag.component.html index 8b9632a65..bcf3542e9 100644 --- a/src-ui/src/app/components/common/tag/tag.component.html +++ b/src-ui/src/app/components/common/tag/tag.component.html @@ -1,2 +1,2 @@ -{{tag.name}} -{{tag.name}} \ No newline at end of file +{{tag.name}} +{{tag.name}} diff --git a/src-ui/src/app/components/common/tag/tag.component.ts b/src-ui/src/app/components/common/tag/tag.component.ts index c032c51db..e552ade9c 100644 --- a/src-ui/src/app/components/common/tag/tag.component.ts +++ b/src-ui/src/app/components/common/tag/tag.component.ts @@ -1,5 +1,5 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag'; +import { Component, Input, OnInit } from '@angular/core'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; @Component({ selector: 'app-tag', @@ -22,8 +22,4 @@ export class TagComponent implements OnInit { ngOnInit(): void { } - getColour() { - return TAG_COLOURS.find(c => c.id == this.tag.colour) - } - } diff --git a/src-ui/src/app/components/common/toasts/toasts.component.html b/src-ui/src/app/components/common/toasts/toasts.component.html index 04aa15a67..9d4ae6bb6 100644 --- a/src-ui/src/app/components/common/toasts/toasts.component.html +++ b/src-ui/src/app/components/common/toasts/toasts.component.html @@ -3,5 +3,6 @@ [header]="toast.title" [autohide]="true" [delay]="toast.delay" [class]="toast.classname" (hide)="toastService.closeToast(toast)"> - {{toast.content}} - \ No newline at end of file +

{{toast.content}}

+

+ diff --git a/src-ui/src/app/components/dashboard/dashboard.component.html b/src-ui/src/app/components/dashboard/dashboard.component.html index 627e7ff22..6b9ea950b 100644 --- a/src-ui/src/app/components/dashboard/dashboard.component.html +++ b/src-ui/src/app/components/dashboard/dashboard.component.html @@ -1,5 +1,8 @@ - - + +
diff --git a/src-ui/src/app/components/dashboard/dashboard.component.ts b/src-ui/src/app/components/dashboard/dashboard.component.ts index c7410c3f2..a5f857b99 100644 --- a/src-ui/src/app/components/dashboard/dashboard.component.ts +++ b/src-ui/src/app/components/dashboard/dashboard.component.ts @@ -1,7 +1,7 @@ import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; -import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; -import { environment } from 'src/environments/environment'; +import { Meta } from '@angular/platform-browser'; +import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; +import { SavedViewService } from 'src/app/services/rest/saved-view.service'; @Component({ @@ -12,15 +12,36 @@ import { environment } from 'src/environments/environment'; export class DashboardComponent implements OnInit { constructor( - public savedViewConfigService: SavedViewConfigService, - private titleService: Title) { } + private savedViewService: SavedViewService, + private meta: Meta + ) { } + get displayName() { + let tagFullName = this.meta.getTag('name=full_name') + let tagUsername = this.meta.getTag('name=username') + if (tagFullName && tagFullName.content) { + return tagFullName.content + } else if (tagUsername && tagUsername.content) { + return tagUsername.content + } else { + return null + } + } - savedViews = [] + get subtitle() { + if (this.displayName) { + return $localize`Hello ${this.displayName}, welcome to Paperless-ng!` + } else { + return $localize`Welcome to Paperless-ng!` + } + } + + savedViews: PaperlessSavedView[] = [] ngOnInit(): void { - this.savedViews = this.savedViewConfigService.getDashboardConfigs() - this.titleService.setTitle(`Dashboard - ${environment.appTitle}`) + this.savedViewService.listAll().subscribe(results => { + this.savedViews = results.results.filter(savedView => savedView.show_on_dashboard) + }) } } diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html index e63ecc47b..d05d8a667 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html @@ -1,21 +1,21 @@ - + - Show all + Show all - - + + - - +
CreatedTitleCreatedTitle
{{doc.created | date}}{{doc.title}} + {{doc.created | customDate}}{{doc.title | documentTitle}}
-
\ No newline at end of file +
diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss index e69de29bb..4e275dc81 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.scss @@ -0,0 +1,8 @@ +table { + overflow-wrap: anywhere; + table-layout: fixed; +} + +th:first-child { + width: 25%; +} diff --git a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts index a55bf57fc..20bc26cdc 100644 --- a/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts +++ b/src-ui/src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.ts @@ -1,8 +1,10 @@ -import { Component, Input, OnInit } from '@angular/core'; +import { Component, Input, OnDestroy, OnInit } from '@angular/core'; import { Router } from '@angular/router'; +import { Subscription } from 'rxjs'; import { PaperlessDocument } from 'src/app/data/paperless-document'; -import { SavedViewConfig } from 'src/app/data/saved-view-config'; +import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; import { DocumentListViewService } from 'src/app/services/document-list-view.service'; +import { ConsumerStatusService } from 'src/app/services/consumer-status.service'; import { DocumentService } from 'src/app/services/rest/document.service'; @Component({ @@ -10,31 +12,45 @@ import { DocumentService } from 'src/app/services/rest/document.service'; templateUrl: './saved-view-widget.component.html', styleUrls: ['./saved-view-widget.component.scss'] }) -export class SavedViewWidgetComponent implements OnInit { +export class SavedViewWidgetComponent implements OnInit, OnDestroy { constructor( private documentService: DocumentService, private router: Router, - private list: DocumentListViewService) { } - + private list: DocumentListViewService, + private consumerStatusService: ConsumerStatusService) { } + @Input() - savedView: SavedViewConfig + savedView: PaperlessSavedView documents: PaperlessDocument[] = [] + subscription: Subscription + ngOnInit(): void { - this.documentService.list(1,10,this.savedView.sortField,this.savedView.sortDirection,this.savedView.filterRules).subscribe(result => { + this.reload() + this.subscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => { + this.reload() + }) + } + + ngOnDestroy(): void { + this.subscription.unsubscribe() + } + + reload() { + this.documentService.listFiltered(1,10,this.savedView.sort_field, this.savedView.sort_reverse, this.savedView.filter_rules).subscribe(result => { this.documents = result.results }) } showAll() { - if (this.savedView.showInSideBar) { + if (this.savedView.show_in_sidebar) { this.router.navigate(['view', this.savedView.id]) } else { - this.list.load(this.savedView) + this.list.loadSavedView(this.savedView, true) this.router.navigate(["documents"]) - } + } } } diff --git a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html index 50d844b36..a29b50f78 100644 --- a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html +++ b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html @@ -1,6 +1,6 @@ - + -

Documents in inbox: {{statistics.documents_inbox}}

-

Total documents: {{statistics.documents_total}}

+

Documents in inbox: {{statistics?.documents_inbox}}

+

Total documents: {{statistics?.documents_total}}

-
\ No newline at end of file +
diff --git a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts index 73eee698c..f1488a66f 100644 --- a/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts +++ b/src-ui/src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.ts @@ -1,6 +1,7 @@ import { HttpClient } from '@angular/common/http'; -import { Component, OnInit } from '@angular/core'; -import { Observable } from 'rxjs'; +import { Component, OnDestroy, OnInit } from '@angular/core'; +import { Observable, Subscription } from 'rxjs'; +import { ConsumerStatusService } from 'src/app/services/consumer-status.service'; import { environment } from 'src/environments/environment'; export interface Statistics { @@ -14,20 +15,34 @@ export interface Statistics { templateUrl: './statistics-widget.component.html', styleUrls: ['./statistics-widget.component.scss'] }) -export class StatisticsWidgetComponent implements OnInit { +export class StatisticsWidgetComponent implements OnInit, OnDestroy { - constructor(private http: HttpClient) { } + constructor(private http: HttpClient, + private consumerStatusService: ConsumerStatusService) { } statistics: Statistics = {} - getStatistics(): Observable { + subscription: Subscription + + private getStatistics(): Observable { return this.http.get(`${environment.apiBaseUrl}statistics/`) } - - ngOnInit(): void { + + reload() { this.getStatistics().subscribe(statistics => { this.statistics = statistics }) } + ngOnInit(): void { + this.reload() + this.subscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(status => { + this.reload() + }) + } + + ngOnDestroy(): void { + this.subscription.unsubscribe() + } + } diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html index 013486a47..c4f7fa21c 100644 --- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html +++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html @@ -1,18 +1,52 @@ - - + +
- - + multiple="true" contentClassName="justify-content-center d-flex align-items-center py-5 px-2" [showBrowseBtn]=true + browseBtnClassName="btn btn-sm btn-outline-primary ml-2" i18n-dropZoneLabel i18n-browseBtnLabel>
-
-

Uploading {{uploadStatus.length}} file(s)

- - +

{{getStatusSummary()}}

+
+ +
+
+

+ {getStatusHidden().length, plural, =1 {One more document} other {{{getStatusHidden().length}} more documents}} +  •  + Show all +

+
+
+ +
+
- \ No newline at end of file + + + + +
{{status.filename}}
+

{{status.message}}

+ +
+ +
+
+
diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss index e69de29bb..c13a7bd47 100644 --- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss +++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.scss @@ -0,0 +1,35 @@ +@import "/src/theme"; + +form { + position: relative; +} + +.alert-heading { + font-size: 80%; + font-weight: bold; +} + +.alerts-hidden { + .btn { + line-height: 1; + } +} + +.btn-open { + line-height: 1; + + svg { + margin-top: -1px; + } +} + +.progress { + position: absolute; + top: 0; + right: 0; + bottom: 0; + left: 0; + height: auto; + mix-blend-mode: soft-light; + pointer-events: none; +} diff --git a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts index 2ea4825f1..e74e2bc41 100644 --- a/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts +++ b/src-ui/src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts @@ -1,14 +1,10 @@ import { HttpEventType } from '@angular/common/http'; import { Component, OnInit } from '@angular/core'; import { FileSystemFileEntry, NgxFileDropEntry } from 'ngx-file-drop'; +import { ConsumerStatusService, FileStatus, FileStatusPhase } from 'src/app/services/consumer-status.service'; import { DocumentService } from 'src/app/services/rest/document.service'; -import { Toast, ToastService } from 'src/app/services/toast.service'; - -interface UploadStatus { - loaded: number - total: number -} +const MAX_ALERTS = 5 @Component({ selector: 'app-upload-file-widget', @@ -16,8 +12,89 @@ interface UploadStatus { styleUrls: ['./upload-file-widget.component.scss'] }) export class UploadFileWidgetComponent implements OnInit { + alertsExpanded = false - constructor(private documentService: DocumentService, private toastService: ToastService) { } + constructor( + private documentService: DocumentService, + private consumerStatusService: ConsumerStatusService + ) { } + + getStatus() { + return this.consumerStatusService.getConsumerStatus().slice(0, MAX_ALERTS) + } + + getStatusSummary() { + let strings = [] + let countUploadingAndProcessing = this.consumerStatusService.getConsumerStatusNotCompleted().length + let countFailed = this.getStatusFailed().length + let countSuccess = this.getStatusSuccess().length + if (countUploadingAndProcessing > 0) { + strings.push($localize`Processing: ${countUploadingAndProcessing}`) + } + if (countFailed > 0) { + strings.push($localize`Failed: ${countFailed}`) + } + if (countSuccess > 0) { + strings.push($localize`Added: ${countSuccess}`) + } + return strings.join($localize`:this string is used to separate processing, failed and added on the file upload widget:, `) + } + + getStatusHidden() { + if (this.consumerStatusService.getConsumerStatus().length < MAX_ALERTS) return [] + else return this.consumerStatusService.getConsumerStatus().slice(MAX_ALERTS) + } + + getStatusUploading() { + return this.consumerStatusService.getConsumerStatus(FileStatusPhase.UPLOADING) + } + + getStatusFailed() { + return this.consumerStatusService.getConsumerStatus(FileStatusPhase.FAILED) + } + + getStatusSuccess() { + return this.consumerStatusService.getConsumerStatus(FileStatusPhase.SUCCESS) + } + + getStatusCompleted() { + return this.consumerStatusService.getConsumerStatusCompleted() + } + getTotalUploadProgress() { + let current = 0 + let max = 0 + + this.getStatusUploading().forEach(status => { + current += status.currentPhaseProgress + max += status.currentPhaseMaxProgress + }) + + return current / Math.max(max, 1) + } + + isFinished(status: FileStatus) { + return status.phase == FileStatusPhase.FAILED || status.phase == FileStatusPhase.SUCCESS + } + + getStatusColor(status: FileStatus) { + switch (status.phase) { + case FileStatusPhase.PROCESSING: + case FileStatusPhase.UPLOADING: + return "primary" + case FileStatusPhase.FAILED: + return "danger" + case FileStatusPhase.SUCCESS: + return "success" + } + } + + dismiss(status: FileStatus) { + this.consumerStatusService.dismiss(status) + } + + dismissCompleted() { + this.consumerStatusService.dismissCompleted() + } ngOnInit(): void { } @@ -28,54 +105,39 @@ export class UploadFileWidgetComponent implements OnInit { public fileLeave(event){ } - uploadStatus: UploadStatus[] = [] - completedFiles = 0 - - uploadVisible = false - - get loadedSum() { - return this.uploadStatus.map(s => s.loaded).reduce((a,b) => a+b, this.completedFiles > 0 ? 1 : 0) - } - - get totalSum() { - return this.uploadStatus.map(s => s.total).reduce((a,b) => a+b, 1) - } - public dropped(files: NgxFileDropEntry[]) { for (const droppedFile of files) { if (droppedFile.fileEntry.isFile) { - let uploadStatusObject: UploadStatus = {loaded: 0, total: 1} - this.uploadStatus.push(uploadStatusObject) - this.uploadVisible = true const fileEntry = droppedFile.fileEntry as FileSystemFileEntry; fileEntry.file((file: File) => { let formData = new FormData() formData.append('document', file, file.name) + let status = this.consumerStatusService.newFileUpload(file.name) + + status.message = $localize`Connecting...` this.documentService.uploadDocument(formData).subscribe(event => { if (event.type == HttpEventType.UploadProgress) { - uploadStatusObject.loaded = event.loaded - uploadStatusObject.total = event.total + status.updateProgress(FileStatusPhase.UPLOADING, event.loaded, event.total) + status.message = $localize`Uploading...` } else if (event.type == HttpEventType.Response) { - this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) - this.completedFiles += 1 - this.toastService.showToast(Toast.make("Information", "The document has been uploaded and will be processed by the consumer shortly.")) + status.taskId = event.body["task_id"] + status.message = $localize`Upload complete, waiting...` } - + }, error => { - this.uploadStatus.splice(this.uploadStatus.indexOf(uploadStatusObject), 1) - this.completedFiles += 1 switch (error.status) { case 400: { - this.toastService.showToast(Toast.makeError(`There was an error while uploading the document: ${error.error.document}`)) + this.consumerStatusService.fail(status, error.error.document) break; } default: { - this.toastService.showToast(Toast.makeError("An error has occurred while uploading the document. Sorry!")) + this.consumerStatusService.fail(status, $localize`HTTP error: ${error.status} ${error.statusText}`) break; } } + }) }); } diff --git a/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html b/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html index 0caf55f11..17c7ffe1e 100644 --- a/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html +++ b/src-ui/src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html @@ -1,16 +1,16 @@ - + -

Paperless is running! :)

-

You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. - After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and have them displayed on the dashboard instead of this message.

-

Paperless offers some more features that try to make your life easier, such as:

+

Paperless is running! :)

+

You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. + After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message.

+

Paperless offers some more features that try to make your life easier:

    -
  • Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.
  • -
  • You can configure paperless to read your mails and add documents from attached files.
  • +
  • Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically.
  • +
  • You can configure paperless to read your mails and add documents from attached files.
-

Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.

+

Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general.

-
\ No newline at end of file +
diff --git a/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html b/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html index 1d7d2d906..1cd440306 100644 --- a/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html +++ b/src-ui/src/app/components/dashboard/widgets/widget-frame/widget-frame.component.html @@ -4,9 +4,9 @@
{{title}}
- +
-
\ No newline at end of file +
diff --git a/src-ui/src/app/components/document-asn/document-asn.component.html b/src-ui/src/app/components/document-asn/document-asn.component.html new file mode 100644 index 000000000..8cb969ea0 --- /dev/null +++ b/src-ui/src/app/components/document-asn/document-asn.component.html @@ -0,0 +1 @@ +

Searching document with asn {{asn}}

diff --git a/src-ui/src/app/components/document-asn/document-asn.component.scss b/src-ui/src/app/components/document-asn/document-asn.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/search/result-highlight/result-highlight.component.spec.ts b/src-ui/src/app/components/document-asn/document-asn.component.spec.ts similarity index 51% rename from src-ui/src/app/components/search/result-highlight/result-highlight.component.spec.ts rename to src-ui/src/app/components/document-asn/document-asn.component.spec.ts index 8e00a9d0b..5a9826f8d 100644 --- a/src-ui/src/app/components/search/result-highlight/result-highlight.component.spec.ts +++ b/src-ui/src/app/components/document-asn/document-asn.component.spec.ts @@ -1,20 +1,20 @@ import { ComponentFixture, TestBed } from '@angular/core/testing'; -import { ResultHighlightComponent } from './result-highlight.component'; +import { DocumentAsnComponent } from './document-asn.component'; -describe('ResultHighlightComponent', () => { - let component: ResultHighlightComponent; - let fixture: ComponentFixture; +describe('DocumentASNComponentComponent', () => { + let component: DocumentAsnComponent; + let fixture: ComponentFixture; beforeEach(async () => { await TestBed.configureTestingModule({ - declarations: [ ResultHighlightComponent ] + declarations: [ DocumentAsnComponent ] }) .compileComponents(); }); beforeEach(() => { - fixture = TestBed.createComponent(ResultHighlightComponent); + fixture = TestBed.createComponent(DocumentAsnComponent); component = fixture.componentInstance; fixture.detectChanges(); }); diff --git a/src-ui/src/app/components/document-asn/document-asn.component.ts b/src-ui/src/app/components/document-asn/document-asn.component.ts new file mode 100644 index 000000000..550c331b9 --- /dev/null +++ b/src-ui/src/app/components/document-asn/document-asn.component.ts @@ -0,0 +1,34 @@ +import { Component, OnInit } from '@angular/core'; +import {DocumentService} from "../../services/rest/document.service"; +import {ActivatedRoute, Router} from "@angular/router"; +import {FILTER_ASN} from "../../data/filter-rule-type"; + +@Component({ + selector: 'app-document-asncomponent', + templateUrl: './document-asn.component.html', + styleUrls: ['./document-asn.component.scss'] +}) +export class DocumentAsnComponent implements OnInit { + + asn: string + constructor( + private documentsService: DocumentService, + private route: ActivatedRoute, + private router: Router) { } + + + ngOnInit(): void { + + this.route.paramMap.subscribe(paramMap => { + this.asn = paramMap.get('id'); + this.documentsService.listAllFilteredIds([{rule_type: FILTER_ASN, value: this.asn}]).subscribe(documentId => { + if (documentId.length == 1) { + this.router.navigate(['documents', documentId[0]]) + } else { + this.router.navigate(['404']) + } + }) + }) + + } +} diff --git a/src-ui/src/app/components/document-detail/document-detail.component.html b/src-ui/src/app/components/document-detail/document-detail.component.html index 970fa361c..10a1ee83c 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.html +++ b/src-ui/src/app/components/document-detail/document-detail.component.html @@ -1,9 +1,18 @@ - + @@ -41,27 +53,23 @@
-
- -
+ +
+ +
+ + + +
+ + + +
diff --git a/src-ui/src/app/components/document-detail/document-detail.component.scss b/src-ui/src/app/components/document-detail/document-detail.component.scss index 998653bab..346c9f4b0 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.scss +++ b/src-ui/src/app/components/document-detail/document-detail.component.scss @@ -1,6 +1,9 @@ -.pdf-viewer-container { +.preview-sticky { height: calc(100vh - 160px); top: 70px; position: sticky; +} + +.pdf-viewer-container { background-color: gray; } diff --git a/src-ui/src/app/components/document-detail/document-detail.component.ts b/src-ui/src/app/components/document-detail/document-detail.component.ts index fcbbf840c..e711b3416 100644 --- a/src-ui/src/app/components/document-detail/document-detail.component.ts +++ b/src-ui/src/app/components/document-detail/document-detail.component.ts @@ -1,21 +1,26 @@ import { Component, OnInit, ViewChild, ElementRef } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; -import { Title } from '@angular/platform-browser'; import { ActivatedRoute, Router } from '@angular/router'; import { NgbModal, NgbNav } from '@ng-bootstrap/ng-bootstrap'; import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; import { PaperlessDocument } from 'src/app/data/paperless-document'; import { PaperlessDocumentMetadata } from 'src/app/data/paperless-document-metadata'; import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; +import { DocumentTitlePipe } from 'src/app/pipes/document-title.pipe'; import { DocumentListViewService } from 'src/app/services/document-list-view.service'; import { OpenDocumentsService } from 'src/app/services/open-documents.service'; import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; import { DocumentService } from 'src/app/services/rest/document.service'; -import { environment } from 'src/environments/environment'; -import { DeleteDialogComponent } from '../common/delete-dialog/delete-dialog.component'; +import { ConfirmDialogComponent } from '../common/confirm-dialog/confirm-dialog.component'; import { CorrespondentEditDialogComponent } from '../manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component'; import { DocumentTypeEditDialogComponent } from '../manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component'; +import { PDFDocumentProxy } from 'ng2-pdf-viewer'; +import { ToastService } from 'src/app/services/toast.service'; +import { TextComponent } from '../common/input/text/text.component'; +import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; +import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions'; +import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type'; @Component({ selector: 'app-document-detail', @@ -24,12 +29,21 @@ import { DocumentTypeEditDialogComponent } from '../manage/document-type-list/do }) export class DocumentDetailComponent implements OnInit { - public expandOriginalMetadata = false; - public expandArchivedMetadata = false; + @ViewChild("inputTitle") + titleInput: TextComponent + + expandOriginalMetadata = false + expandArchivedMetadata = false + + error: any + + networkActive = false documentId: number document: PaperlessDocument metadata: PaperlessDocumentMetadata + suggestions: PaperlessDocumentSuggestions + title: string previewUrl: string downloadUrl: string @@ -48,6 +62,9 @@ export class DocumentDetailComponent implements OnInit { tags: new FormControl([]) }) + previewCurrentPage: number = 1 + previewNumPages: number = 1 + @ViewChild('nav') nav: NgbNav @ViewChild('pdfPreview') set pdfPreview(element) { // this gets called when compontent added or removed from DOM @@ -66,7 +83,13 @@ export class DocumentDetailComponent implements OnInit { private modalService: NgbModal, private openDocumentService: OpenDocumentsService, private documentListViewService: DocumentListViewService, - private titleService: Title) { } + private documentTitlePipe: DocumentTitlePipe, + private toastService: ToastService, + private settings: SettingsService) { } + + get useNativePdfViewer(): boolean { + return this.settings.get(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER) + } getContentType() { return this.metadata?.has_archive_version ? 'application/pdf' : this.metadata?.original_mime_type @@ -85,6 +108,7 @@ export class DocumentDetailComponent implements OnInit { this.previewUrl = this.documentsService.getPreviewUrl(this.documentId) this.downloadUrl = this.documentsService.getDownloadUrl(this.documentId) this.downloadOriginalUrl = this.documentsService.getDownloadUrl(this.documentId, true) + this.suggestions = null if (this.openDocumentService.getOpenDocument(this.documentId)) { this.updateComponent(this.openDocumentService.getOpenDocument(this.documentId)) } else { @@ -99,17 +123,24 @@ export class DocumentDetailComponent implements OnInit { updateComponent(doc: PaperlessDocument) { this.document = doc - this.titleService.setTitle(`${doc.title} - ${environment.appTitle}`) this.documentsService.getMetadata(doc.id).subscribe(result => { this.metadata = result + }, error => { + this.metadata = null }) - this.title = doc.title + this.documentsService.getSuggestions(doc.id).subscribe(result => { + this.suggestions = result + }, error => { + this.suggestions = null + }) + this.title = this.documentTitlePipe.transform(doc.title) this.documentForm.patchValue(doc) } - createDocumentType() { + createDocumentType(newName: string) { var modal = this.modalService.open(DocumentTypeEditDialogComponent, {backdrop: 'static'}) modal.componentInstance.dialogMode = 'create' + if (newName) modal.componentInstance.object = { name: newName } modal.componentInstance.success.subscribe(newDocumentType => { this.documentTypeService.listAll().subscribe(documentTypes => { this.documentTypes = documentTypes.results @@ -118,9 +149,10 @@ export class DocumentDetailComponent implements OnInit { }) } - createCorrespondent() { + createCorrespondent(newName: string) { var modal = this.modalService.open(CorrespondentEditDialogComponent, {backdrop: 'static'}) modal.componentInstance.dialogMode = 'create' + if (newName) modal.componentInstance.object = { name: newName } modal.componentInstance.success.subscribe(newCorrespondent => { this.correspondentService.listAll().subscribe(correspondents => { this.correspondents = correspondents.results @@ -138,55 +170,76 @@ export class DocumentDetailComponent implements OnInit { } save() { + this.networkActive = true this.documentsService.update(this.document).subscribe(result => { this.close() + this.networkActive = false + this.error = null + }, error => { + this.networkActive = false + this.error = error.error }) } saveEditNext() { + this.networkActive = true this.documentsService.update(this.document).subscribe(result => { + this.error = null this.documentListViewService.getNext(this.document.id).subscribe(nextDocId => { + this.networkActive = false if (nextDocId) { this.openDocumentService.closeDocument(this.document) this.router.navigate(['documents', nextDocId]) + this.titleInput.focus() } + }, error => { + this.networkActive = false }) + }, error => { + this.networkActive = false + this.error = error.error }) } close() { this.openDocumentService.closeDocument(this.document) - if (this.documentListViewService.savedViewId) { - this.router.navigate(['view', this.documentListViewService.savedViewId]) + if (this.documentListViewService.activeSavedViewId) { + this.router.navigate(['view', this.documentListViewService.activeSavedViewId]) } else { this.router.navigate(['documents']) } } delete() { - let modal = this.modalService.open(DeleteDialogComponent, {backdrop: 'static'}) - modal.componentInstance.message = `Do you really want to delete document '${this.document.title}'?` - modal.componentInstance.message2 = `The files for this document will be deleted permanently. This operation cannot be undone.` - modal.componentInstance.deleteClicked.subscribe(() => { + let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + modal.componentInstance.title = $localize`Confirm delete` + modal.componentInstance.messageBold = $localize`Do you really want to delete document "${this.document.title}"?` + modal.componentInstance.message = $localize`The files for this document will be deleted permanently. This operation cannot be undone.` + modal.componentInstance.btnClass = "btn-danger" + modal.componentInstance.btnCaption = $localize`Delete document` + modal.componentInstance.confirmClicked.subscribe(() => { + modal.componentInstance.buttonsEnabled = false this.documentsService.delete(this.document).subscribe(() => { modal.close() this.close() + }, error => { + this.toastService.showError($localize`Error deleting document: ${JSON.stringify(error)}`) + modal.componentInstance.buttonsEnabled = true }) }) } + moreLike() { + this.documentListViewService.quickFilter([{rule_type: FILTER_FULLTEXT_MORELIKE, value: this.documentId.toString()}]) + } + hasNext() { return this.documentListViewService.hasNext(this.documentId) } - previewCreated() { - console.log('Preview Created'); - + pdfPreviewLoaded(pdf: PDFDocumentProxy) { + this.previewNumPages = pdf.numPages } - mobilePreviewCreated() { - console.log('Mobile Preview Created'); - - } } diff --git a/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.html b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.html new file mode 100644 index 000000000..b0f323de0 --- /dev/null +++ b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.html @@ -0,0 +1,23 @@ +
+ + {{title}} +
+ +
+ + + + + + + +
{{m.prefix}}:{{m.key}}
+
diff --git a/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.scss b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.scss new file mode 100644 index 000000000..8f851848b --- /dev/null +++ b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.scss @@ -0,0 +1,3 @@ +.metadata-column { + overflow-wrap: anywhere; +} diff --git a/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.spec.ts b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.spec.ts new file mode 100644 index 000000000..2bd96760b --- /dev/null +++ b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { MetadataCollapseComponent } from './metadata-collapse.component'; + +describe('MetadataCollapseComponent', () => { + let component: MetadataCollapseComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ MetadataCollapseComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(MetadataCollapseComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts new file mode 100644 index 000000000..34bbbd655 --- /dev/null +++ b/src-ui/src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts @@ -0,0 +1,23 @@ +import { Component, Input, OnInit } from '@angular/core'; + +@Component({ + selector: 'app-metadata-collapse', + templateUrl: './metadata-collapse.component.html', + styleUrls: ['./metadata-collapse.component.scss'] +}) +export class MetadataCollapseComponent implements OnInit { + + constructor() { } + + expand = false + + @Input() + metadata + + @Input() + title = $localize`Metadata` + + ngOnInit(): void { + } + +} diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html new file mode 100644 index 000000000..90cda1224 --- /dev/null +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.html @@ -0,0 +1,79 @@ +
+
+ +
+
+ +
+ + +
+
+
+
+
+ + + + + + + +
+
+
+
+ +
+ + +
+
+ + +
+
diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.scss b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.scss new file mode 100644 index 000000000..e69de29bb diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts new file mode 100644 index 000000000..140d73301 --- /dev/null +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.spec.ts @@ -0,0 +1,25 @@ +import { ComponentFixture, TestBed } from '@angular/core/testing'; + +import { BulkEditorComponent } from './bulk-editor.component'; + +describe('BulkEditorComponent', () => { + let component: BulkEditorComponent; + let fixture: ComponentFixture; + + beforeEach(async () => { + await TestBed.configureTestingModule({ + declarations: [ BulkEditorComponent ] + }) + .compileComponents(); + }); + + beforeEach(() => { + fixture = TestBed.createComponent(BulkEditorComponent); + component = fixture.componentInstance; + fixture.detectChanges(); + }); + + it('should create', () => { + expect(component).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts new file mode 100644 index 000000000..310290291 --- /dev/null +++ b/src-ui/src/app/components/document-list/bulk-editor/bulk-editor.component.ts @@ -0,0 +1,217 @@ +import { Component } from '@angular/core'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; +import { TagService } from 'src/app/services/rest/tag.service'; +import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; +import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; +import { DocumentListViewService } from 'src/app/services/document-list-view.service'; +import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { DocumentService, SelectionDataItem } from 'src/app/services/rest/document.service'; +import { OpenDocumentsService } from 'src/app/services/open-documents.service'; +import { ConfirmDialogComponent } from 'src/app/components/common/confirm-dialog/confirm-dialog.component'; +import { ChangedItems, FilterableDropdownSelectionModel } from '../../common/filterable-dropdown/filterable-dropdown.component'; +import { ToggleableItemState } from '../../common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; +import { MatchingModel } from 'src/app/data/matching-model'; +import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; +import { ToastService } from 'src/app/services/toast.service'; +import { saveAs } from 'file-saver'; + +@Component({ + selector: 'app-bulk-editor', + templateUrl: './bulk-editor.component.html', + styleUrls: ['./bulk-editor.component.scss'] +}) +export class BulkEditorComponent { + + tags: PaperlessTag[] + correspondents: PaperlessCorrespondent[] + documentTypes: PaperlessDocumentType[] + + tagSelectionModel = new FilterableDropdownSelectionModel() + correspondentSelectionModel = new FilterableDropdownSelectionModel() + documentTypeSelectionModel = new FilterableDropdownSelectionModel() + + constructor( + private documentTypeService: DocumentTypeService, + private tagService: TagService, + private correspondentService: CorrespondentService, + public list: DocumentListViewService, + private documentService: DocumentService, + private modalService: NgbModal, + private openDocumentService: OpenDocumentsService, + private settings: SettingsService, + private toastService: ToastService + ) { } + + applyOnClose: boolean = this.settings.get(SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE) + showConfirmationDialogs: boolean = this.settings.get(SETTINGS_KEYS.BULK_EDIT_CONFIRMATION_DIALOGS) + + ngOnInit() { + this.tagService.listAll().subscribe(result => this.tags = result.results) + this.correspondentService.listAll().subscribe(result => this.correspondents = result.results) + this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results) + } + + private executeBulkOperation(modal, method: string, args) { + if (modal) { + modal.componentInstance.buttonsEnabled = false + } + this.documentService.bulkEdit(Array.from(this.list.selected), method, args).subscribe( + response => { + this.list.reload() + this.list.reduceSelectionToFilter() + this.list.selected.forEach(id => { + this.openDocumentService.refreshDocument(id) + }) + if (modal) { + modal.close() + } + }, error => { + if (modal) { + modal.componentInstance.buttonsEnabled = true + } + this.toastService.showError($localize`Error executing bulk operation: ${JSON.stringify(error.error)}`) + } + ) + } + + private applySelectionData(items: SelectionDataItem[], selectionModel: FilterableDropdownSelectionModel) { + let selectionData = new Map() + items.forEach(i => { + if (i.document_count == this.list.selected.size) { + selectionData.set(i.id, ToggleableItemState.Selected) + } else if (i.document_count > 0) { + selectionData.set(i.id, ToggleableItemState.PartiallySelected) + } + }) + selectionModel.init(selectionData) + } + + openTagsDropdown() { + this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { + this.applySelectionData(s.selected_tags, this.tagSelectionModel) + }) + } + + openDocumentTypeDropdown() { + this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { + this.applySelectionData(s.selected_document_types, this.documentTypeSelectionModel) + }) + } + + openCorrespondentDropdown() { + this.documentService.getSelectionData(Array.from(this.list.selected)).subscribe(s => { + this.applySelectionData(s.selected_correspondents, this.correspondentSelectionModel) + }) + } + + private _localizeList(items: MatchingModel[]) { + if (items.length == 0) { + return "" + } else if (items.length == 1) { + return $localize`"${items[0].name}"` + } else if (items.length == 2) { + return $localize`:This is for messages like 'modify "tag1" and "tag2"':"${items[0].name}" and "${items[1].name}"` + } else { + let list = items.slice(0, items.length - 1).map(i => $localize`"${i.name}"`).join($localize`:this is used to separate enumerations and should probably be a comma and a whitespace in most languages:, `) + return $localize`:this is for messages like 'modify "tag1", "tag2" and "tag3"':${list} and "${items[items.length - 1].name}"` + } + } + + setTags(changedTags: ChangedItems) { + if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length == 0) return + + if (this.showConfirmationDialogs) { + let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + modal.componentInstance.title = $localize`Confirm tags assignment` + if (changedTags.itemsToAdd.length == 1 && changedTags.itemsToRemove.length == 0) { + let tag = changedTags.itemsToAdd[0] + modal.componentInstance.message = $localize`This operation will add the tag "${tag.name}" to ${this.list.selected.size} selected document(s).` + } else if (changedTags.itemsToAdd.length > 1 && changedTags.itemsToRemove.length == 0) { + modal.componentInstance.message = $localize`This operation will add the tags ${this._localizeList(changedTags.itemsToAdd)} to ${this.list.selected.size} selected document(s).` + } else if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length == 1) { + let tag = changedTags.itemsToRemove[0] + modal.componentInstance.message = $localize`This operation will remove the tag "${tag.name}" from ${this.list.selected.size} selected document(s).` + } else if (changedTags.itemsToAdd.length == 0 && changedTags.itemsToRemove.length > 1) { + modal.componentInstance.message = $localize`This operation will remove the tags ${this._localizeList(changedTags.itemsToRemove)} from ${this.list.selected.size} selected document(s).` + } else { + modal.componentInstance.message = $localize`This operation will add the tags ${this._localizeList(changedTags.itemsToAdd)} and remove the tags ${this._localizeList(changedTags.itemsToRemove)} on ${this.list.selected.size} selected document(s).` + } + + modal.componentInstance.btnClass = "btn-warning" + modal.componentInstance.btnCaption = $localize`Confirm` + modal.componentInstance.confirmClicked.subscribe(() => { + this.executeBulkOperation(modal, 'modify_tags', {"add_tags": changedTags.itemsToAdd.map(t => t.id), "remove_tags": changedTags.itemsToRemove.map(t => t.id)}) + }) + } else { + this.executeBulkOperation(null, 'modify_tags', {"add_tags": changedTags.itemsToAdd.map(t => t.id), "remove_tags": changedTags.itemsToRemove.map(t => t.id)}) + } + } + + setCorrespondents(changedCorrespondents: ChangedItems) { + if (changedCorrespondents.itemsToAdd.length == 0 && changedCorrespondents.itemsToRemove.length == 0) return + + let correspondent = changedCorrespondents.itemsToAdd.length > 0 ? changedCorrespondents.itemsToAdd[0] : null + + if (this.showConfirmationDialogs) { + let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + modal.componentInstance.title = $localize`Confirm correspondent assignment` + if (correspondent) { + modal.componentInstance.message = $localize`This operation will assign the correspondent "${correspondent.name}" to ${this.list.selected.size} selected document(s).` + } else { + modal.componentInstance.message = $localize`This operation will remove the correspondent from ${this.list.selected.size} selected document(s).` + } + modal.componentInstance.btnClass = "btn-warning" + modal.componentInstance.btnCaption = $localize`Confirm` + modal.componentInstance.confirmClicked.subscribe(() => { + this.executeBulkOperation(modal, 'set_correspondent', {"correspondent": correspondent ? correspondent.id : null}) + }) + } else { + this.executeBulkOperation(null, 'set_correspondent', {"correspondent": correspondent ? correspondent.id : null}) + } + } + + setDocumentTypes(changedDocumentTypes: ChangedItems) { + if (changedDocumentTypes.itemsToAdd.length == 0 && changedDocumentTypes.itemsToRemove.length == 0) return + + let documentType = changedDocumentTypes.itemsToAdd.length > 0 ? changedDocumentTypes.itemsToAdd[0] : null + + if (this.showConfirmationDialogs) { + let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + modal.componentInstance.title = $localize`Confirm document type assignment` + if (documentType) { + modal.componentInstance.message = $localize`This operation will assign the document type "${documentType.name}" to ${this.list.selected.size} selected document(s).` + } else { + modal.componentInstance.message = $localize`This operation will remove the document type from ${this.list.selected.size} selected document(s).` + } + modal.componentInstance.btnClass = "btn-warning" + modal.componentInstance.btnCaption = $localize`Confirm` + modal.componentInstance.confirmClicked.subscribe(() => { + this.executeBulkOperation(modal, 'set_document_type', {"document_type": documentType ? documentType.id : null}) + }) + } else { + this.executeBulkOperation(null, 'set_document_type', {"document_type": documentType ? documentType.id : null}) + } + } + + applyDelete() { + let modal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + modal.componentInstance.delayConfirm(5) + modal.componentInstance.title = $localize`Delete confirm` + modal.componentInstance.messageBold = $localize`This operation will permanently delete ${this.list.selected.size} selected document(s).` + modal.componentInstance.message = $localize`This operation cannot be undone.` + modal.componentInstance.btnClass = "btn-danger" + modal.componentInstance.btnCaption = $localize`Delete document(s)` + modal.componentInstance.confirmClicked.subscribe(() => { + modal.componentInstance.buttonsEnabled = false + this.executeBulkOperation(modal, "delete", {}) + }) + } + + downloadSelected(content = "archive") { + this.documentService.bulkDownload(Array.from(this.list.selected), content).subscribe((result: any) => { + saveAs(result, 'documents.zip'); + }) + } +} diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html index 8f3fced66..d9132c9aa 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.html @@ -1,55 +1,96 @@ -
+
-
- +
+ + +
+
+ + +
+
+
-
+
-
+
- {{(document.correspondent$ | async)?.name}} + {{(document.correspondent$ | async)?.name}} {{(document.correspondent$ | async)?.name}}: {{document.title | documentTitle}} - +
-
#{{document.archive_serial_number}}

- - {{getDetailsAsString()}} + + {{contentTrimmed}}

-
+
- Created: {{document.created | date}} + +
+ +
+ + #{{document.archive_serial_number}} +
+
+ + {{document.created | customDate:'mediumDate'}} +
+ +
+ Score: + +
+
-
\ No newline at end of file +
diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.scss b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.scss index d6be8837e..fb7c14b26 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.scss +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.scss @@ -1,11 +1,67 @@ +@import "/src/theme"; + .result-content { - color: darkgray; + overflow-wrap: anywhere; } .doc-img { object-fit: cover; - object-position: top; + object-position: top left; height: 100%; position: absolute; + mix-blend-mode: multiply; +} -} \ No newline at end of file +.card-title { + word-break: break-word; +} + +.search-score-bar { + width: 100px; + height: 5px; + margin-top: 1px; +} + +.document-card-check { + display: none +} + +.document-card:hover .document-card-check { + display: block; +} + +.card-selected { + border-color: $primary; +} + +.doc-img-background-selected { + background-color: $primaryFaded; +} + +.card-info { + line-height: 1; + + button { + line-height: 1; + + &:hover, + &:focus { + background-color: transparent !important; + } + } + + .metadata-icon { + width: 0.9rem; + height: 0.9rem; + padding: 0.05rem; + } + + .search-score { + padding-top: 0.35rem !important; + } +} + +span ::ng-deep .match { + color: black; + background-color: rgb(255, 211, 66); +} diff --git a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts index ac2fdba27..d8f29ef5a 100644 --- a/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts +++ b/src-ui/src/app/components/document-list/document-card-large/document-card-large.component.ts @@ -1,44 +1,68 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core'; import { DomSanitizer } from '@angular/platform-browser'; import { PaperlessDocument } from 'src/app/data/paperless-document'; -import { PaperlessTag } from 'src/app/data/paperless-tag'; import { DocumentService } from 'src/app/services/rest/document.service'; +import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; +import { NgbPopover } from '@ng-bootstrap/ng-bootstrap'; +import { DocumentListViewService } from 'src/app/services/document-list-view.service'; +import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type'; @Component({ selector: 'app-document-card-large', templateUrl: './document-card-large.component.html', - styleUrls: ['./document-card-large.component.scss'] + styleUrls: ['./document-card-large.component.scss', '../popover-preview/popover-preview.scss'] }) export class DocumentCardLargeComponent implements OnInit { - constructor(private documentService: DocumentService, private sanitizer: DomSanitizer) { } + constructor(private documentService: DocumentService, private sanitizer: DomSanitizer, private settingsService: SettingsService) { } + + @Input() + selected = false + + @Output() + toggleSelected = new EventEmitter() + + get selectable() { + return this.toggleSelected.observers.length > 0 + } @Input() document: PaperlessDocument - @Input() - details: any - @Output() clickTag = new EventEmitter() @Output() clickCorrespondent = new EventEmitter() + @Output() + clickDocumentType = new EventEmitter() + + @Output() + clickMoreLike= new EventEmitter() + + @ViewChild('popover') popover: NgbPopover + + mouseOnPreview = false + popoverHidden = true + + get searchScoreClass() { + if (this.document.__search_hit__) { + if (this.document.__search_hit__.score > 0.7) { + return "success" + } else if (this.document.__search_hit__.score > 0.3) { + return "warning" + } else { + return "danger" + } + } + } + ngOnInit(): void { } - getDetailsAsString() { - if (typeof this.details === 'string') { - return this.details.substring(0, 500) - } - } - - getDetailsAsHighlight() { - //TODO: this is not an exact typecheck, can we do better - if (this.details instanceof Array) { - return this.details - } + getIsThumbInverted() { + return this.settingsService.get(SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED) } getThumbUrl() { @@ -49,7 +73,36 @@ export class DocumentCardLargeComponent implements OnInit { return this.documentService.getDownloadUrl(this.document.id) } - getPreviewUrl() { + get previewUrl() { return this.documentService.getPreviewUrl(this.document.id) } + + mouseEnterPreview() { + this.mouseOnPreview = true + if (!this.popover.isOpen()) { + // we're going to open but hide to pre-load content during hover delay + this.popover.open() + this.popoverHidden = true + setTimeout(() => { + if (this.mouseOnPreview) { + // show popover + this.popoverHidden = false + } else { + this.popover.close() + } + }, 600); + } + } + + mouseLeavePreview() { + this.mouseOnPreview = false + } + + mouseLeaveCard() { + this.popover.close() + } + + get contentTrimmed() { + return this.document.content.substr(0, 500) + } } diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html index 86e28442c..abe9f5299 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.html @@ -1,50 +1,92 @@ -
-
-
- +
+
+
+ + +
+
+ + +
+
+
- +
+ {{moreTags}}
- + - +
diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.scss b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.scss index 0068667d0..943df03b9 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.scss +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.scss @@ -1,5 +1,69 @@ +@import "/src/theme"; + +.card-text { + font-size: 90%; +} + .doc-img { object-fit: cover; - object-position: top; - height: 200px; -} \ No newline at end of file + object-position: top left; + height: 175px; + mix-blend-mode: multiply; +} + +.document-card-check { + display: none; + position: absolute; + top: 0; + left: 0; + + .custom-control { + margin-left: 4px; + margin-right: -3px; + } +} + +.document-card:hover .document-card-check { + display: block; +} + +.card-selected { + border-color: $primary; + + .document-card-check { + display: block; + } +} + +.doc-img-background-selected { + background-color: $primaryFaded; +} + +.card-info { + line-height: 1; + + button { + line-height: 1; + + &:hover, + &:focus { + background-color: transparent !important; + color: $primary; + } + } + + .metadata-icon { + width: 0.9rem; + height: 0.9rem; + padding: 0.05rem; + } +} + +.card-footer .btn { + padding-top: .10rem; +} + +::ng-deep .tooltip-inner { + text-align: left !important; + font-size: 90%; +} diff --git a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts index d60552d4f..10c2c5d23 100644 --- a/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts +++ b/src-ui/src/app/components/document-list/document-card-small/document-card-small.component.ts @@ -1,17 +1,24 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, Input, OnInit, Output, ViewChild } from '@angular/core'; import { map } from 'rxjs/operators'; import { PaperlessDocument } from 'src/app/data/paperless-document'; -import { PaperlessTag } from 'src/app/data/paperless-tag'; import { DocumentService } from 'src/app/services/rest/document.service'; +import { SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; +import { NgbPopover } from '@ng-bootstrap/ng-bootstrap'; @Component({ selector: 'app-document-card-small', templateUrl: './document-card-small.component.html', - styleUrls: ['./document-card-small.component.scss'] + styleUrls: ['./document-card-small.component.scss', '../popover-preview/popover-preview.scss'] }) export class DocumentCardSmallComponent implements OnInit { - constructor(private documentService: DocumentService) { } + constructor(private documentService: DocumentService, private settingsService: SettingsService) { } + + @Input() + selected = false + + @Output() + toggleSelected = new EventEmitter() @Input() document: PaperlessDocument @@ -22,11 +29,23 @@ export class DocumentCardSmallComponent implements OnInit { @Output() clickCorrespondent = new EventEmitter() + @Output() + clickDocumentType = new EventEmitter() + moreTags: number = null + @ViewChild('popover') popover: NgbPopover + + mouseOnPreview = false + popoverHidden = true + ngOnInit(): void { } + getIsThumbInverted() { + return this.settingsService.get(SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED) + } + getThumbUrl() { return this.documentService.getThumbUrl(this.document.id) } @@ -35,7 +54,7 @@ export class DocumentCardSmallComponent implements OnInit { return this.documentService.getDownloadUrl(this.document.id) } - getPreviewUrl() { + get previewUrl() { return this.documentService.getPreviewUrl(this.document.id) } @@ -52,4 +71,28 @@ export class DocumentCardSmallComponent implements OnInit { ) } + mouseEnterPreview() { + this.mouseOnPreview = true + if (!this.popover.isOpen()) { + // we're going to open but hide to pre-load content during hover delay + this.popover.open() + this.popoverHidden = true + setTimeout(() => { + if (this.mouseOnPreview) { + // show popover + this.popoverHidden = false + } else { + this.popover.close() + } + }, 600); + } + } + + mouseLeavePreview() { + this.mouseOnPreview = false + } + + mouseLeaveCard() { + this.popover.close() + } } diff --git a/src-ui/src/app/components/document-list/document-list.component.html b/src-ui/src/app/components/document-list/document-list.component.html index c4fa0d4d7..1b805bb2e 100644 --- a/src-ui/src/app/components/document-list/document-list.component.html +++ b/src-ui/src/app/components/document-list/document-list.component.html @@ -1,6 +1,19 @@ -
+ +
+ + + +
+
+ +
-
-
- -
+ +
+ + - -
-
- +
+ + + -
-
-
Filter
- -
+
+ +
-

{{list.collectionSize || 0}} document(s) (filtered)

+

+ {list.collectionSize, plural, =1 {Selected {{list.selected.size}} of one document} other {Selected {{list.selected.size}} of {{list.collectionSize || 0}} documents}} + {list.collectionSize, plural, =1 {One document} other {{{list.collectionSize || 0}} documents}} (filtered) +

+ [rotate]="true" aria-label="Default pagination">
-
- - -
+ + + - - - - - - - - - - - - - - - - - - - -
ASNCorrespondentTitleDocument typeCreatedAdded
- {{d.archive_serial_number}} - - - {{(d.correspondent$ | async)?.name}} - - - {{d.title | documentTitle}} - - - - {{(d.document_type$ | async)?.name}} - - - {{d.created | date}} - - {{d.added | date}} -
+ + +
+ + +
+ + + + + + + + + + + + + + + + + + + + + + +
ASNCorrespondentTitleDocument typeCreatedAdded
+
+ + +
+
+ {{d.archive_serial_number}} + + + {{(d.correspondent$ | async)?.name}} + + + {{d.title | documentTitle}} + + + + {{(d.document_type$ | async)?.name}} + + + {{d.created | customDate}} + + {{d.added | customDate}} +
+ +
+ +
-
- -
+
diff --git a/src-ui/src/app/components/document-list/document-list.component.scss b/src-ui/src/app/components/document-list/document-list.component.scss index e69de29bb..34c8aed5e 100644 --- a/src-ui/src/app/components/document-list/document-list.component.scss +++ b/src-ui/src/app/components/document-list/document-list.component.scss @@ -0,0 +1,45 @@ +@import "/src/theme"; + +tr { + user-select: none; +} + +.table-row-selected { + background-color: $primaryFaded; +} + +$paperless-card-breakpoints: ( + 0: 2, // xs + 768px: 3, //md + 992px: 4, //lg + 1200px: 5, //xl + 1400px: 6, // xxl + 1600px: 7, + 1800px: 8, + 2000px: 9 +); + +.row-cols-paperless-cards { + @each $width, $n_cols in $paperless-card-breakpoints { + @media(min-width: $width) { + > * { + flex: 0 0 auto; + width: 100% / $n_cols; + } + } + } +} + +.dropdown-menu-right { + right: 0 !important; + left: auto !important; +} + +.sticky-top { + z-index: 990; // below main navbar + top: calc(7rem - 2px); // height of navbar (mobile) + + @media (min-width: 580px) { + top: 3.5rem; // height of navbar + } +} diff --git a/src-ui/src/app/components/document-list/document-list.component.ts b/src-ui/src/app/components/document-list/document-list.component.ts index 09e73dd96..79adb99d0 100644 --- a/src-ui/src/app/components/document-list/document-list.component.ts +++ b/src-ui/src/app/components/document-list/document-list.component.ts @@ -1,15 +1,18 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; -import { ActivatedRoute } from '@angular/router'; +import { Component, OnDestroy, OnInit, QueryList, ViewChild, ViewChildren } from '@angular/core'; +import { ActivatedRoute, Router } from '@angular/router'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; -import { cloneFilterRules, FilterRule } from 'src/app/data/filter-rule'; -import { FILTER_CORRESPONDENT, FILTER_DOCUMENT_TYPE, FILTER_HAS_TAG, FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type'; -import { SavedViewConfig } from 'src/app/data/saved-view-config'; +import { Subscription } from 'rxjs'; +import { FilterRule, isFullTextFilterRule } from 'src/app/data/filter-rule'; +import { FILTER_FULLTEXT_MORELIKE } from 'src/app/data/filter-rule-type'; +import { PaperlessDocument } from 'src/app/data/paperless-document'; +import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; +import { SortableDirective, SortEvent } from 'src/app/directives/sortable.directive'; +import { ConsumerStatusService } from 'src/app/services/consumer-status.service'; import { DocumentListViewService } from 'src/app/services/document-list-view.service'; -import { DOCUMENT_SORT_FIELDS } from 'src/app/services/rest/document.service'; -import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; -import { Toast, ToastService } from 'src/app/services/toast.service'; -import { environment } from 'src/environments/environment'; +import { DOCUMENT_SORT_FIELDS, DOCUMENT_SORT_FIELDS_FULLTEXT } from 'src/app/services/rest/document.service'; +import { SavedViewService } from 'src/app/services/rest/saved-view.service'; +import { ToastService } from 'src/app/services/toast.service'; +import { FilterEditorComponent } from './filter-editor/filter-editor.component'; import { SaveViewConfigDialogComponent } from './save-view-config-dialog/save-view-config-dialog.component'; @Component({ @@ -17,31 +20,47 @@ import { SaveViewConfigDialogComponent } from './save-view-config-dialog/save-vi templateUrl: './document-list.component.html', styleUrls: ['./document-list.component.scss'] }) -export class DocumentListComponent implements OnInit { +export class DocumentListComponent implements OnInit, OnDestroy { constructor( public list: DocumentListViewService, - public savedViewConfigService: SavedViewConfigService, + public savedViewService: SavedViewService, public route: ActivatedRoute, + private router: Router, private toastService: ToastService, - public modalService: NgbModal, - private titleService: Title) { } + private modalService: NgbModal, + private consumerStatusService: ConsumerStatusService + ) { } + + @ViewChild("filterEditor") + private filterEditor: FilterEditorComponent + + @ViewChildren(SortableDirective) headers: QueryList; displayMode = 'smallCards' // largeCards, smallCards, details - filterRules: FilterRule[] = [] - showFilter = false + unmodifiedFilterRules: FilterRule[] = [] + + private consumptionFinishedSubscription: Subscription get isFiltered() { return this.list.filterRules?.length > 0 } getTitle() { - return this.list.savedViewTitle || "Documents" + return this.list.activeSavedViewTitle || $localize`Documents` } getSortFields() { - return DOCUMENT_SORT_FIELDS + return isFullTextFilterRule(this.list.filterRules) ? DOCUMENT_SORT_FIELDS_FULLTEXT : DOCUMENT_SORT_FIELDS + } + + onSort(event: SortEvent) { + this.list.setSort(event.column, event.reverse) + } + + get isBulkEditing(): boolean { + return this.list.selected.size > 0 } saveDisplayMode() { @@ -52,94 +71,109 @@ export class DocumentListComponent implements OnInit { if (localStorage.getItem('document-list:displayMode') != null) { this.displayMode = localStorage.getItem('document-list:displayMode') } + this.consumptionFinishedSubscription = this.consumerStatusService.onDocumentConsumptionFinished().subscribe(() => { + this.list.reload() + }) this.route.paramMap.subscribe(params => { if (params.has('id')) { - this.list.savedView = this.savedViewConfigService.getConfig(params.get('id')) - this.filterRules = this.list.filterRules - this.showFilter = false - this.titleService.setTitle(`${this.list.savedView.title} - ${environment.appTitle}`) + this.savedViewService.getCached(+params.get('id')).subscribe(view => { + if (!view) { + this.router.navigate(["404"]) + return + } + this.list.activateSavedView(view) + this.list.reload() + this.unmodifiedFilterRules = view.filter_rules + }) } else { - this.list.savedView = null - this.filterRules = this.list.filterRules - this.showFilter = this.filterRules.length > 0 - this.titleService.setTitle(`Documents - ${environment.appTitle}`) + this.list.activateSavedView(null) + this.list.reload() + this.unmodifiedFilterRules = [] } - this.list.clear() - this.list.reload() }) } - applyFilterRules() { - this.list.filterRules = this.filterRules + ngOnDestroy() { + if (this.consumptionFinishedSubscription) { + this.consumptionFinishedSubscription.unsubscribe() + } } - clearFilterRules() { - this.list.filterRules = this.filterRules - this.showFilter = false - } - - loadViewConfig(config: SavedViewConfig) { - this.filterRules = cloneFilterRules(config.filterRules) - this.list.load(config) + loadViewConfig(view: PaperlessSavedView) { + this.list.loadSavedView(view) + this.list.reload() } saveViewConfig() { - this.savedViewConfigService.updateConfig(this.list.savedView) - this.toastService.showToast(Toast.make("Information", `View "${this.list.savedView.title}" saved successfully.`)) + if (this.list.activeSavedViewId != null) { + let savedView: PaperlessSavedView = { + id: this.list.activeSavedViewId, + filter_rules: this.list.filterRules, + sort_field: this.list.sortField, + sort_reverse: this.list.sortReverse + } + this.savedViewService.patch(savedView).subscribe(result => { + this.toastService.showInfo($localize`View "${this.list.activeSavedViewTitle}" saved successfully.`) + this.unmodifiedFilterRules = this.list.filterRules + }) + } } saveViewConfigAs() { let modal = this.modalService.open(SaveViewConfigDialogComponent, {backdrop: 'static'}) + modal.componentInstance.defaultName = this.filterEditor.generateFilterName() modal.componentInstance.saveClicked.subscribe(formValue => { - this.savedViewConfigService.newConfig({ - title: formValue.title, - showInDashboard: formValue.showInDashboard, - showInSideBar: formValue.showInSideBar, - filterRules: this.list.filterRules, - sortDirection: this.list.sortDirection, - sortField: this.list.sortField + modal.componentInstance.buttonsEnabled = false + let savedView: PaperlessSavedView = { + name: formValue.name, + show_on_dashboard: formValue.showOnDashboard, + show_in_sidebar: formValue.showInSideBar, + filter_rules: this.list.filterRules, + sort_reverse: this.list.sortReverse, + sort_field: this.list.sortField + } + + this.savedViewService.create(savedView).subscribe(() => { + modal.close() + this.toastService.showInfo($localize`View "${savedView.name}" created successfully.`) + }, error => { + modal.componentInstance.error = error.error + modal.componentInstance.buttonsEnabled = true }) - modal.close() }) } - filterByTag(tag_id: number) { - let filterRules = this.list.filterRules - if (filterRules.find(rule => rule.type.id == FILTER_HAS_TAG && rule.value == tag_id)) { - return - } - - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_HAS_TAG), value: tag_id}) - this.filterRules = filterRules - this.applyFilterRules() + toggleSelected(document: PaperlessDocument, event: MouseEvent): void { + if (!event.shiftKey) this.list.toggleSelected(document) + else this.list.selectRangeTo(document) } - filterByCorrespondent(correspondent_id: number) { - let filterRules = this.list.filterRules - let existing_rule = filterRules.find(rule => rule.type.id == FILTER_CORRESPONDENT) - if (existing_rule && existing_rule.value == correspondent_id) { - return - } else if (existing_rule) { - existing_rule.value = correspondent_id - } else { - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_CORRESPONDENT), value: correspondent_id}) - } - this.filterRules = filterRules - this.applyFilterRules() + clickTag(tagID: number) { + this.list.selectNone() + setTimeout(() => { + this.filterEditor.toggleTag(tagID) + }) } - filterByDocumentType(document_type_id: number) { - let filterRules = this.list.filterRules - let existing_rule = filterRules.find(rule => rule.type.id == FILTER_DOCUMENT_TYPE) - if (existing_rule && existing_rule.value == document_type_id) { - return - } else if (existing_rule) { - existing_rule.value = document_type_id - } else { - filterRules.push({type: FILTER_RULE_TYPES.find(t => t.id == FILTER_DOCUMENT_TYPE), value: document_type_id}) - } - this.filterRules = filterRules - this.applyFilterRules() + clickCorrespondent(correspondentID: number) { + this.list.selectNone() + setTimeout(() => { + this.filterEditor.toggleCorrespondent(correspondentID) + }) } + clickDocumentType(documentTypeID: number) { + this.list.selectNone() + setTimeout(() => { + this.filterEditor.toggleDocumentType(documentTypeID) + }) + } + + clickMoreLike(documentID: number) { + this.list.quickFilter([{rule_type: FILTER_FULLTEXT_MORELIKE, value: documentID.toString()}]) + } + + trackByDocumentId(index, item: PaperlessDocument) { + return item.id + } } diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html new file mode 100644 index 000000000..7267d4d7b --- /dev/null +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.html @@ -0,0 +1,61 @@ +
+
+
+
+
+ + +
+ +
+
+
+
+
+
+ + + + + +
+
+
+
+ +
+
diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.scss b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.scss new file mode 100644 index 000000000..05df7b213 --- /dev/null +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.scss @@ -0,0 +1,10 @@ +.quick-filter { + min-width: 250px; + max-height: 400px; + overflow-y: scroll; + + .selected-icon { + min-width: 1em; + min-height: 1em; + } +} diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.spec.ts b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts similarity index 100% rename from src-ui/src/app/components/filter-editor/filter-editor.component.spec.ts rename to src-ui/src/app/components/document-list/filter-editor/filter-editor.component.spec.ts diff --git a/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts new file mode 100644 index 000000000..23c3d810a --- /dev/null +++ b/src-ui/src/app/components/document-list/filter-editor/filter-editor.component.ts @@ -0,0 +1,343 @@ +import { Component, EventEmitter, Input, Output, OnInit, OnDestroy, ViewChild, ElementRef } from '@angular/core'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; +import { Subject, Subscription } from 'rxjs'; +import { debounceTime, distinctUntilChanged } from 'rxjs/operators'; +import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; +import { TagService } from 'src/app/services/rest/tag.service'; +import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; +import { FilterRule } from 'src/app/data/filter-rule'; +import { FILTER_ADDED_AFTER, FILTER_ADDED_BEFORE, FILTER_ASN, FILTER_CORRESPONDENT, FILTER_CREATED_AFTER, FILTER_CREATED_BEFORE, FILTER_DOCUMENT_TYPE, FILTER_FULLTEXT_MORELIKE, FILTER_FULLTEXT_QUERY, FILTER_HAS_ANY_TAG, FILTER_HAS_TAG, FILTER_TITLE, FILTER_TITLE_CONTENT } from 'src/app/data/filter-rule-type'; +import { FilterableDropdownSelectionModel } from '../../common/filterable-dropdown/filterable-dropdown.component'; +import { ToggleableItemState } from '../../common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; +import { DocumentService } from 'src/app/services/rest/document.service'; +import { PaperlessDocument } from 'src/app/data/paperless-document'; + +const TEXT_FILTER_TARGET_TITLE = "title" +const TEXT_FILTER_TARGET_TITLE_CONTENT = "title-content" +const TEXT_FILTER_TARGET_ASN = "asn" +const TEXT_FILTER_TARGET_FULLTEXT_QUERY = "fulltext-query" +const TEXT_FILTER_TARGET_FULLTEXT_MORELIKE = "fulltext-morelike" + +@Component({ + selector: 'app-filter-editor', + templateUrl: './filter-editor.component.html', + styleUrls: ['./filter-editor.component.scss'] +}) +export class FilterEditorComponent implements OnInit, OnDestroy { + + generateFilterName() { + if (this.filterRules.length == 1) { + let rule = this.filterRules[0] + switch(this.filterRules[0].rule_type) { + + case FILTER_CORRESPONDENT: + if (rule.value) { + return $localize`Correspondent: ${this.correspondents.find(c => c.id == +rule.value)?.name}` + } else { + return $localize`Without correspondent` + } + + case FILTER_DOCUMENT_TYPE: + if (rule.value) { + return $localize`Type: ${this.documentTypes.find(dt => dt.id == +rule.value)?.name}` + } else { + return $localize`Without document type` + } + + case FILTER_HAS_TAG: + return $localize`Tag: ${this.tags.find(t => t.id == +rule.value)?.name}` + + case FILTER_HAS_ANY_TAG: + if (rule.value == "false") { + return $localize`Without any tag` + } + + case FILTER_TITLE: + return $localize`Title: ${rule.value}` + + case FILTER_ASN: + return $localize`ASN: ${rule.value}` + } + } + + return "" + } + + constructor( + private documentTypeService: DocumentTypeService, + private tagService: TagService, + private correspondentService: CorrespondentService, + private documentService: DocumentService + ) { } + + @ViewChild("textFilterInput") + textFilterInput: ElementRef + + tags: PaperlessTag[] = [] + correspondents: PaperlessCorrespondent[] = [] + documentTypes: PaperlessDocumentType[] = [] + + _textFilter = "" + _moreLikeId: number + _moreLikeDoc: PaperlessDocument + + get textFilterTargets() { + let targets = [ + {id: TEXT_FILTER_TARGET_TITLE, name: $localize`Title`}, + {id: TEXT_FILTER_TARGET_TITLE_CONTENT, name: $localize`Title & content`}, + {id: TEXT_FILTER_TARGET_ASN, name: $localize`ASN`}, + {id: TEXT_FILTER_TARGET_FULLTEXT_QUERY, name: $localize`Advanced search`} + ] + if (this.textFilterTarget == TEXT_FILTER_TARGET_FULLTEXT_MORELIKE) { + targets.push({id: TEXT_FILTER_TARGET_FULLTEXT_MORELIKE, name: $localize`More like`}) + } + return targets + } + + textFilterTarget = TEXT_FILTER_TARGET_TITLE_CONTENT + + get textFilterTargetName() { + return this.textFilterTargets.find(t => t.id == this.textFilterTarget)?.name + } + + + tagSelectionModel = new FilterableDropdownSelectionModel() + correspondentSelectionModel = new FilterableDropdownSelectionModel() + documentTypeSelectionModel = new FilterableDropdownSelectionModel() + + dateCreatedBefore: string + dateCreatedAfter: string + dateAddedBefore: string + dateAddedAfter: string + + _unmodifiedFilterRules: FilterRule[] = [] + _filterRules: FilterRule[] = [] + + @Input() + set unmodifiedFilterRules(value: FilterRule[]) { + this._unmodifiedFilterRules = value + this.checkIfRulesHaveChanged() + } + + get unmodifiedFilterRules(): FilterRule[] { + return this._unmodifiedFilterRules + } + + @Input() + set filterRules (value: FilterRule[]) { + this._filterRules = value + + this.documentTypeSelectionModel.clear(false) + this.tagSelectionModel.clear(false) + this.correspondentSelectionModel.clear(false) + this._textFilter = null + this._moreLikeId = null + this.dateAddedBefore = null + this.dateAddedAfter = null + this.dateCreatedBefore = null + this.dateCreatedAfter = null + + value.forEach(rule => { + switch (rule.rule_type) { + case FILTER_TITLE: + this._textFilter = rule.value + this.textFilterTarget = TEXT_FILTER_TARGET_TITLE + break + case FILTER_TITLE_CONTENT: + this._textFilter = rule.value + this.textFilterTarget = TEXT_FILTER_TARGET_TITLE_CONTENT + break + case FILTER_ASN: + this._textFilter = rule.value + this.textFilterTarget = TEXT_FILTER_TARGET_ASN + break + case FILTER_FULLTEXT_QUERY: + this._textFilter = rule.value + this.textFilterTarget = TEXT_FILTER_TARGET_FULLTEXT_QUERY + break + case FILTER_FULLTEXT_MORELIKE: + this._moreLikeId = +rule.value + this.textFilterTarget = TEXT_FILTER_TARGET_FULLTEXT_MORELIKE + this.documentService.get(this._moreLikeId).subscribe(result => { + this._moreLikeDoc = result + this._textFilter = result.title + }) + break + case FILTER_CREATED_AFTER: + this.dateCreatedAfter = rule.value + break + case FILTER_CREATED_BEFORE: + this.dateCreatedBefore = rule.value + break + case FILTER_ADDED_AFTER: + this.dateAddedAfter = rule.value + break + case FILTER_ADDED_BEFORE: + this.dateAddedBefore = rule.value + break + case FILTER_HAS_TAG: + this.tagSelectionModel.set(rule.value ? +rule.value : null, ToggleableItemState.Selected, false) + break + case FILTER_HAS_ANY_TAG: + this.tagSelectionModel.set(null, ToggleableItemState.Selected, false) + break + case FILTER_CORRESPONDENT: + this.correspondentSelectionModel.set(rule.value ? +rule.value : null, ToggleableItemState.Selected, false) + break + case FILTER_DOCUMENT_TYPE: + this.documentTypeSelectionModel.set(rule.value ? +rule.value : null, ToggleableItemState.Selected, false) + break + } + }) + this.checkIfRulesHaveChanged() + } + + get filterRules(): FilterRule[] { + let filterRules: FilterRule[] = [] + if (this._textFilter && this.textFilterTarget == TEXT_FILTER_TARGET_TITLE_CONTENT) { + filterRules.push({rule_type: FILTER_TITLE_CONTENT, value: this._textFilter}) + } + if (this._textFilter && this.textFilterTarget == TEXT_FILTER_TARGET_TITLE) { + filterRules.push({rule_type: FILTER_TITLE, value: this._textFilter}) + } + if (this._textFilter && this.textFilterTarget == TEXT_FILTER_TARGET_ASN) { + filterRules.push({rule_type: FILTER_ASN, value: this._textFilter}) + } + if (this._textFilter && this.textFilterTarget == TEXT_FILTER_TARGET_FULLTEXT_QUERY) { + filterRules.push({rule_type: FILTER_FULLTEXT_QUERY, value: this._textFilter}) + } + if (this._moreLikeId && this.textFilterTarget == TEXT_FILTER_TARGET_FULLTEXT_MORELIKE) { + filterRules.push({rule_type: FILTER_FULLTEXT_MORELIKE, value: this._moreLikeId?.toString()}) + } + if (this.tagSelectionModel.isNoneSelected()) { + filterRules.push({rule_type: FILTER_HAS_ANY_TAG, value: "false"}) + } else { + this.tagSelectionModel.getSelectedItems().filter(tag => tag.id).forEach(tag => { + filterRules.push({rule_type: FILTER_HAS_TAG, value: tag.id?.toString()}) + }) + } + this.correspondentSelectionModel.getSelectedItems().forEach(correspondent => { + filterRules.push({rule_type: FILTER_CORRESPONDENT, value: correspondent.id?.toString()}) + }) + this.documentTypeSelectionModel.getSelectedItems().forEach(documentType => { + filterRules.push({rule_type: FILTER_DOCUMENT_TYPE, value: documentType.id?.toString()}) + }) + if (this.dateCreatedBefore) { + filterRules.push({rule_type: FILTER_CREATED_BEFORE, value: this.dateCreatedBefore}) + } + if (this.dateCreatedAfter) { + filterRules.push({rule_type: FILTER_CREATED_AFTER, value: this.dateCreatedAfter}) + } + if (this.dateAddedBefore) { + filterRules.push({rule_type: FILTER_ADDED_BEFORE, value: this.dateAddedBefore}) + } + if (this.dateAddedAfter) { + filterRules.push({rule_type: FILTER_ADDED_AFTER, value: this.dateAddedAfter}) + } + return filterRules + } + + @Output() + filterRulesChange = new EventEmitter() + + rulesModified: boolean = false + + private checkIfRulesHaveChanged() { + let modified = false + if (this._unmodifiedFilterRules.length != this._filterRules.length) { + modified = true + } else { + modified = this._unmodifiedFilterRules.some(rule => { + return (this._filterRules.find(fri => fri.rule_type == rule.rule_type && fri.value == rule.value) == undefined) + }) + + if (!modified) { + // only check other direction if we havent already determined is modified + modified = this._filterRules.some(rule => { + this._unmodifiedFilterRules.find(fr => fr.rule_type == rule.rule_type && fr.value == rule.value) == undefined + }) + } + } + this.rulesModified = modified + } + + updateRules() { + this.filterRulesChange.next(this.filterRules) + } + + get textFilter() { + return this._textFilter + } + + set textFilter(value) { + this.textFilterDebounce.next(value) + } + + textFilterDebounce: Subject + subscription: Subscription + + ngOnInit() { + this.tagService.listAll().subscribe(result => this.tags = result.results) + this.correspondentService.listAll().subscribe(result => this.correspondents = result.results) + this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results) + + this.textFilterDebounce = new Subject() + + this.subscription = this.textFilterDebounce.pipe( + debounceTime(400), + distinctUntilChanged() + ).subscribe(text => { + this._textFilter = text + this.documentService.searchQuery = text + this.updateRules() + }) + + if (this._textFilter) this.documentService.searchQuery = this._textFilter + + } + + ngOnDestroy() { + this.textFilterDebounce.complete() + } + + resetSelected() { + this.textFilterTarget = TEXT_FILTER_TARGET_TITLE_CONTENT + this.filterRules = this._unmodifiedFilterRules + this.updateRules() + } + + toggleTag(tagId: number) { + this.tagSelectionModel.toggle(tagId) + } + + toggleCorrespondent(correspondentId: number) { + this.correspondentSelectionModel.toggle(correspondentId) + } + + toggleDocumentType(documentTypeId: number) { + this.documentTypeSelectionModel.toggle(documentTypeId) + } + + onTagsDropdownOpen() { + this.tagSelectionModel.apply() + } + + onCorrespondentDropdownOpen() { + this.correspondentSelectionModel.apply() + } + + onDocumentTypeDropdownOpen() { + this.documentTypeSelectionModel.apply() + } + + changeTextFilterTarget(target) { + if (this.textFilterTarget == TEXT_FILTER_TARGET_FULLTEXT_MORELIKE && target != TEXT_FILTER_TARGET_FULLTEXT_MORELIKE) { + this._textFilter = "" + } + this.textFilterTarget = target + this.textFilterInput.nativeElement.focus() + this.updateRules() + } +} diff --git a/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss b/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss new file mode 100644 index 000000000..d232930c2 --- /dev/null +++ b/src-ui/src/app/components/document-list/popover-preview/popover-preview.scss @@ -0,0 +1,22 @@ +::ng-deep .popover { + max-width: 40rem; + + .preview { + min-width: 30rem; + min-height: 18rem; + max-height: 35rem; + overflow-y: scroll; + } + + .spinner-border { + position: absolute; + top: 4rem; + left: calc(50% - 0.5rem); + z-index: 0; + } +} + + ::ng-deep .popover-hidden .popover { + opacity: 0; + pointer-events: none; +} diff --git a/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html b/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html index 870431096..7f1d39859 100644 --- a/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html +++ b/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html @@ -1,17 +1,17 @@ -
+
diff --git a/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.ts b/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.ts index 0dd351770..4f70941dd 100644 --- a/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.ts +++ b/src-ui/src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.ts @@ -1,4 +1,4 @@ -import { Component, EventEmitter, OnInit, Output } from '@angular/core'; +import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; @@ -14,13 +14,37 @@ export class SaveViewConfigDialogComponent implements OnInit { @Output() public saveClicked = new EventEmitter() + @Input() + error + + @Input() + buttonsEnabled = true + + closeEnabled = false + + _defaultName = "" + + get defaultName() { + return this._defaultName + } + + @Input() + set defaultName(value: string) { + this._defaultName = value + this.saveViewConfigForm.patchValue({name: value}) + } + saveViewConfigForm = new FormGroup({ - title: new FormControl(''), + name: new FormControl(''), showInSideBar: new FormControl(false), - showInDashboard: new FormControl(false), + showOnDashboard: new FormControl(false), }) ngOnInit(): void { + // wait to enable close button so it doesnt steal focus from input since its the first clickable element in the DOM + setTimeout(() => { + this.closeEnabled = true + }); } save() { diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.html b/src-ui/src/app/components/filter-editor/filter-editor.component.html deleted file mode 100644 index 48780e950..000000000 --- a/src-ui/src/app/components/filter-editor/filter-editor.component.html +++ /dev/null @@ -1,52 +0,0 @@ -
-
- {{rule.type.name}} -
-
- - - - - - - - - - - - -
-
- -
-
- -
-
- -
-
- -
-
- -
-
- -
-
diff --git a/src-ui/src/app/components/filter-editor/filter-editor.component.ts b/src-ui/src/app/components/filter-editor/filter-editor.component.ts deleted file mode 100644 index b04127287..000000000 --- a/src-ui/src/app/components/filter-editor/filter-editor.component.ts +++ /dev/null @@ -1,67 +0,0 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; -import { FilterRule } from 'src/app/data/filter-rule'; -import { FilterRuleType, FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type'; -import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; -import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; -import { PaperlessTag } from 'src/app/data/paperless-tag'; -import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; -import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; -import { TagService } from 'src/app/services/rest/tag.service'; - - -@Component({ - selector: 'app-filter-editor', - templateUrl: './filter-editor.component.html', - styleUrls: ['./filter-editor.component.scss'] -}) -export class FilterEditorComponent implements OnInit { - - constructor(private documentTypeService: DocumentTypeService, private tagService: TagService, private correspondentService: CorrespondentService) { } - - @Output() - clear = new EventEmitter() - - @Input() - filterRules: FilterRule[] = [] - - @Output() - apply = new EventEmitter() - - selectedRuleType: FilterRuleType = FILTER_RULE_TYPES[0] - - correspondents: PaperlessCorrespondent[] = [] - tags: PaperlessTag[] = [] - documentTypes: PaperlessDocumentType[] = [] - - newRuleClicked() { - this.filterRules.push({type: this.selectedRuleType, value: this.selectedRuleType.default}) - this.selectedRuleType = this.getRuleTypes().length > 0 ? this.getRuleTypes()[0] : null - } - - removeRuleClicked(rule) { - let index = this.filterRules.findIndex(r => r == rule) - if (index > -1) { - this.filterRules.splice(index, 1) - } - } - - applyClicked() { - this.apply.next() - } - - clearClicked() { - this.filterRules.splice(0,this.filterRules.length) - this.clear.next() - } - - ngOnInit(): void { - this.correspondentService.listAll().subscribe(result => {this.correspondents = result.results}) - this.tagService.listAll().subscribe(result => this.tags = result.results) - this.documentTypeService.listAll().subscribe(result => this.documentTypes = result.results) - } - - getRuleTypes() { - return FILTER_RULE_TYPES.filter(rt => rt.multi || !this.filterRules.find(r => r.type == rt)) - } - -} diff --git a/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html b/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html index 307c78c3c..26a6dc219 100644 --- a/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html +++ b/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html @@ -1,20 +1,18 @@ -
+ -
\ No newline at end of file + diff --git a/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts b/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts index 855fc159c..b2fe8a929 100644 --- a/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts +++ b/src-ui/src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts @@ -1,4 +1,4 @@ -import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core'; +import { Component } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'; @@ -14,7 +14,15 @@ import { ToastService } from 'src/app/services/toast.service'; export class CorrespondentEditDialogComponent extends EditDialogComponent { constructor(service: CorrespondentService, activeModal: NgbActiveModal, toastService: ToastService) { - super(service, activeModal, toastService, 'correspondent') + super(service, activeModal, toastService) + } + + getCreateTitle() { + return $localize`Create new correspondent` + } + + getEditTitle() { + return $localize`Edit correspondent` } getForm(): FormGroup { @@ -24,6 +32,6 @@ export class CorrespondentEditDialogComponent extends EditDialogComponent - + + -
- +
+
+
+ + +
+
+ +
- +
- - - - - + + + + + @@ -23,12 +28,26 @@ - + diff --git a/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts b/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts index 11027c60f..fc057e87a 100644 --- a/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts +++ b/src-ui/src/app/components/manage/correspondent-list/correspondent-list.component.ts @@ -1,9 +1,10 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; +import { Component } from '@angular/core'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { FILTER_CORRESPONDENT } from 'src/app/data/filter-rule-type'; import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; +import { DocumentListViewService } from 'src/app/services/document-list-view.service'; import { CorrespondentService } from 'src/app/services/rest/correspondent.service'; -import { environment } from 'src/environments/environment'; +import { ToastService } from 'src/app/services/toast.service'; import { GenericListComponent } from '../generic-list/generic-list.component'; import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog/correspondent-edit-dialog.component'; @@ -12,19 +13,20 @@ import { CorrespondentEditDialogComponent } from './correspondent-edit-dialog/co templateUrl: './correspondent-list.component.html', styleUrls: ['./correspondent-list.component.scss'] }) -export class CorrespondentListComponent extends GenericListComponent implements OnInit { +export class CorrespondentListComponent extends GenericListComponent { - constructor(correspondentsService: CorrespondentService, modalService: NgbModal, private titleService: Title) { - super(correspondentsService,modalService,CorrespondentEditDialogComponent) + constructor(correspondentsService: CorrespondentService, modalService: NgbModal, + private list: DocumentListViewService, + toastService: ToastService + ) { + super(correspondentsService,modalService,CorrespondentEditDialogComponent, toastService) } - getObjectName(object: PaperlessCorrespondent) { - return `correspondent '${object.name}'` + getDeleteMessage(object: PaperlessCorrespondent) { + return $localize`Do you really want to delete the correspondent "${object.name}"?` } - ngOnInit(): void { - super.ngOnInit() - this.titleService.setTitle(`Correspondents - ${environment.appTitle}`) + filterDocuments(object: PaperlessCorrespondent) { + this.list.quickFilter([{rule_type: FILTER_CORRESPONDENT, value: object.id.toString()}]) } - } diff --git a/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.html b/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.html index 013c5a947..55cd3de5b 100644 --- a/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.html +++ b/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.html @@ -1,20 +1,20 @@ -
+ - \ No newline at end of file + diff --git a/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts b/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts index 087eede8c..b4a15d303 100644 --- a/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts +++ b/src-ui/src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts @@ -1,4 +1,4 @@ -import { Component, OnInit } from '@angular/core'; +import { Component } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'; @@ -13,8 +13,16 @@ import { ToastService } from 'src/app/services/toast.service'; }) export class DocumentTypeEditDialogComponent extends EditDialogComponent { - constructor(service: DocumentTypeService, activeModal: NgbActiveModal, toastService: ToastService) { - super(service, activeModal, toastService, 'document type') + constructor(service: DocumentTypeService, activeModal: NgbActiveModal, toastService: ToastService) { + super(service, activeModal, toastService) + } + + getCreateTitle() { + return $localize`Create new document type` + } + + getEditTitle() { + return $localize`Edit document type` } getForm(): FormGroup { diff --git a/src-ui/src/app/components/manage/document-type-list/document-type-list.component.html b/src-ui/src/app/components/manage/document-type-list/document-type-list.component.html index 78c86daf3..4d9bedcb1 100644 --- a/src-ui/src/app/components/manage/document-type-list/document-type-list.component.html +++ b/src-ui/src/app/components/manage/document-type-list/document-type-list.component.html @@ -1,21 +1,25 @@ - - + + -
- +
+
+
+ + +
+
+ +
-
NameMatchingDocument countLast correspondenceActionsNameMatchingDocument countLast correspondenceActions
{{ correspondent.name }} {{ getMatching(correspondent) }} {{ correspondent.document_count }}{{ correspondent.last_correspondence | date }}{{ correspondent.last_correspondence | customDate }}
- - -
+ + + +
+
- - - - + + + + @@ -25,10 +29,24 @@ -
NameMatchingDocument countActionsNameMatchingDocument countActions
{{ document_type.document_count }}
- - + + +
\ No newline at end of file + diff --git a/src-ui/src/app/components/manage/document-type-list/document-type-list.component.ts b/src-ui/src/app/components/manage/document-type-list/document-type-list.component.ts index 316024514..02b9231ee 100644 --- a/src-ui/src/app/components/manage/document-type-list/document-type-list.component.ts +++ b/src-ui/src/app/components/manage/document-type-list/document-type-list.component.ts @@ -1,9 +1,10 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; +import { Component } from '@angular/core'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { FILTER_DOCUMENT_TYPE } from 'src/app/data/filter-rule-type'; import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; +import { DocumentListViewService } from 'src/app/services/document-list-view.service'; import { DocumentTypeService } from 'src/app/services/rest/document-type.service'; -import { environment } from 'src/environments/environment'; +import { ToastService } from 'src/app/services/toast.service'; import { GenericListComponent } from '../generic-list/generic-list.component'; import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog/document-type-edit-dialog.component'; @@ -12,18 +13,21 @@ import { DocumentTypeEditDialogComponent } from './document-type-edit-dialog/doc templateUrl: './document-type-list.component.html', styleUrls: ['./document-type-list.component.scss'] }) -export class DocumentTypeListComponent extends GenericListComponent implements OnInit { +export class DocumentTypeListComponent extends GenericListComponent { - constructor(service: DocumentTypeService, modalService: NgbModal, private titleService: Title) { - super(service, modalService, DocumentTypeEditDialogComponent) + constructor(service: DocumentTypeService, modalService: NgbModal, + private list: DocumentListViewService, + toastService: ToastService + ) { + super(service, modalService, DocumentTypeEditDialogComponent, toastService) } - getObjectName(object: PaperlessDocumentType) { - return `document type '${object.name}'` + getDeleteMessage(object: PaperlessDocumentType) { + return $localize`Do you really want to delete the document type "${object.name}"?` } - ngOnInit(): void { - super.ngOnInit() - this.titleService.setTitle(`Document types - ${environment.appTitle}`) + + filterDocuments(object: PaperlessDocumentType) { + this.list.quickFilter([{rule_type: FILTER_DOCUMENT_TYPE, value: object.id.toString()}]) } } diff --git a/src-ui/src/app/components/manage/generic-list/generic-list.component.ts b/src-ui/src/app/components/manage/generic-list/generic-list.component.ts index d5477d010..7c5dbc8e3 100644 --- a/src-ui/src/app/components/manage/generic-list/generic-list.component.ts +++ b/src-ui/src/app/components/manage/generic-list/generic-list.component.ts @@ -1,18 +1,22 @@ -import { Directive, OnInit, QueryList, ViewChildren } from '@angular/core'; +import { Directive, OnDestroy, OnInit, QueryList, ViewChildren } from '@angular/core'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; +import { Subject, Subscription } from 'rxjs'; +import { debounceTime, distinctUntilChanged } from 'rxjs/operators'; import { MatchingModel, MATCHING_ALGORITHMS, MATCH_AUTO } from 'src/app/data/matching-model'; import { ObjectWithId } from 'src/app/data/object-with-id'; import { SortableDirective, SortEvent } from 'src/app/directives/sortable.directive'; -import { AbstractPaperlessService } from 'src/app/services/rest/abstract-paperless-service'; -import { DeleteDialogComponent } from '../../common/delete-dialog/delete-dialog.component'; +import { AbstractNameFilterService } from 'src/app/services/rest/abstract-name-filter-service'; +import { ToastService } from 'src/app/services/toast.service'; +import { ConfirmDialogComponent } from '../../common/confirm-dialog/confirm-dialog.component'; @Directive() -export abstract class GenericListComponent implements OnInit { - +export abstract class GenericListComponent implements OnInit, OnDestroy { + constructor( - private service: AbstractPaperlessService, + private service: AbstractNameFilterService, private modalService: NgbModal, - private editDialogComponent: any) { + private editDialogComponent: any, + private toastService: ToastService) { } @ViewChildren(SortableDirective) headers: QueryList; @@ -24,43 +28,49 @@ export abstract class GenericListComponent implements On public collectionSize = 0 public sortField: string - public sortDirection: string + public sortReverse: boolean + + private nameFilterDebounce: Subject + private subscription: Subscription + private _nameFilter: string getMatching(o: MatchingModel) { if (o.matching_algorithm == MATCH_AUTO) { - return "Automatic" + return $localize`Automatic` } else if (o.match && o.match.length > 0) { - return `${o.match} (${MATCHING_ALGORITHMS.find(a => a.id == o.matching_algorithm).name})` + return `${MATCHING_ALGORITHMS.find(a => a.id == o.matching_algorithm).shortName}: ${o.match}` } else { return "-" } } onSort(event: SortEvent) { - - if (event.direction && event.direction.length > 0) { - this.sortField = event.column - this.sortDirection = event.direction - } else { - this.sortField = null - this.sortDirection = null - } - - this.headers.forEach(header => { - if (header.sortable !== this.sortField) { - header.direction = ''; - } - }); - + this.sortField = event.column + this.sortReverse = event.reverse this.reloadData() } + ngOnInit(): void { this.reloadData() + + this.nameFilterDebounce = new Subject() + + this.subscription = this.nameFilterDebounce.pipe( + debounceTime(400), + distinctUntilChanged() + ).subscribe(title => { + this._nameFilter = title + this.reloadData() + }) + } + + ngOnDestroy() { + this.subscription.unsubscribe() } reloadData() { - this.service.list(this.page, null, this.sortField, this.sortDirection).subscribe(c => { + this.service.listFiltered(this.page, null, this.sortField, this.sortReverse, this._nameFilter).subscribe(c => { this.data = c.results this.collectionSize = c.count }); @@ -83,20 +93,35 @@ export abstract class GenericListComponent implements On }) } - getObjectName(object: T) { - return object.toString() + getDeleteMessage(object: T) { + return $localize`Do you really want to delete this element?` } openDeleteDialog(object: T) { - var activeModal = this.modalService.open(DeleteDialogComponent, {backdrop: 'static'}) - activeModal.componentInstance.message = `Do you really want to delete ${this.getObjectName(object)}?` - activeModal.componentInstance.message2 = "Associated documents will not be deleted." - activeModal.componentInstance.deleteClicked.subscribe(() => { + var activeModal = this.modalService.open(ConfirmDialogComponent, {backdrop: 'static'}) + activeModal.componentInstance.title = $localize`Confirm delete` + activeModal.componentInstance.messageBold = this.getDeleteMessage(object) + activeModal.componentInstance.message = $localize`Associated documents will not be deleted.` + activeModal.componentInstance.btnClass = "btn-danger" + activeModal.componentInstance.btnCaption = $localize`Delete` + activeModal.componentInstance.confirmClicked.subscribe(() => { + activeModal.componentInstance.buttonsEnabled = false this.service.delete(object).subscribe(_ => { activeModal.close() this.reloadData() + }, error => { + activeModal.componentInstance.buttonsEnabled = true + this.toastService.showError($localize`Error while deleting element: ${JSON.stringify(error.error)}`) }) } ) } + + get nameFilter() { + return this._nameFilter + } + + set nameFilter(nameFilter: string) { + this.nameFilterDebounce.next(nameFilter) + } } diff --git a/src-ui/src/app/components/manage/logs/logs.component.html b/src-ui/src/app/components/manage/logs/logs.component.html index 6af482c66..0633daf32 100644 --- a/src-ui/src/app/components/manage/logs/logs.component.html +++ b/src-ui/src/app/components/manage/logs/logs.component.html @@ -1,26 +1,18 @@ - - -
- -
- -
-
+ -
diff --git a/src-ui/src/app/components/manage/logs/logs.component.scss b/src-ui/src/app/components/manage/logs/logs.component.scss index dee9b10dc..834c8c1cb 100644 --- a/src-ui/src/app/components/manage/logs/logs.component.scss +++ b/src-ui/src/app/components/manage/logs/logs.component.scss @@ -13,4 +13,14 @@ .log-entry-50 { color: lightcoral !important; font-weight: bold; -} \ No newline at end of file +} + +.log-container { + overflow-y: scroll; + height: calc(100vh - 200px); + top: 70px; + + p { + white-space: pre-wrap; + } +} diff --git a/src-ui/src/app/components/manage/logs/logs.component.ts b/src-ui/src/app/components/manage/logs/logs.component.ts index 44d0fa24d..26ceb5c57 100644 --- a/src-ui/src/app/components/manage/logs/logs.component.ts +++ b/src-ui/src/app/components/manage/logs/logs.component.ts @@ -1,51 +1,65 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; -import { LOG_LEVELS, LOG_LEVEL_INFO, PaperlessLog } from 'src/app/data/paperless-log'; +import { Component, ElementRef, OnInit, AfterViewChecked, ViewChild } from '@angular/core'; import { LogService } from 'src/app/services/rest/log.service'; -import { environment } from 'src/environments/environment'; @Component({ selector: 'app-logs', templateUrl: './logs.component.html', styleUrls: ['./logs.component.scss'] }) -export class LogsComponent implements OnInit { +export class LogsComponent implements OnInit, AfterViewChecked { - constructor(private logService: LogService, private titleService: Title) { } + constructor(private logService: LogService) { } - logs: PaperlessLog[] = [] - level: number = LOG_LEVEL_INFO + logs: string[] = [] + + logFiles: string[] = [] + + activeLog: string + + @ViewChild('logContainer') logContainer: ElementRef ngOnInit(): void { - this.reload() - this.titleService.setTitle(`Logs - ${environment.appTitle}`) - } - - reload() { - this.logService.list(1, 50, 'created', 'des', {'level__gte': this.level}).subscribe(result => this.logs = result.results) - } - - getLevelText(level: number) { - return LOG_LEVELS.find(l => l.id == level)?.name - } - - onScroll() { - let lastCreated = null - if (this.logs.length > 0) { - lastCreated = new Date(this.logs[this.logs.length-1].created).toISOString() - } - this.logService.list(1, 25, 'created', 'des', {'created__lt': lastCreated, 'level__gte': this.level}).subscribe(result => { - this.logs.push(...result.results) + this.logService.list().subscribe(result => { + this.logFiles = result + if (this.logFiles.length > 0) { + this.activeLog = this.logFiles[0] + this.reloadLogs() + } }) } - getLevels() { - return LOG_LEVELS + ngAfterViewChecked() { + this.scrollToBottom(); } - setLevel(id) { - this.level = id - this.reload() + reloadLogs() { + this.logService.get(this.activeLog).subscribe(result => { + this.logs = result + }, error => { + this.logs = [] + }) + } + + getLogLevel(log: string) { + if (log.indexOf("[DEBUG]") != -1) { + return 10 + } else if (log.indexOf("[WARNING]") != -1) { + return 30 + } else if (log.indexOf("[ERROR]") != -1) { + return 40 + } else if (log.indexOf("[CRITICAL]") != -1) { + return 50 + } else { + return 20 + } + } + + scrollToBottom(): void { + this.logContainer?.nativeElement.scroll({ + top: this.logContainer.nativeElement.scrollHeight, + left: 0, + behavior: 'auto' + }); } } diff --git a/src-ui/src/app/components/manage/settings/settings.component.html b/src-ui/src/app/components/manage/settings/settings.component.html index 7a500e6eb..c2a32906e 100644 --- a/src-ui/src/app/components/manage/settings/settings.component.html +++ b/src-ui/src/app/components/manage/settings/settings.component.html @@ -1,4 +1,4 @@ - + @@ -7,57 +7,168 @@ -
+
- - \ No newline at end of file + + diff --git a/src-ui/src/app/components/manage/settings/settings.component.ts b/src-ui/src/app/components/manage/settings/settings.component.ts index c7b976c65..ab71b1028 100644 --- a/src-ui/src/app/components/manage/settings/settings.component.ts +++ b/src-ui/src/app/components/manage/settings/settings.component.ts @@ -1,11 +1,10 @@ -import { Component, OnInit } from '@angular/core'; +import { Component, Inject, LOCALE_ID, OnInit, Renderer2 } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; -import { Title } from '@angular/platform-browser'; -import { SavedViewConfig } from 'src/app/data/saved-view-config'; -import { GENERAL_SETTINGS } from 'src/app/data/storage-keys'; +import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; import { DocumentListViewService } from 'src/app/services/document-list-view.service'; -import { SavedViewConfigService } from 'src/app/services/saved-view-config.service'; -import { environment } from 'src/environments/environment'; +import { SavedViewService } from 'src/app/services/rest/saved-view.service'; +import { LanguageOption, SettingsService, SETTINGS_KEYS } from 'src/app/services/settings.service'; +import { ToastService } from 'src/app/services/toast.service'; @Component({ selector: 'app-settings', @@ -14,26 +13,112 @@ import { environment } from 'src/environments/environment'; }) export class SettingsComponent implements OnInit { + savedViewGroup = new FormGroup({}) + settingsForm = new FormGroup({ - 'documentListItemPerPage': new FormControl(+localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT) + 'bulkEditConfirmationDialogs': new FormControl(this.settings.get(SETTINGS_KEYS.BULK_EDIT_CONFIRMATION_DIALOGS)), + 'bulkEditApplyOnClose': new FormControl(this.settings.get(SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE)), + 'documentListItemPerPage': new FormControl(this.settings.get(SETTINGS_KEYS.DOCUMENT_LIST_SIZE)), + 'darkModeUseSystem': new FormControl(this.settings.get(SETTINGS_KEYS.DARK_MODE_USE_SYSTEM)), + 'darkModeEnabled': new FormControl(this.settings.get(SETTINGS_KEYS.DARK_MODE_ENABLED)), + 'darkModeInvertThumbs': new FormControl(this.settings.get(SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED)), + 'useNativePdfViewer': new FormControl(this.settings.get(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER)), + 'savedViews': this.savedViewGroup, + 'displayLanguage': new FormControl(this.settings.getLanguage()), + 'dateLocale': new FormControl(this.settings.get(SETTINGS_KEYS.DATE_LOCALE)), + 'dateFormat': new FormControl(this.settings.get(SETTINGS_KEYS.DATE_FORMAT)), + 'notificationsConsumerNewDocument': new FormControl(this.settings.get(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_NEW_DOCUMENT)), + 'notificationsConsumerSuccess': new FormControl(this.settings.get(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUCCESS)), + 'notificationsConsumerFailed': new FormControl(this.settings.get(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_FAILED)), + 'notificationsConsumerSuppressOnDashboard': new FormControl(this.settings.get(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD)), }) - constructor( - private savedViewConfigService: SavedViewConfigService, - private documentListViewService: DocumentListViewService, - private titleService: Title - ) { } + savedViews: PaperlessSavedView[] - ngOnInit(): void { - this.titleService.setTitle(`Settings - ${environment.appTitle}`) + get computedDateLocale(): string { + return this.settingsForm.value.dateLocale || this.settingsForm.value.displayLanguage || this.currentLocale } - deleteViewConfig(config: SavedViewConfig) { - this.savedViewConfigService.deleteConfig(config) + constructor( + public savedViewService: SavedViewService, + private documentListViewService: DocumentListViewService, + private toastService: ToastService, + private settings: SettingsService, + @Inject(LOCALE_ID) public currentLocale: string + ) { } + + ngOnInit() { + this.savedViewService.listAll().subscribe(r => { + this.savedViews = r.results + for (let view of this.savedViews) { + this.savedViewGroup.addControl(view.id.toString(), new FormGroup({ + "id": new FormControl(view.id), + "name": new FormControl(view.name), + "show_on_dashboard": new FormControl(view.show_on_dashboard), + "show_in_sidebar": new FormControl(view.show_in_sidebar) + })) + } + }) + } + + deleteSavedView(savedView: PaperlessSavedView) { + this.savedViewService.delete(savedView).subscribe(() => { + this.savedViewGroup.removeControl(savedView.id.toString()) + this.savedViews.splice(this.savedViews.indexOf(savedView), 1) + this.toastService.showInfo($localize`Saved view "${savedView.name}" deleted.`) + }) + } + + private saveLocalSettings() { + this.settings.set(SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE, this.settingsForm.value.bulkEditApplyOnClose) + this.settings.set(SETTINGS_KEYS.BULK_EDIT_CONFIRMATION_DIALOGS, this.settingsForm.value.bulkEditConfirmationDialogs) + this.settings.set(SETTINGS_KEYS.DOCUMENT_LIST_SIZE, this.settingsForm.value.documentListItemPerPage) + this.settings.set(SETTINGS_KEYS.DARK_MODE_USE_SYSTEM, this.settingsForm.value.darkModeUseSystem) + this.settings.set(SETTINGS_KEYS.DARK_MODE_ENABLED, (this.settingsForm.value.darkModeEnabled == true).toString()) + this.settings.set(SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED, (this.settingsForm.value.darkModeInvertThumbs == true).toString()) + this.settings.set(SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, this.settingsForm.value.useNativePdfViewer) + this.settings.set(SETTINGS_KEYS.DATE_LOCALE, this.settingsForm.value.dateLocale) + this.settings.set(SETTINGS_KEYS.DATE_FORMAT, this.settingsForm.value.dateFormat) + this.settings.set(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_NEW_DOCUMENT, this.settingsForm.value.notificationsConsumerNewDocument) + this.settings.set(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUCCESS, this.settingsForm.value.notificationsConsumerSuccess) + this.settings.set(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_FAILED, this.settingsForm.value.notificationsConsumerFailed) + this.settings.set(SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD, this.settingsForm.value.notificationsConsumerSuppressOnDashboard) + this.settings.setLanguage(this.settingsForm.value.displayLanguage) + this.documentListViewService.updatePageSize() + this.settings.updateDarkModeSettings() + this.toastService.showInfo($localize`Settings saved successfully.`) + } + + get displayLanguageOptions(): LanguageOption[] { + return [ + {code: "", name: $localize`Use system language`} + ].concat(this.settings.getLanguageOptions()) + } + + get dateLocaleOptions(): LanguageOption[] { + return [ + {code: "", name: $localize`Use date format of display language`} + ].concat(this.settings.getDateLocaleOptions()) + } + + get today() { + return new Date() } saveSettings() { - localStorage.setItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE, this.settingsForm.value.documentListItemPerPage) - this.documentListViewService.updatePageSize() + let x = [] + for (let id in this.savedViewGroup.value) { + x.push(this.savedViewGroup.value[id]) + } + if (x.length > 0) { + this.savedViewService.patchMany(x).subscribe(s => { + this.saveLocalSettings() + }, error => { + this.toastService.showError($localize`Error while storing settings on server: ${JSON.stringify(error.error)}`) + }) + } else { + this.saveLocalSettings() + } + } } diff --git a/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html b/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html index 8048b0c80..d9db9283a 100644 --- a/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html +++ b/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html @@ -1,20 +1,22 @@ -
+
diff --git a/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts b/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts index bb0162608..623b033ec 100644 --- a/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts +++ b/src-ui/src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts @@ -2,9 +2,10 @@ import { Component } from '@angular/core'; import { FormControl, FormGroup } from '@angular/forms'; import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'; import { EditDialogComponent } from 'src/app/components/common/edit-dialog/edit-dialog.component'; -import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; import { TagService } from 'src/app/services/rest/tag.service'; import { ToastService } from 'src/app/services/toast.service'; +import { randomColor } from 'src/app/utils/color'; @Component({ selector: 'app-tag-edit-dialog', @@ -13,14 +14,22 @@ import { ToastService } from 'src/app/services/toast.service'; }) export class TagEditDialogComponent extends EditDialogComponent { - constructor(service: TagService, activeModal: NgbActiveModal, toastService: ToastService) { - super(service, activeModal, toastService, 'tag') + constructor(service: TagService, activeModal: NgbActiveModal, toastService: ToastService) { + super(service, activeModal, toastService) + } + + getCreateTitle() { + return $localize`Create new tag` + } + + getEditTitle() { + return $localize`Edit tag` } getForm(): FormGroup { return new FormGroup({ name: new FormControl(''), - colour: new FormControl(1), + color: new FormControl(randomColor()), is_inbox_tag: new FormControl(false), matching_algorithm: new FormControl(1), match: new FormControl(""), @@ -28,12 +37,4 @@ export class TagEditDialogComponent extends EditDialogComponent { }) } - getColours() { - return TAG_COLOURS - } - - getColor(id: number) { - return TAG_COLOURS.find(c => c.id == id) - } - } diff --git a/src-ui/src/app/components/manage/tag-list/tag-list.component.html b/src-ui/src/app/components/manage/tag-list/tag-list.component.html index e68b997d1..7eae1e21e 100644 --- a/src-ui/src/app/components/manage/tag-list/tag-list.component.html +++ b/src-ui/src/app/components/manage/tag-list/tag-list.component.html @@ -1,37 +1,55 @@ - - + + -
- +
+
+
+ + +
+
+ +
- +
- - - - - + + + + + - + -
NameColourMatchingDocument countActionsNameColorMatchingDocument countActions
{{ tag.name }}{{ getColor(tag.colour).name }}{{tag.color}} {{ getMatching(tag) }} {{ tag.document_count }}
- - + + +
\ No newline at end of file + diff --git a/src-ui/src/app/components/manage/tag-list/tag-list.component.ts b/src-ui/src/app/components/manage/tag-list/tag-list.component.ts index efbe11321..7ea379f64 100644 --- a/src-ui/src/app/components/manage/tag-list/tag-list.component.ts +++ b/src-ui/src/app/components/manage/tag-list/tag-list.component.ts @@ -1,9 +1,10 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; +import { Component } from '@angular/core'; import { NgbModal } from '@ng-bootstrap/ng-bootstrap'; -import { TAG_COLOURS, PaperlessTag } from 'src/app/data/paperless-tag'; +import { FILTER_HAS_TAG } from 'src/app/data/filter-rule-type'; +import { PaperlessTag } from 'src/app/data/paperless-tag'; +import { DocumentListViewService } from 'src/app/services/document-list-view.service'; import { TagService } from 'src/app/services/rest/tag.service'; -import { environment } from 'src/environments/environment'; +import { ToastService } from 'src/app/services/toast.service'; import { GenericListComponent } from '../generic-list/generic-list.component'; import { TagEditDialogComponent } from './tag-edit-dialog/tag-edit-dialog.component'; @@ -12,23 +13,21 @@ import { TagEditDialogComponent } from './tag-edit-dialog/tag-edit-dialog.compon templateUrl: './tag-list.component.html', styleUrls: ['./tag-list.component.scss'] }) -export class TagListComponent extends GenericListComponent implements OnInit { +export class TagListComponent extends GenericListComponent { - constructor(tagService: TagService, modalService: NgbModal, private titleService: Title) { - super(tagService, modalService, TagEditDialogComponent) + constructor(tagService: TagService, modalService: NgbModal, + private list: DocumentListViewService, + toastService: ToastService + ) { + super(tagService, modalService, TagEditDialogComponent, toastService) } - - ngOnInit(): void { - super.ngOnInit() - this.titleService.setTitle(`Tags - ${environment.appTitle}`) + getDeleteMessage(object: PaperlessTag) { + return $localize`Do you really want to delete the tag "${object.name}"?` } - getColor(id) { - return TAG_COLOURS.find(c => c.id == id) - } + filterDocuments(object: PaperlessTag) { + this.list.quickFilter([{rule_type: FILTER_HAS_TAG, value: object.id.toString()}]) - getObjectName(object: PaperlessTag) { - return `tag '${object.name}'` } } diff --git a/src-ui/src/app/components/not-found/not-found.component.html b/src-ui/src/app/components/not-found/not-found.component.html index 6b21cf3ba..913132d1b 100644 --- a/src-ui/src/app/components/not-found/not-found.component.html +++ b/src-ui/src/app/components/not-found/not-found.component.html @@ -4,5 +4,5 @@ -

404 Not Found

-
\ No newline at end of file +

404 Not Found

+
diff --git a/src-ui/src/app/components/search/result-highlight/result-highlight.component.html b/src-ui/src/app/components/search/result-highlight/result-highlight.component.html deleted file mode 100644 index 1842f5cea..000000000 --- a/src-ui/src/app/components/search/result-highlight/result-highlight.component.html +++ /dev/null @@ -1,3 +0,0 @@ -... - {{token.text}} ... - \ No newline at end of file diff --git a/src-ui/src/app/components/search/result-highlight/result-highlight.component.scss b/src-ui/src/app/components/search/result-highlight/result-highlight.component.scss deleted file mode 100644 index 645fb0426..000000000 --- a/src-ui/src/app/components/search/result-highlight/result-highlight.component.scss +++ /dev/null @@ -1,4 +0,0 @@ -.match { - color: black; - background-color: orange; -} \ No newline at end of file diff --git a/src-ui/src/app/components/search/result-highlight/result-highlight.component.ts b/src-ui/src/app/components/search/result-highlight/result-highlight.component.ts deleted file mode 100644 index d9a1a50b1..000000000 --- a/src-ui/src/app/components/search/result-highlight/result-highlight.component.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { Component, Input, OnInit } from '@angular/core'; -import { SearchHitHighlight } from 'src/app/data/search-result'; - -@Component({ - selector: 'app-result-highlight', - templateUrl: './result-highlight.component.html', - styleUrls: ['./result-highlight.component.scss'] -}) -export class ResultHighlightComponent implements OnInit { - - constructor() { } - - @Input() - highlights: SearchHitHighlight[][] - - ngOnInit(): void { - } - -} diff --git a/src-ui/src/app/components/search/search.component.html b/src-ui/src/app/components/search/search.component.html deleted file mode 100644 index 55fcee900..000000000 --- a/src-ui/src/app/components/search/search.component.html +++ /dev/null @@ -1,21 +0,0 @@ - - - -
Invalid search query: {{errorMessage}}
- -

- Search string: {{query}} - - - Did you mean "{{correctedQuery}}"? - - -

- -
-

{{resultCount}} result(s)

- - - -
diff --git a/src-ui/src/app/components/search/search.component.scss b/src-ui/src/app/components/search/search.component.scss deleted file mode 100644 index 40ca79a61..000000000 --- a/src-ui/src/app/components/search/search.component.scss +++ /dev/null @@ -1,15 +0,0 @@ -.result-content { - color: darkgray; -} - -.doc-img { - object-fit: cover; - object-position: top; - height: 100%; - position: absolute; - -} - -.result-content-searching { - opacity: 0.3; -} \ No newline at end of file diff --git a/src-ui/src/app/components/search/search.component.ts b/src-ui/src/app/components/search/search.component.ts deleted file mode 100644 index 3371debd2..000000000 --- a/src-ui/src/app/components/search/search.component.ts +++ /dev/null @@ -1,77 +0,0 @@ -import { Component, OnInit } from '@angular/core'; -import { Title } from '@angular/platform-browser'; -import { ActivatedRoute, Router } from '@angular/router'; -import { SearchHit } from 'src/app/data/search-result'; -import { SearchService } from 'src/app/services/rest/search.service'; -import { environment } from 'src/environments/environment'; - -@Component({ - selector: 'app-search', - templateUrl: './search.component.html', - styleUrls: ['./search.component.scss'] -}) -export class SearchComponent implements OnInit { - - results: SearchHit[] = [] - - query: string = "" - - searching = false - - currentPage = 1 - - pageCount = 1 - - resultCount - - correctedQuery: string = null - - errorMessage: string - - constructor(private searchService: SearchService, private route: ActivatedRoute, private router: Router, private titleService: Title) { } - - ngOnInit(): void { - this.route.queryParamMap.subscribe(paramMap => { - this.query = paramMap.get('query') - this.searching = true - this.currentPage = 1 - this.loadPage() - this.titleService.setTitle(`Search: ${this.query} - ${environment.appTitle}`) - }) - - } - - searchCorrectedQuery() { - this.router.navigate(["search"], {queryParams: {query: this.correctedQuery}}) - } - - loadPage(append: boolean = false) { - this.errorMessage = null - this.correctedQuery = null - this.searchService.search(this.query, this.currentPage).subscribe(result => { - if (append) { - this.results.push(...result.results) - } else { - this.results = result.results - } - this.pageCount = result.page_count - this.searching = false - this.resultCount = result.count - this.correctedQuery = result.corrected_query - }, error => { - this.searching = false - this.resultCount = 1 - this.pageCount = 1 - this.results = [] - this.errorMessage = error.error - }) - } - - onScroll() { - if (this.currentPage < this.pageCount) { - this.currentPage += 1 - this.loadPage(true) - } - } - -} diff --git a/src-ui/src/app/data/filter-rule-type.ts b/src-ui/src/app/data/filter-rule-type.ts index a35759f69..c215be84e 100644 --- a/src-ui/src/app/data/filter-rule-type.ts +++ b/src-ui/src/app/data/filter-rule-type.ts @@ -18,40 +18,54 @@ export const FILTER_MODIFIED_AFTER = 16 export const FILTER_DOES_NOT_HAVE_TAG = 17 +export const FILTER_ASN_ISNULL = 18 + +export const FILTER_TITLE_CONTENT = 19 + +export const FILTER_FULLTEXT_QUERY = 20 +export const FILTER_FULLTEXT_MORELIKE = 21 + export const FILTER_RULE_TYPES: FilterRuleType[] = [ - {id: FILTER_TITLE, name: "Title contains", filtervar: "title__icontains", datatype: "string", multi: false, default: ""}, - {id: FILTER_CONTENT, name: "Content contains", filtervar: "content__icontains", datatype: "string", multi: false, default: ""}, - - {id: FILTER_ASN, name: "ASN is", filtervar: "archive_serial_number", datatype: "number", multi: false}, - - {id: FILTER_CORRESPONDENT, name: "Correspondent is", filtervar: "correspondent__id", datatype: "correspondent", multi: false}, - {id: FILTER_DOCUMENT_TYPE, name: "Document type is", filtervar: "document_type__id", datatype: "document_type", multi: false}, + {id: FILTER_TITLE, filtervar: "title__icontains", datatype: "string", multi: false, default: ""}, + {id: FILTER_CONTENT, filtervar: "content__icontains", datatype: "string", multi: false, default: ""}, - {id: FILTER_IS_IN_INBOX, name: "Is in Inbox", filtervar: "is_in_inbox", datatype: "boolean", multi: false, default: true}, - {id: FILTER_HAS_TAG, name: "Has tag", filtervar: "tags__id__all", datatype: "tag", multi: true}, - {id: FILTER_DOES_NOT_HAVE_TAG, name: "Does not have tag", filtervar: "tags__id__none", datatype: "tag", multi: true}, - {id: FILTER_HAS_ANY_TAG, name: "Has any tag", filtervar: "is_tagged", datatype: "boolean", multi: false, default: true}, + {id: FILTER_ASN, filtervar: "archive_serial_number", datatype: "number", multi: false}, - {id: FILTER_CREATED_BEFORE, name: "Created before", filtervar: "created__date__lt", datatype: "date", multi: false}, - {id: FILTER_CREATED_AFTER, name: "Created after", filtervar: "created__date__gt", datatype: "date", multi: false}, + {id: FILTER_CORRESPONDENT, filtervar: "correspondent__id", isnull_filtervar: "correspondent__isnull", datatype: "correspondent", multi: false}, + {id: FILTER_DOCUMENT_TYPE, filtervar: "document_type__id", isnull_filtervar: "document_type__isnull", datatype: "document_type", multi: false}, - {id: FILTER_CREATED_YEAR, name: "Year created is", filtervar: "created__year", datatype: "number", multi: false}, - {id: FILTER_CREATED_MONTH, name: "Month created is", filtervar: "created__month", datatype: "number", multi: false}, - {id: FILTER_CREATED_DAY, name: "Day created is", filtervar: "created__day", datatype: "number", multi: false}, + {id: FILTER_IS_IN_INBOX, filtervar: "is_in_inbox", datatype: "boolean", multi: false, default: true}, + {id: FILTER_HAS_TAG, filtervar: "tags__id__all", datatype: "tag", multi: true}, + {id: FILTER_DOES_NOT_HAVE_TAG, filtervar: "tags__id__none", datatype: "tag", multi: true}, + {id: FILTER_HAS_ANY_TAG, filtervar: "is_tagged", datatype: "boolean", multi: false, default: true}, - {id: FILTER_ADDED_BEFORE, name: "Added before", filtervar: "added__date__lt", datatype: "date", multi: false}, - {id: FILTER_ADDED_AFTER, name: "Added after", filtervar: "added__date__gt", datatype: "date", multi: false}, - - {id: FILTER_MODIFIED_BEFORE, name: "Modified before", filtervar: "modified__date__lt", datatype: "date", multi: false}, - {id: FILTER_MODIFIED_AFTER, name: "Modified after", filtervar: "modified__date__gt", datatype: "date", multi: false}, + {id: FILTER_CREATED_BEFORE, filtervar: "created__date__lt", datatype: "date", multi: false}, + {id: FILTER_CREATED_AFTER, filtervar: "created__date__gt", datatype: "date", multi: false}, + + {id: FILTER_CREATED_YEAR, filtervar: "created__year", datatype: "number", multi: false}, + {id: FILTER_CREATED_MONTH, filtervar: "created__month", datatype: "number", multi: false}, + {id: FILTER_CREATED_DAY, filtervar: "created__day", datatype: "number", multi: false}, + + {id: FILTER_ADDED_BEFORE, filtervar: "added__date__lt", datatype: "date", multi: false}, + {id: FILTER_ADDED_AFTER, filtervar: "added__date__gt", datatype: "date", multi: false}, + + {id: FILTER_MODIFIED_BEFORE, filtervar: "modified__date__lt", datatype: "date", multi: false}, + {id: FILTER_MODIFIED_AFTER, filtervar: "modified__date__gt", datatype: "date", multi: false}, + {id: FILTER_ASN_ISNULL, filtervar: "archive_serial_number__isnull", datatype: "boolean", multi: false}, + + {id: FILTER_TITLE_CONTENT, filtervar: "title_content", datatype: "string", multi: false}, + + {id: FILTER_FULLTEXT_QUERY, filtervar: "query", datatype: "string", multi: false}, + + {id: FILTER_FULLTEXT_MORELIKE, filtervar: "more_like_id", datatype: "number", multi: false}, ] export interface FilterRuleType { id: number - name: string filtervar: string + isnull_filtervar?: string datatype: string //number, string, boolean, date multi: boolean default?: any -} \ No newline at end of file +} diff --git a/src-ui/src/app/data/filter-rule.ts b/src-ui/src/app/data/filter-rule.ts index 2dc632d9c..c8b790e21 100644 --- a/src-ui/src/app/data/filter-rule.ts +++ b/src-ui/src/app/data/filter-rule.ts @@ -1,18 +1,22 @@ -import { FilterRuleType } from './filter-rule-type'; +import { FILTER_FULLTEXT_MORELIKE, FILTER_FULLTEXT_QUERY } from "./filter-rule-type" export function cloneFilterRules(filterRules: FilterRule[]): FilterRule[] { if (filterRules) { let newRules: FilterRule[] = [] for (let rule of filterRules) { - newRules.push({type: rule.type, value: rule.value}) + newRules.push({rule_type: rule.rule_type, value: rule.value}) } - return newRules + return newRules } else { return null } } +export function isFullTextFilterRule(filterRules: FilterRule[]): boolean { + return filterRules.find(r => r.rule_type == FILTER_FULLTEXT_QUERY || r.rule_type == FILTER_FULLTEXT_MORELIKE) != null +} + export interface FilterRule { - type: FilterRuleType - value: any -} \ No newline at end of file + rule_type: number + value: string +} diff --git a/src-ui/src/app/data/matching-model.ts b/src-ui/src/app/data/matching-model.ts index 698c32da5..49cb5cad4 100644 --- a/src-ui/src/app/data/matching-model.ts +++ b/src-ui/src/app/data/matching-model.ts @@ -9,12 +9,12 @@ export const MATCH_FUZZY = 5 export const MATCH_AUTO = 6 export const MATCHING_ALGORITHMS = [ - {id: MATCH_ANY, name: "Any"}, - {id: MATCH_ALL, name: "All"}, - {id: MATCH_LITERAL, name: "Literal"}, - {id: MATCH_REGEX, name: "Regular Expression"}, - {id: MATCH_FUZZY, name: "Fuzzy Match"}, - {id: MATCH_AUTO, name: "Auto"}, + {id: MATCH_ANY, shortName: $localize`Any word`, name: $localize`Any: Document contains any of these words (space separated)`}, + {id: MATCH_ALL, shortName: $localize`All words`, name: $localize`All: Document contains all of these words (space separated)`}, + {id: MATCH_LITERAL, shortName: $localize`Exact match`, name: $localize`Exact: Document contains this string`}, + {id: MATCH_REGEX, shortName: $localize`Regular expression`, name: $localize`Regular expression: Document matches this regular expression`}, + {id: MATCH_FUZZY, shortName: $localize`Fuzzy word`, name: $localize`Fuzzy: Document contains a word similar to this word`}, + {id: MATCH_AUTO, shortName: $localize`Automatic`, name: $localize`Auto: Learn matching automatically`}, ] export interface MatchingModel extends ObjectWithId { @@ -29,4 +29,6 @@ export interface MatchingModel extends ObjectWithId { is_insensitive?: boolean + document_count?: number + } diff --git a/src-ui/src/app/data/paperless-correspondent.ts b/src-ui/src/app/data/paperless-correspondent.ts index 217e62529..3f5bb5719 100644 --- a/src-ui/src/app/data/paperless-correspondent.ts +++ b/src-ui/src/app/data/paperless-correspondent.ts @@ -2,8 +2,6 @@ import { MatchingModel } from './matching-model'; export interface PaperlessCorrespondent extends MatchingModel { - document_count?: number - last_correspondence?: Date } diff --git a/src-ui/src/app/data/paperless-document-metadata.ts b/src-ui/src/app/data/paperless-document-metadata.ts index 12f0a78d8..a1063adf9 100644 --- a/src-ui/src/app/data/paperless-document-metadata.ts +++ b/src-ui/src/app/data/paperless-document-metadata.ts @@ -1,5 +1,5 @@ export interface PaperlessDocumentMetadata { - + original_checksum?: string archived_checksum?: string @@ -10,4 +10,4 @@ export interface PaperlessDocumentMetadata { has_archive_version?: boolean -} \ No newline at end of file +} diff --git a/src-ui/src/app/data/paperless-document-suggestions.ts b/src-ui/src/app/data/paperless-document-suggestions.ts new file mode 100644 index 000000000..cc68b2b27 --- /dev/null +++ b/src-ui/src/app/data/paperless-document-suggestions.ts @@ -0,0 +1,9 @@ +export interface PaperlessDocumentSuggestions { + + tags?: number[] + + correspondents?: number[] + + document_types?: number[] + +} diff --git a/src-ui/src/app/data/paperless-document-type.ts b/src-ui/src/app/data/paperless-document-type.ts index d099bec47..b1d002461 100644 --- a/src-ui/src/app/data/paperless-document-type.ts +++ b/src-ui/src/app/data/paperless-document-type.ts @@ -2,6 +2,4 @@ import { MatchingModel } from './matching-model'; export interface PaperlessDocumentType extends MatchingModel { - document_count?: number - } diff --git a/src-ui/src/app/data/paperless-document.ts b/src-ui/src/app/data/paperless-document.ts index 9d0aeda88..e7412278b 100644 --- a/src-ui/src/app/data/paperless-document.ts +++ b/src-ui/src/app/data/paperless-document.ts @@ -4,6 +4,15 @@ import { PaperlessTag } from './paperless-tag' import { PaperlessDocumentType } from './paperless-document-type' import { Observable } from 'rxjs' +export interface SearchHit { + + score?: number + rank?: number + + highlights?: string + +} + export interface PaperlessDocument extends ObjectWithId { correspondent$?: Observable @@ -40,4 +49,6 @@ export interface PaperlessDocument extends ObjectWithId { archive_serial_number?: number + __search_hit__?: SearchHit + } diff --git a/src-ui/src/app/data/paperless-log.ts b/src-ui/src/app/data/paperless-log.ts deleted file mode 100644 index 61a6fce99..000000000 --- a/src-ui/src/app/data/paperless-log.ts +++ /dev/null @@ -1,27 +0,0 @@ -export const LOG_LEVEL_DEBUG = 10 -export const LOG_LEVEL_INFO = 20 -export const LOG_LEVEL_WARNING = 30 -export const LOG_LEVEL_ERROR = 40 -export const LOG_LEVEL_CRITICAL = 50 - -export const LOG_LEVELS = [ - {id: LOG_LEVEL_DEBUG, name: "DEBUG"}, - {id: LOG_LEVEL_INFO, name: "INFO"}, - {id: LOG_LEVEL_WARNING, name: "WARNING"}, - {id: LOG_LEVEL_ERROR, name: "ERROR"}, - {id: LOG_LEVEL_CRITICAL, name: "CRITICAL"} -] - -export interface PaperlessLog { - - id?: number - - group?: string - - message?: string - - created?: Date - - level?: number - -} diff --git a/src-ui/src/app/data/paperless-saved-view.ts b/src-ui/src/app/data/paperless-saved-view.ts new file mode 100644 index 000000000..8a4cd58b7 --- /dev/null +++ b/src-ui/src/app/data/paperless-saved-view.ts @@ -0,0 +1,18 @@ +import { FilterRule } from './filter-rule'; +import { ObjectWithId } from './object-with-id'; + +export interface PaperlessSavedView extends ObjectWithId { + + name?: string + + show_on_dashboard?: boolean + + show_in_sidebar?: boolean + + sort_field: string + + sort_reverse: boolean + + filter_rules: FilterRule[] + +} diff --git a/src-ui/src/app/data/paperless-tag.ts b/src-ui/src/app/data/paperless-tag.ts index 551c6e03a..a999666c4 100644 --- a/src-ui/src/app/data/paperless-tag.ts +++ b/src-ui/src/app/data/paperless-tag.ts @@ -1,28 +1,11 @@ -import { MatchingModel } from './matching-model'; -import { ObjectWithId } from './object-with-id'; - - -export const TAG_COLOURS = [ - {id: 1, value: "#a6cee3", name: "Light Blue", textColor: "#000000"}, - {id: 2, value: "#1f78b4", name: "Blue", textColor: "#ffffff"}, - {id: 3, value: "#b2df8a", name: "Light Green", textColor: "#000000"}, - {id: 4, value: "#33a02c", name: "Green", textColor: "#000000"}, - {id: 5, value: "#fb9a99", name: "Light Red", textColor: "#000000"}, - {id: 6, value: "#e31a1c", name: "Red ", textColor: "#ffffff"}, - {id: 7, value: "#fdbf6f", name: "Light Orange", textColor: "#000000"}, - {id: 8, value: "#ff7f00", name: "Orange", textColor: "#000000"}, - {id: 9, value: "#cab2d6", name: "Light Violet", textColor: "#000000"}, - {id: 10, value: "#6a3d9a", name: "Violet", textColor: "#ffffff"}, - {id: 11, value: "#b15928", name: "Brown", textColor: "#000000"}, - {id: 12, value: "#000000", name: "Black", textColor: "#ffffff"}, - {id: 13, value: "#cccccc", name: "Light Grey", textColor: "#000000"} -] +import { MatchingModel } from "./matching-model"; export interface PaperlessTag extends MatchingModel { - colour?: number + color?: string + + text_color?: string is_inbox_tag?: boolean - - document_count?: number + } diff --git a/src-ui/src/app/data/results.ts b/src-ui/src/app/data/results.ts index 83e9c583c..839c79ed0 100644 --- a/src-ui/src/app/data/results.ts +++ b/src-ui/src/app/data/results.ts @@ -3,5 +3,5 @@ export interface Results { count: number results: T[] - + } diff --git a/src-ui/src/app/data/saved-view-config.ts b/src-ui/src/app/data/saved-view-config.ts deleted file mode 100644 index 9d7076215..000000000 --- a/src-ui/src/app/data/saved-view-config.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { FilterRule } from './filter-rule'; - -export interface SavedViewConfig { - - id?: string - - filterRules: FilterRule[] - - sortField: string - - sortDirection: string - - title?: string - - showInSideBar?: boolean - - showInDashboard?: boolean - -} \ No newline at end of file diff --git a/src-ui/src/app/data/search-result.ts b/src-ui/src/app/data/search-result.ts deleted file mode 100644 index a769a8351..000000000 --- a/src-ui/src/app/data/search-result.ts +++ /dev/null @@ -1,29 +0,0 @@ -import { PaperlessDocument } from './paperless-document' - -export class SearchHitHighlight { - text?: string - term?: number -} - -export interface SearchHit { - id?: number - title?: string - score?: number - rank?: number - - highlights?: SearchHitHighlight[][] - document?: PaperlessDocument -} - -export interface SearchResult { - - count?: number - page?: number - page_count?: number - - corrected_query?: string - - results?: SearchHit[] - - -} diff --git a/src-ui/src/app/data/storage-keys.ts b/src-ui/src/app/data/storage-keys.ts index 13b41d4a7..ec91c06ac 100644 --- a/src-ui/src/app/data/storage-keys.ts +++ b/src-ui/src/app/data/storage-keys.ts @@ -5,8 +5,3 @@ export const OPEN_DOCUMENT_SERVICE = { export const DOCUMENT_LIST_SERVICE = { CURRENT_VIEW_CONFIG: 'document-list-service:currentViewConfig' } - -export const GENERAL_SETTINGS = { - DOCUMENT_LIST_SIZE: 'general-settings:documentListSize', - DOCUMENT_LIST_SIZE_DEFAULT: 50 -} \ No newline at end of file diff --git a/src-ui/src/app/data/websocket-consumer-status-message.ts b/src-ui/src/app/data/websocket-consumer-status-message.ts new file mode 100644 index 000000000..c20882067 --- /dev/null +++ b/src-ui/src/app/data/websocket-consumer-status-message.ts @@ -0,0 +1,11 @@ +export interface WebsocketConsumerStatusMessage { + + filename?: string + task_id?: string + current_progress?: number + max_progress?: number + status?: string + message?: string + document_id: number + +} diff --git a/src-ui/src/app/directives/sortable.directive.ts b/src-ui/src/app/directives/sortable.directive.ts index 11c474dbb..22750273d 100644 --- a/src-ui/src/app/directives/sortable.directive.ts +++ b/src-ui/src/app/directives/sortable.directive.ts @@ -1,17 +1,15 @@ import { Directive, EventEmitter, Input, Output } from '@angular/core'; export interface SortEvent { - column: string; - direction: string; + column: string + reverse: boolean } -const rotate: {[key: string]: string} = { 'asc': 'des', 'des': '', '': 'asc' }; - @Directive({ selector: 'th[sortable]', host: { - '[class.asc]': 'direction === "asc"', - '[class.des]': 'direction === "des"', + '[class.asc]': 'currentSortField == sortable && !currentSortReverse', + '[class.des]': 'currentSortField == sortable && currentSortReverse', '(click)': 'rotate()' } }) @@ -19,12 +17,24 @@ export class SortableDirective { constructor() { } - @Input() sortable: string = ''; - @Input() direction: string = ''; + @Input() + sortable: string = ''; + + @Input() + currentSortReverse: boolean = false + + @Input() + currentSortField: string + @Output() sort = new EventEmitter(); rotate() { - this.direction = rotate[this.direction]; - this.sort.emit({column: this.sortable, direction: this.direction}); + if (this.currentSortField != this.sortable) { + this.sort.emit({column: this.sortable, reverse: false}); + } else if (this.currentSortField == this.sortable && !this.currentSortReverse) { + this.sort.emit({column: this.currentSortField, reverse: true}); + } else { + this.sort.emit({column: null, reverse: false}); + } } } diff --git a/src-ui/src/app/interceptors/api-version.interceptor.spec.ts b/src-ui/src/app/interceptors/api-version.interceptor.spec.ts new file mode 100644 index 000000000..7ad51f687 --- /dev/null +++ b/src-ui/src/app/interceptors/api-version.interceptor.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { ApiVersionInterceptor } from './api-version.interceptor'; + +describe('ApiVersionInterceptor', () => { + beforeEach(() => TestBed.configureTestingModule({ + providers: [ + ApiVersionInterceptor + ] + })); + + it('should be created', () => { + const interceptor: ApiVersionInterceptor = TestBed.inject(ApiVersionInterceptor); + expect(interceptor).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/interceptors/api-version.interceptor.ts b/src-ui/src/app/interceptors/api-version.interceptor.ts new file mode 100644 index 000000000..4df2efba6 --- /dev/null +++ b/src-ui/src/app/interceptors/api-version.interceptor.ts @@ -0,0 +1,25 @@ +import { Injectable } from '@angular/core'; +import { + HttpRequest, + HttpHandler, + HttpEvent, + HttpInterceptor +} from '@angular/common/http'; +import { Observable } from 'rxjs'; +import { environment } from 'src/environments/environment'; + +@Injectable() +export class ApiVersionInterceptor implements HttpInterceptor { + + constructor() {} + + intercept(request: HttpRequest, next: HttpHandler): Observable> { + request = request.clone({ + setHeaders: { + 'Accept': `application/json; version=${environment.apiVersion}` + } + }) + + return next.handle(request); + } +} diff --git a/src-ui/src/app/interceptors/csrf.interceptor.ts b/src-ui/src/app/interceptors/csrf.interceptor.ts index 32f3e99dc..1817e3a66 100644 --- a/src-ui/src/app/interceptors/csrf.interceptor.ts +++ b/src-ui/src/app/interceptors/csrf.interceptor.ts @@ -7,16 +7,21 @@ import { } from '@angular/common/http'; import { Observable } from 'rxjs'; import { CookieService } from 'ngx-cookie-service'; +import { Meta } from '@angular/platform-browser'; @Injectable() export class CsrfInterceptor implements HttpInterceptor { - constructor(private cookieService: CookieService) { + constructor(private cookieService: CookieService, private meta: Meta) { } intercept(request: HttpRequest, next: HttpHandler): Observable> { - let csrfToken = this.cookieService.get('csrftoken') + let prefix = "" + if (this.meta.getTag('name=cookie_prefix')) { + prefix = this.meta.getTag('name=cookie_prefix').content + } + let csrfToken = this.cookieService.get(`${prefix?prefix:''}csrftoken`) if (csrfToken) { request = request.clone({ setHeaders: { diff --git a/src-ui/src/app/pipes/custom-date.pipe.spec.ts b/src-ui/src/app/pipes/custom-date.pipe.spec.ts new file mode 100644 index 000000000..822966baf --- /dev/null +++ b/src-ui/src/app/pipes/custom-date.pipe.spec.ts @@ -0,0 +1,8 @@ +import { CustomDatePipe } from './custom-date.pipe'; + +describe('CustomDatePipe', () => { + it('create an instance', () => { + const pipe = new CustomDatePipe(); + expect(pipe).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/pipes/custom-date.pipe.ts b/src-ui/src/app/pipes/custom-date.pipe.ts new file mode 100644 index 000000000..71f386b8c --- /dev/null +++ b/src-ui/src/app/pipes/custom-date.pipe.ts @@ -0,0 +1,32 @@ +import { DatePipe } from '@angular/common'; +import { Inject, LOCALE_ID, Pipe, PipeTransform } from '@angular/core'; +import { SettingsService, SETTINGS_KEYS } from '../services/settings.service'; + +const FORMAT_TO_ISO_FORMAT = { + "longDate": "y-MM-dd", + "mediumDate": "y-MM-dd", + "shortDate": "y-MM-dd" +} + +@Pipe({ + name: 'customDate' +}) +export class CustomDatePipe implements PipeTransform { + + private defaultLocale: string + + constructor(@Inject(LOCALE_ID) locale: string, private datePipe: DatePipe, private settings: SettingsService) { + this.defaultLocale = locale + } + + transform(value: any, format?: string, timezone?: string, locale?: string): string | null { + let l = locale || this.settings.get(SETTINGS_KEYS.DATE_LOCALE) || this.defaultLocale + let f = format || this.settings.get(SETTINGS_KEYS.DATE_FORMAT) + if (l == "iso-8601") { + return this.datePipe.transform(value, FORMAT_TO_ISO_FORMAT[f], timezone) + } else { + return this.datePipe.transform(value, format || this.settings.get(SETTINGS_KEYS.DATE_FORMAT), timezone, l) + } + } + +} diff --git a/src-ui/src/app/pipes/document-title.pipe.ts b/src-ui/src/app/pipes/document-title.pipe.ts index 09445f595..60a552ca8 100644 --- a/src-ui/src/app/pipes/document-title.pipe.ts +++ b/src-ui/src/app/pipes/document-title.pipe.ts @@ -5,11 +5,11 @@ import { Pipe, PipeTransform } from '@angular/core'; }) export class DocumentTitlePipe implements PipeTransform { - transform(value: string): unknown { + transform(value: string): string { if (value) { return value } else { - return "(no title)" + return $localize`(no title)` } } diff --git a/src-ui/src/app/pipes/file-size.pipe.ts b/src-ui/src/app/pipes/file-size.pipe.ts index 7d742c876..4ac1e4d61 100644 --- a/src-ui/src/app/pipes/file-size.pipe.ts +++ b/src-ui/src/app/pipes/file-size.pipe.ts @@ -1,6 +1,6 @@ /** * https://gist.github.com/JonCatmull/ecdf9441aaa37336d9ae2c7f9cb7289a - * + * * @license * Copyright (c) 2019 Jonathan Catmull. * diff --git a/src-ui/src/app/pipes/filter.pipe.ts b/src-ui/src/app/pipes/filter.pipe.ts new file mode 100644 index 000000000..d83ccc07a --- /dev/null +++ b/src-ui/src/app/pipes/filter.pipe.ts @@ -0,0 +1,19 @@ +import { Pipe, PipeTransform } from '@angular/core'; +import { ToggleableItem } from 'src/app/components/common/filterable-dropdown/toggleable-dropdown-button/toggleable-dropdown-button.component'; +import { MatchingModel } from '../data/matching-model'; + +@Pipe({ + name: 'filter' +}) +export class FilterPipe implements PipeTransform { + transform(items: MatchingModel[], searchText: string): MatchingModel[] { + if (!items) return []; + if (!searchText) return items; + + return items.filter(item => { + return Object.keys(item).some(key => { + return String(item[key]).toLowerCase().includes(searchText.toLowerCase()); + }); + }); + } +} diff --git a/src-ui/src/app/pipes/safe.pipe.spec.ts b/src-ui/src/app/pipes/safe.pipe.spec.ts new file mode 100644 index 000000000..49ee0ad14 --- /dev/null +++ b/src-ui/src/app/pipes/safe.pipe.spec.ts @@ -0,0 +1,8 @@ +import { SafePipe } from './safe.pipe'; + +describe('SafePipe', () => { + it('create an instance', () => { + const pipe = new SafePipe(); + expect(pipe).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/pipes/safe.pipe.ts b/src-ui/src/app/pipes/safe.pipe.ts new file mode 100644 index 000000000..3aa987c08 --- /dev/null +++ b/src-ui/src/app/pipes/safe.pipe.ts @@ -0,0 +1,19 @@ +import { Pipe, PipeTransform } from '@angular/core'; +import { DomSanitizer } from '@angular/platform-browser'; + +@Pipe({ + name: 'safe' +}) +export class SafePipe implements PipeTransform { + + constructor(private sanitizer: DomSanitizer) { } + + transform(url) { + if (url == null) { + return this.sanitizer.bypassSecurityTrustResourceUrl("") + } else { + return this.sanitizer.bypassSecurityTrustResourceUrl(url); + } + } + +} diff --git a/src-ui/src/app/pipes/yes-no.pipe.ts b/src-ui/src/app/pipes/yes-no.pipe.ts index 9a4ed56ef..f12c844a5 100644 --- a/src-ui/src/app/pipes/yes-no.pipe.ts +++ b/src-ui/src/app/pipes/yes-no.pipe.ts @@ -6,7 +6,7 @@ import { Pipe, PipeTransform } from '@angular/core'; export class YesNoPipe implements PipeTransform { transform(value: boolean): unknown { - return value ? "Yes" : "No" + return value ? $localize`Yes` : $localize`No` } } diff --git a/src-ui/src/app/services/consumer-status.service.spec.ts b/src-ui/src/app/services/consumer-status.service.spec.ts new file mode 100644 index 000000000..d19f455e2 --- /dev/null +++ b/src-ui/src/app/services/consumer-status.service.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { ConsumerStatusService } from './consumer-status.service'; + +describe('ConsumerStatusService', () => { + let service: ConsumerStatusService; + + beforeEach(() => { + TestBed.configureTestingModule({}); + service = TestBed.inject(ConsumerStatusService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/services/consumer-status.service.ts b/src-ui/src/app/services/consumer-status.service.ts new file mode 100644 index 000000000..4a729b29d --- /dev/null +++ b/src-ui/src/app/services/consumer-status.service.ts @@ -0,0 +1,200 @@ +import { Injectable } from '@angular/core'; +import { Subject } from 'rxjs'; +import { environment } from 'src/environments/environment'; +import { WebsocketConsumerStatusMessage } from '../data/websocket-consumer-status-message'; + +export enum FileStatusPhase { + STARTED = 0, + UPLOADING = 1, + PROCESSING = 2, + SUCCESS = 3, + FAILED = 4 +} + +export const FILE_STATUS_MESSAGES = { + "document_already_exists": $localize`Document already exists.`, + "file_not_found": $localize`File not found.`, + "pre_consume_script_not_found": $localize`:Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation:Pre-consume script does not exist.`, + "pre_consume_script_error": $localize`:Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation:Error while executing pre-consume script.`, + "post_consume_script_not_found": $localize`:Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation:Post-consume script does not exist.`, + "post_consume_script_error": $localize`:Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation:Error while executing post-consume script.`, + "new_file": $localize`Received new file.`, + "unsupported_type": $localize`File type not supported.`, + "parsing_document": $localize`Processing document...`, + "generating_thumbnail": $localize`Generating thumbnail...`, + "parse_date": $localize`Retrieving date from document...`, + "save_document": $localize`Saving document...`, + "finished": $localize`Finished.` +} + +export class FileStatus { + + filename: string + + taskId: string + + phase: FileStatusPhase = FileStatusPhase.STARTED + + currentPhaseProgress: number + + currentPhaseMaxProgress: number + + message: string + + documentId: number + + getProgress(): number { + switch (this.phase) { + case FileStatusPhase.STARTED: + return 0.0 + case FileStatusPhase.UPLOADING: + return this.currentPhaseProgress / this.currentPhaseMaxProgress * 0.2 + case FileStatusPhase.PROCESSING: + return (this.currentPhaseProgress / this.currentPhaseMaxProgress * 0.8) + 0.2 + case FileStatusPhase.SUCCESS: + case FileStatusPhase.FAILED: + return 1.0 + } + } + + updateProgress(status: FileStatusPhase, currentProgress?: number, maxProgress?: number) { + if (status >= this.phase) { + this.phase = status + if (currentProgress != null) { + this.currentPhaseProgress = currentProgress + } + if (maxProgress != null) { + this.currentPhaseMaxProgress = maxProgress + } + } + } + +} + +@Injectable({ + providedIn: 'root' +}) +export class ConsumerStatusService { + + constructor() { } + + private statusWebSocket: WebSocket + + private consumerStatus: FileStatus[] = [] + + private documentDetectedSubject = new Subject() + private documentConsumptionFinishedSubject = new Subject() + private documentConsumptionFailedSubject = new Subject() + + private get(taskId: string, filename?: string) { + let status = this.consumerStatus.find(e => e.taskId == taskId) || this.consumerStatus.find(e => e.filename == filename && e.taskId == null) + let created = false + if (!status) { + status = new FileStatus() + this.consumerStatus.push(status) + created = true + } + status.taskId = taskId + status.filename = filename + return {'status': status, 'created': created} + } + + newFileUpload(filename: string): FileStatus { + let status = new FileStatus() + status.filename = filename + this.consumerStatus.push(status) + return status + } + + getConsumerStatus(phase?: FileStatusPhase) { + if (phase != null) { + return this.consumerStatus.filter(s => s.phase == phase) + } else { + return this.consumerStatus + } + } + + getConsumerStatusNotCompleted() { + return this.consumerStatus.filter(s => s.phase < FileStatusPhase.SUCCESS) + } + + getConsumerStatusCompleted() { + return this.consumerStatus.filter(s => s.phase == FileStatusPhase.FAILED || s.phase == FileStatusPhase.SUCCESS) + } + + connect() { + this.disconnect() + + this.statusWebSocket = new WebSocket(`${environment.webSocketProtocol}//${environment.webSocketHost}${environment.webSocketBaseUrl}status/`); + this.statusWebSocket.onmessage = (ev) => { + let statusMessage: WebsocketConsumerStatusMessage = JSON.parse(ev['data']) + + let statusMessageGet = this.get(statusMessage.task_id, statusMessage.filename) + let status = statusMessageGet.status + let created = statusMessageGet.created + + status.updateProgress(FileStatusPhase.PROCESSING, statusMessage.current_progress, statusMessage.max_progress) + if (statusMessage.message && statusMessage.message in FILE_STATUS_MESSAGES) { + status.message = FILE_STATUS_MESSAGES[statusMessage.message] + } else if (statusMessage.message) { + status.message = statusMessage.message + } + status.documentId = statusMessage.document_id + + if (created && statusMessage.status == 'STARTING') { + this.documentDetectedSubject.next(status) + } + if (statusMessage.status == "SUCCESS") { + status.phase = FileStatusPhase.SUCCESS + this.documentConsumptionFinishedSubject.next(status) + } + if (statusMessage.status == "FAILED") { + status.phase = FileStatusPhase.FAILED + this.documentConsumptionFailedSubject.next(status) + } + } + } + + fail(status: FileStatus, message: string) { + status.message = message + status.phase = FileStatusPhase.FAILED + this.documentConsumptionFailedSubject.next(status) + } + + disconnect() { + if (this.statusWebSocket) { + this.statusWebSocket.close() + this.statusWebSocket = null + } + } + + dismiss(status: FileStatus) { + let index + if (status.taskId != null) { + index = this.consumerStatus.findIndex(s => s.taskId == status.taskId) + } else { + index = this.consumerStatus.findIndex(s => s.filename == status.filename) + } + + if (index > -1) { + this.consumerStatus.splice(index, 1) + } + } + + dismissCompleted() { + this.consumerStatus = this.consumerStatus.filter(status => status.phase != FileStatusPhase.SUCCESS) + } + + onDocumentConsumptionFinished() { + return this.documentConsumptionFinishedSubject + } + + onDocumentConsumptionFailed() { + return this.documentConsumptionFailedSubject + } + + onDocumentDetected() { + return this.documentDetectedSubject + } + +} diff --git a/src-ui/src/app/services/document-list-view.service.ts b/src-ui/src/app/services/document-list-view.service.ts index 811ac3c4b..4e1a8aad4 100644 --- a/src-ui/src/app/services/document-list-view.service.ts +++ b/src-ui/src/app/services/document-list-view.service.ts @@ -1,15 +1,60 @@ import { Injectable } from '@angular/core'; +import { ActivatedRoute, Router } from '@angular/router'; import { Observable } from 'rxjs'; -import { cloneFilterRules, FilterRule } from '../data/filter-rule'; +import { cloneFilterRules, FilterRule, isFullTextFilterRule } from '../data/filter-rule'; import { PaperlessDocument } from '../data/paperless-document'; -import { SavedViewConfig } from '../data/saved-view-config'; -import { DOCUMENT_LIST_SERVICE, GENERAL_SETTINGS } from '../data/storage-keys'; +import { PaperlessSavedView } from '../data/paperless-saved-view'; +import { DOCUMENT_LIST_SERVICE } from '../data/storage-keys'; import { DocumentService } from './rest/document.service'; +import { SettingsService, SETTINGS_KEYS } from './settings.service'; +/** + * Captures the current state of the list view. + */ +interface ListViewState { + + /** + * Title of the document list view. Either "Documents" (localized) or the name of a saved view. + */ + title?: string + + /** + * Current paginated list of documents displayed. + */ + documents?: PaperlessDocument[] + + currentPage: number + + /** + * Total amount of documents with the current filter rules. Used to calculate the number of pages. + */ + collectionSize: number + + /** + * Currently selected sort field. + */ + sortField: string + + /** + * True if the list is sorted in reverse. + */ + sortReverse: boolean + + /** + * Filter rules for the current list view. + */ + filterRules: FilterRule[] + + /** + * Contains the IDs of all selected documents. + */ + selected?: Set + +} /** * This service manages the document list which is displayed using the document list view. - * + * * This service also serves saved views by transparently switching between the document list * and saved views on request. See below. */ @@ -18,135 +63,192 @@ import { DocumentService } from './rest/document.service'; }) export class DocumentListViewService { - static DEFAULT_SORT_FIELD = 'created' - isReloading: boolean = false - documents: PaperlessDocument[] = [] - currentPage = 1 - currentPageSize: number = +localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT - collectionSize: number - - /** - * This is the current config for the document list. The service will always remember the last settings used for the document list. - */ - private _documentListViewConfig: SavedViewConfig - /** - * Optionally, this is the currently selected saved view, which might be null. - */ - private _savedViewConfig: SavedViewConfig + error: string = null - get savedView() { - return this._savedViewConfig + rangeSelectionAnchorIndex: number + lastRangeSelectionToIndex: number + + currentPageSize: number = this.settings.get(SETTINGS_KEYS.DOCUMENT_LIST_SIZE) + + private listViewStates: Map = new Map() + + private _activeSavedViewId: number = null + + get activeSavedViewId() { + return this._activeSavedViewId } - set savedView(value) { - if (value) { - //this is here so that we don't modify value, which might be the actual instance of the saved view. - this._savedViewConfig = Object.assign({}, value) + get activeSavedViewTitle() { + return this.activeListViewState.title + } + + private defaultListViewState(): ListViewState { + return { + title: null, + documents: [], + currentPage: 1, + collectionSize: null, + sortField: "created", + sortReverse: true, + filterRules: [], + selected: new Set() + } + } + + private get activeListViewState() { + if (!this.listViewStates.has(this._activeSavedViewId)) { + this.listViewStates.set(this._activeSavedViewId, this.defaultListViewState()) + } + return this.listViewStates.get(this._activeSavedViewId) + } + + activateSavedView(view: PaperlessSavedView) { + this.rangeSelectionAnchorIndex = this.lastRangeSelectionToIndex = null + if (view) { + this._activeSavedViewId = view.id + this.loadSavedView(view) } else { - this._savedViewConfig = null + this._activeSavedViewId = null } } - get savedViewId() { - return this.savedView?.id - } - - get savedViewTitle() { - return this.savedView?.title - } - - get documentListView() { - return this._documentListViewConfig - } - - set documentListView(value) { - if (value) { - this._documentListViewConfig = Object.assign({}, value) - this.saveDocumentListView() + loadSavedView(view: PaperlessSavedView, closeCurrentView: boolean = false) { + if (closeCurrentView) { + this._activeSavedViewId = null } - } - - /** - * This is what switches between the saved views and the document list view. Everything on the document list uses - * this property to determine the settings for the currently displayed document list. - */ - get view() { - return this.savedView || this.documentListView - } - - load(config: SavedViewConfig) { - this.view.filterRules = cloneFilterRules(config.filterRules) - this.view.sortDirection = config.sortDirection - this.view.sortField = config.sortField - this.reload() - } - - clear() { - this.collectionSize = null - this.documents = [] - this.currentPage = 1 + this.activeListViewState.filterRules = cloneFilterRules(view.filter_rules) + this.activeListViewState.sortField = view.sort_field + this.activeListViewState.sortReverse = view.sort_reverse + if (this._activeSavedViewId) { + this.activeListViewState.title = view.name + } + this.reduceSelectionToFilter() } reload(onFinish?) { this.isReloading = true - this.documentService.list( - this.currentPage, + this.error = null + let activeListViewState = this.activeListViewState + + this.documentService.listFiltered( + activeListViewState.currentPage, this.currentPageSize, - this.view.sortField, - this.view.sortDirection, - this.view.filterRules).subscribe( + activeListViewState.sortField, + activeListViewState.sortReverse, + activeListViewState.filterRules).subscribe( result => { - this.collectionSize = result.count - this.documents = result.results + this.isReloading = false + activeListViewState.collectionSize = result.count + activeListViewState.documents = result.results if (onFinish) { onFinish() } - this.isReloading = false + this.rangeSelectionAnchorIndex = this.lastRangeSelectionToIndex = null }, error => { - if (error.error['detail'] == 'Invalid page.') { - this.currentPage = 1 - this.reload() - } this.isReloading = false + if (activeListViewState.currentPage != 1 && error.status == 404) { + // this happens when applying a filter: the current page might not be available anymore due to the reduced result set. + activeListViewState.currentPage = 1 + this.reload() + } else { + this.error = error.error + } }) } set filterRules(filterRules: FilterRule[]) { - //we're going to clone the filterRules object, since we don't - //want changes in the filter editor to propagate into here right away. - this.view.filterRules = cloneFilterRules(filterRules) + if (!isFullTextFilterRule(filterRules) && this.activeListViewState.sortField == "score") { + this.activeListViewState.sortField = "created" + } + this.activeListViewState.filterRules = filterRules this.reload() + this.reduceSelectionToFilter() this.saveDocumentListView() } get filterRules(): FilterRule[] { - return cloneFilterRules(this.view.filterRules) + return this.activeListViewState.filterRules } set sortField(field: string) { - this.view.sortField = field - this.saveDocumentListView() + this.activeListViewState.sortField = field this.reload() + this.saveDocumentListView() } get sortField(): string { - return this.view.sortField + return this.activeListViewState.sortField } - set sortDirection(direction: string) { - this.view.sortDirection = direction - this.saveDocumentListView() + set sortReverse(reverse: boolean) { + this.activeListViewState.sortReverse = reverse this.reload() + this.saveDocumentListView() } - get sortDirection(): string { - return this.view.sortDirection + get sortReverse(): boolean { + return this.activeListViewState.sortReverse + } + + get collectionSize(): number { + return this.activeListViewState.collectionSize + } + + get currentPage(): number { + return this.activeListViewState.currentPage + } + + set currentPage(page: number) { + this.activeListViewState.currentPage = page + this.reload() + this.saveDocumentListView() + } + + get documents(): PaperlessDocument[] { + return this.activeListViewState.documents + } + + get selected(): Set { + return this.activeListViewState.selected + } + + setSort(field: string, reverse: boolean) { + this.activeListViewState.sortField = field + this.activeListViewState.sortReverse = reverse + this.reload() + this.saveDocumentListView() } private saveDocumentListView() { - sessionStorage.setItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG, JSON.stringify(this.documentListView)) + if (this._activeSavedViewId == null) { + let savedState: ListViewState = { + collectionSize: this.activeListViewState.collectionSize, + currentPage: this.activeListViewState.currentPage, + filterRules: this.activeListViewState.filterRules, + sortField: this.activeListViewState.sortField, + sortReverse: this.activeListViewState.sortReverse + } + localStorage.setItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG, JSON.stringify(savedState)) + } + } + + quickFilter(filterRules: FilterRule[]) { + this._activeSavedViewId = null + this.activeListViewState.filterRules = filterRules + this.activeListViewState.currentPage = 1 + if (isFullTextFilterRule(filterRules)) { + this.activeListViewState.sortField = "score" + this.activeListViewState.sortReverse = false + } + this.reduceSelectionToFilter() + this.saveDocumentListView() + if (this.router.url == "/documents") { + this.reload() + } else { + this.router.navigate(["documents"]) + } } getLastPage(): number { @@ -185,28 +287,93 @@ export class DocumentListViewService { } updatePageSize() { - let newPageSize = +localStorage.getItem(GENERAL_SETTINGS.DOCUMENT_LIST_SIZE) || GENERAL_SETTINGS.DOCUMENT_LIST_SIZE_DEFAULT + let newPageSize = this.settings.get(SETTINGS_KEYS.DOCUMENT_LIST_SIZE) if (newPageSize != this.currentPageSize) { this.currentPageSize = newPageSize - //this.reload() } } - constructor(private documentService: DocumentService) { - let documentListViewConfigJson = sessionStorage.getItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) + selectNone() { + this.selected.clear() + this.rangeSelectionAnchorIndex = this.lastRangeSelectionToIndex = null + } + + reduceSelectionToFilter() { + if (this.selected.size > 0) { + this.documentService.listAllFilteredIds(this.filterRules).subscribe(ids => { + for (let id of this.selected) { + if (!ids.includes(id)) { + this.selected.delete(id) + } + } + }) + } + } + + selectAll() { + this.documentService.listAllFilteredIds(this.filterRules).subscribe(ids => ids.forEach(id => this.selected.add(id))) + } + + selectPage() { + this.selected.clear() + this.documents.forEach(doc => { + this.selected.add(doc.id) + }) + } + + isSelected(d: PaperlessDocument) { + return this.selected.has(d.id) + } + + toggleSelected(d: PaperlessDocument): void { + if (this.selected.has(d.id)) this.selected.delete(d.id) + else this.selected.add(d.id) + this.rangeSelectionAnchorIndex = this.documentIndexInCurrentView(d.id) + this.lastRangeSelectionToIndex = null + } + + selectRangeTo(d: PaperlessDocument) { + if (this.rangeSelectionAnchorIndex !== null) { + const documentToIndex = this.documentIndexInCurrentView(d.id) + const fromIndex = Math.min(this.rangeSelectionAnchorIndex, documentToIndex) + const toIndex = Math.max(this.rangeSelectionAnchorIndex, documentToIndex) + + if (this.lastRangeSelectionToIndex !== null) { + // revert the old selection + this.documents.slice(Math.min(this.rangeSelectionAnchorIndex, this.lastRangeSelectionToIndex), Math.max(this.rangeSelectionAnchorIndex, this.lastRangeSelectionToIndex) + 1).forEach(d => { + this.selected.delete(d.id) + }) + } + + this.documents.slice(fromIndex, toIndex + 1).forEach(d => { + this.selected.add(d.id) + }) + this.lastRangeSelectionToIndex = documentToIndex + } else { // e.g. shift key but was first click + this.toggleSelected(d) + } + } + + documentIndexInCurrentView(documentID: number): number { + return this.documents.map(d => d.id).indexOf(documentID) + } + + constructor(private documentService: DocumentService, private settings: SettingsService, private router: Router, private route: ActivatedRoute) { + let documentListViewConfigJson = localStorage.getItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) if (documentListViewConfigJson) { try { - this.documentListView = JSON.parse(documentListViewConfigJson) + let savedState: ListViewState = JSON.parse(documentListViewConfigJson) + // Remove null elements from the restored state + Object.keys(savedState).forEach(k => { + if (savedState[k] == null) { + delete savedState[k] + } + }) + //only use restored state attributes instead of defaults if they are not null + let newState = Object.assign(this.defaultListViewState(), savedState) + this.listViewStates.set(null, newState) } catch (e) { - sessionStorage.removeItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) - this.documentListView = null - } - } - if (!this.documentListView) { - this.documentListView = { - filterRules: [], - sortDirection: 'des', - sortField: 'created' + localStorage.removeItem(DOCUMENT_LIST_SERVICE.CURRENT_VIEW_CONFIG) } } } diff --git a/src-ui/src/app/services/open-documents.service.ts b/src-ui/src/app/services/open-documents.service.ts index e37f5db8c..01f168aeb 100644 --- a/src-ui/src/app/services/open-documents.service.ts +++ b/src-ui/src/app/services/open-documents.service.ts @@ -1,6 +1,7 @@ import { Injectable } from '@angular/core'; import { PaperlessDocument } from '../data/paperless-document'; import { OPEN_DOCUMENT_SERVICE } from '../data/storage-keys'; +import { DocumentService } from './rest/document.service'; @Injectable({ providedIn: 'root' @@ -9,7 +10,7 @@ export class OpenDocumentsService { private MAX_OPEN_DOCUMENTS = 5 - constructor() { + constructor(private documentService: DocumentService) { if (sessionStorage.getItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS)) { try { this.openDocuments = JSON.parse(sessionStorage.getItem(OPEN_DOCUMENT_SERVICE.DOCUMENTS)) @@ -22,6 +23,18 @@ export class OpenDocumentsService { private openDocuments: PaperlessDocument[] = [] + refreshDocument(id: number) { + let index = this.openDocuments.findIndex(doc => doc.id == id) + if (index > -1) { + this.documentService.get(id).subscribe(doc => { + this.openDocuments[index] = doc + }, error => { + this.openDocuments.splice(index, 1) + this.save() + }) + } + } + getOpenDocuments(): PaperlessDocument[] { return this.openDocuments } diff --git a/src-ui/src/app/services/rest/abstract-name-filter-service.ts b/src-ui/src/app/services/rest/abstract-name-filter-service.ts new file mode 100644 index 000000000..d605fef49 --- /dev/null +++ b/src-ui/src/app/services/rest/abstract-name-filter-service.ts @@ -0,0 +1,14 @@ +import { ObjectWithId } from 'src/app/data/object-with-id' +import { AbstractPaperlessService } from './abstract-paperless-service' + +export abstract class AbstractNameFilterService extends AbstractPaperlessService { + + listFiltered(page?: number, pageSize?: number, sortField?: string, sortReverse?: boolean, nameFilter?: string) { + let params = {} + if (nameFilter) { + params = {'name__icontains': nameFilter} + } + return this.list(page, pageSize, sortField, sortReverse, params) + } + +} diff --git a/src-ui/src/app/services/rest/abstract-paperless-service.ts b/src-ui/src/app/services/rest/abstract-paperless-service.ts index 3feed320e..8ad1a2141 100644 --- a/src-ui/src/app/services/rest/abstract-paperless-service.ts +++ b/src-ui/src/app/services/rest/abstract-paperless-service.ts @@ -1,5 +1,5 @@ import { HttpClient, HttpParams } from '@angular/common/http' -import { Observable, of, Subject } from 'rxjs' +import { Observable } from 'rxjs' import { map, publishReplay, refCount } from 'rxjs/operators' import { ObjectWithId } from 'src/app/data/object-with-id' import { Results } from 'src/app/data/results' @@ -22,17 +22,15 @@ export abstract class AbstractPaperlessService { return url } - private getOrderingQueryParam(sortField: string, sortDirection: string) { - if (sortField && sortDirection) { - return (sortDirection == 'des' ? '-' : '') + sortField - } else if (sortField) { - return sortField + private getOrderingQueryParam(sortField: string, sortReverse: boolean) { + if (sortField) { + return (sortReverse ? '-' : '') + sortField } else { return null } } - list(page?: number, pageSize?: number, sortField?: string, sortDirection?: string, extraParams?): Observable> { + list(page?: number, pageSize?: number, sortField?: string, sortReverse?: boolean, extraParams?): Observable> { let httpParams = new HttpParams() if (page) { httpParams = httpParams.set('page', page.toString()) @@ -40,7 +38,7 @@ export abstract class AbstractPaperlessService { if (pageSize) { httpParams = httpParams.set('page_size', pageSize.toString()) } - let ordering = this.getOrderingQueryParam(sortField, sortDirection) + let ordering = this.getOrderingQueryParam(sortField, sortReverse) if (ordering) { httpParams = httpParams.set('ordering', ordering) } @@ -54,9 +52,9 @@ export abstract class AbstractPaperlessService { private _listAll: Observable> - listAll(ordering?: string, extraParams?): Observable> { + listAll(sortField?: string, sortReverse?: boolean, extraParams?): Observable> { if (!this._listAll) { - this._listAll = this.list(1, 100000, ordering, extraParams).pipe( + this._listAll = this.list(1, 100000, sortField, sortReverse, extraParams).pipe( publishReplay(1), refCount() ) @@ -76,22 +74,32 @@ export abstract class AbstractPaperlessService { ) } + clearCache() { + this._listAll = null + } + get(id: number): Observable { return this.http.get(this.getResourceUrl(id)) } create(o: T): Observable { - this._listAll = null + this.clearCache() return this.http.post(this.getResourceUrl(), o) } delete(o: T): Observable { - this._listAll = null + this.clearCache() return this.http.delete(this.getResourceUrl(o.id)) } update(o: T): Observable { - this._listAll = null + this.clearCache() return this.http.put(this.getResourceUrl(o.id), o) } -} \ No newline at end of file + + patch(o: T): Observable { + this.clearCache() + return this.http.patch(this.getResourceUrl(o.id), o) + } + +} diff --git a/src-ui/src/app/services/rest/correspondent.service.ts b/src-ui/src/app/services/rest/correspondent.service.ts index a609b7dd8..7eac24971 100644 --- a/src-ui/src/app/services/rest/correspondent.service.ts +++ b/src-ui/src/app/services/rest/correspondent.service.ts @@ -1,12 +1,12 @@ import { HttpClient } from '@angular/common/http'; import { Injectable } from '@angular/core'; import { PaperlessCorrespondent } from 'src/app/data/paperless-correspondent'; -import { AbstractPaperlessService } from './abstract-paperless-service'; +import { AbstractNameFilterService } from './abstract-name-filter-service'; @Injectable({ providedIn: 'root' }) -export class CorrespondentService extends AbstractPaperlessService { +export class CorrespondentService extends AbstractNameFilterService { constructor(http: HttpClient) { super(http, 'correspondents') diff --git a/src-ui/src/app/services/rest/document-type.service.ts b/src-ui/src/app/services/rest/document-type.service.ts index a3ba0d858..4f5b7d0ce 100644 --- a/src-ui/src/app/services/rest/document-type.service.ts +++ b/src-ui/src/app/services/rest/document-type.service.ts @@ -1,12 +1,12 @@ import { HttpClient } from '@angular/common/http'; import { Injectable } from '@angular/core'; import { PaperlessDocumentType } from 'src/app/data/paperless-document-type'; -import { AbstractPaperlessService } from './abstract-paperless-service'; +import { AbstractNameFilterService } from './abstract-name-filter-service'; @Injectable({ providedIn: 'root' }) -export class DocumentTypeService extends AbstractPaperlessService { +export class DocumentTypeService extends AbstractNameFilterService { constructor(http: HttpClient) { super(http, 'document_types') diff --git a/src-ui/src/app/services/rest/document.service.ts b/src-ui/src/app/services/rest/document.service.ts index 81693ec68..c6bc61dc8 100644 --- a/src-ui/src/app/services/rest/document.service.ts +++ b/src-ui/src/app/services/rest/document.service.ts @@ -2,7 +2,7 @@ import { Injectable } from '@angular/core'; import { PaperlessDocument } from 'src/app/data/paperless-document'; import { PaperlessDocumentMetadata } from 'src/app/data/paperless-document-metadata'; import { AbstractPaperlessService } from './abstract-paperless-service'; -import { HttpClient } from '@angular/common/http'; +import { HttpClient, HttpParams } from '@angular/common/http'; import { Observable } from 'rxjs'; import { Results } from 'src/app/data/results'; import { FilterRule } from 'src/app/data/filter-rule'; @@ -10,27 +10,42 @@ import { map } from 'rxjs/operators'; import { CorrespondentService } from './correspondent.service'; import { DocumentTypeService } from './document-type.service'; import { TagService } from './tag.service'; - +import { FILTER_RULE_TYPES } from 'src/app/data/filter-rule-type'; +import { PaperlessDocumentSuggestions } from 'src/app/data/paperless-document-suggestions'; export const DOCUMENT_SORT_FIELDS = [ - { field: "correspondent__name", name: "Correspondent" }, - { field: "document_type__name", name: "Document type" }, - { field: 'title', name: 'Title' }, - { field: 'archive_serial_number', name: 'ASN' }, - { field: 'created', name: 'Created' }, - { field: 'added', name: 'Added' }, - { field: 'modified', name: 'Modified' } + { field: 'archive_serial_number', name: $localize`ASN` }, + { field: "correspondent__name", name: $localize`Correspondent` }, + { field: 'title', name: $localize`Title` }, + { field: "document_type__name", name: $localize`Document type` }, + { field: 'created', name: $localize`Created` }, + { field: 'added', name: $localize`Added` }, + { field: 'modified', name: $localize`Modified` } ] -export const SORT_DIRECTION_ASCENDING = "asc" -export const SORT_DIRECTION_DESCENDING = "des" +export const DOCUMENT_SORT_FIELDS_FULLTEXT = [ + ...DOCUMENT_SORT_FIELDS, + { field: 'score', name: $localize`:Score is a value returned by the full text search engine and specifies how well a result matches the given query:Search score` } +] +export interface SelectionDataItem { + id: number + document_count: number +} + +export interface SelectionData { + selected_correspondents: SelectionDataItem[] + selected_tags: SelectionDataItem[] + selected_document_types: SelectionDataItem[] +} @Injectable({ providedIn: 'root' }) export class DocumentService extends AbstractPaperlessService { + private _searchQuery: string + constructor(http: HttpClient, private correspondentService: CorrespondentService, private documentTypeService: DocumentTypeService, private tagService: TagService) { super(http, 'documents') } @@ -39,10 +54,13 @@ export class DocumentService extends AbstractPaperlessService if (filterRules) { let params = {} for (let rule of filterRules) { - if (rule.type.multi) { - params[rule.type.filtervar] = params[rule.type.filtervar] ? params[rule.type.filtervar] + "," + rule.value : rule.value + let ruleType = FILTER_RULE_TYPES.find(t => t.id == rule.rule_type) + if (ruleType.multi) { + params[ruleType.filtervar] = params[ruleType.filtervar] ? params[ruleType.filtervar] + "," + rule.value : rule.value + } else if (ruleType.isnull_filtervar && rule.value == null) { + params[ruleType.isnull_filtervar] = true } else { - params[rule.type.filtervar] = rule.value + params[ruleType.filtervar] = rule.value } } return params @@ -64,8 +82,8 @@ export class DocumentService extends AbstractPaperlessService return doc } - list(page?: number, pageSize?: number, sortField?: string, sortDirection?: string, filterRules?: FilterRule[]): Observable> { - return super.list(page, pageSize, sortField, sortDirection, this.filterRulesToQueryParams(filterRules)).pipe( + listFiltered(page?: number, pageSize?: number, sortField?: string, sortReverse?: boolean, filterRules?: FilterRule[], extraParams = {}): Observable> { + return this.list(page, pageSize, sortField, sortReverse, Object.assign(extraParams, this.filterRulesToQueryParams(filterRules))).pipe( map(results => { results.results.forEach(doc => this.addObservablesToDocument(doc)) return results @@ -73,8 +91,15 @@ export class DocumentService extends AbstractPaperlessService ) } + listAllFilteredIds(filterRules?: FilterRule[]): Observable { + return this.listFiltered(1, 100000, null, null, filterRules, {"fields": "id"}).pipe( + map(response => response.results.map(doc => doc.id)) + ) + } + getPreviewUrl(id: number, original: boolean = false): string { let url = this.getResourceUrl(id, 'preview') + if (this._searchQuery) url += `#search="${this._searchQuery}"` if (original) { url += "?original=true" } @@ -101,4 +126,28 @@ export class DocumentService extends AbstractPaperlessService return this.http.get(this.getResourceUrl(id, 'metadata')) } + bulkEdit(ids: number[], method: string, args: any) { + return this.http.post(this.getResourceUrl(null, 'bulk_edit'), { + 'documents': ids, + 'method': method, + 'parameters': args + }) + } + + getSelectionData(ids: number[]): Observable { + return this.http.post(this.getResourceUrl(null, 'selection_data'), {"documents": ids}) + } + + getSuggestions(id: number): Observable { + return this.http.get(this.getResourceUrl(id, 'suggestions')) + } + + bulkDownload(ids: number[], content="both") { + return this.http.post(this.getResourceUrl(null, 'bulk_download'), {"documents": ids, "content": content}, { responseType: 'blob' }) + } + + public set searchQuery(query: string) { + this._searchQuery = query + } + } diff --git a/src-ui/src/app/services/rest/log.service.ts b/src-ui/src/app/services/rest/log.service.ts index 797d9b6b9..3abf2fb6f 100644 --- a/src-ui/src/app/services/rest/log.service.ts +++ b/src-ui/src/app/services/rest/log.service.ts @@ -1,14 +1,21 @@ import { HttpClient } from '@angular/common/http'; import { Injectable } from '@angular/core'; -import { PaperlessLog } from 'src/app/data/paperless-log'; -import { AbstractPaperlessService } from './abstract-paperless-service'; +import { Observable } from 'rxjs'; +import { environment } from 'src/environments/environment'; @Injectable({ providedIn: 'root' }) -export class LogService extends AbstractPaperlessService { +export class LogService { - constructor(http: HttpClient) { - super(http, 'logs') + constructor(private http: HttpClient) { + } + + list(): Observable { + return this.http.get(`${environment.apiBaseUrl}logs/`) + } + + get(id: string): Observable { + return this.http.get(`${environment.apiBaseUrl}logs/${id}/`) } } diff --git a/src-ui/src/app/services/rest/saved-view.service.spec.ts b/src-ui/src/app/services/rest/saved-view.service.spec.ts new file mode 100644 index 000000000..588cf6347 --- /dev/null +++ b/src-ui/src/app/services/rest/saved-view.service.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { SavedViewService } from './saved-view.service'; + +describe('SavedViewService', () => { + let service: SavedViewService; + + beforeEach(() => { + TestBed.configureTestingModule({}); + service = TestBed.inject(SavedViewService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/services/rest/saved-view.service.ts b/src-ui/src/app/services/rest/saved-view.service.ts new file mode 100644 index 000000000..c765de7f4 --- /dev/null +++ b/src-ui/src/app/services/rest/saved-view.service.ts @@ -0,0 +1,59 @@ +import { HttpClient } from '@angular/common/http'; +import { Injectable } from '@angular/core'; +import { combineLatest, Observable } from 'rxjs'; +import { tap } from 'rxjs/operators'; +import { PaperlessSavedView } from 'src/app/data/paperless-saved-view'; +import { AbstractPaperlessService } from './abstract-paperless-service'; + +@Injectable({ + providedIn: 'root' +}) +export class SavedViewService extends AbstractPaperlessService { + + constructor(http: HttpClient) { + super(http, 'saved_views') + this.reload() + } + + private reload() { + this.listAll().subscribe(r => this.savedViews = r.results) + } + + private savedViews: PaperlessSavedView[] = [] + + get allViews() { + return this.savedViews + } + + get sidebarViews() { + return this.savedViews.filter(v => v.show_in_sidebar) + } + + get dashboardViews() { + return this.savedViews.filter(v => v.show_on_dashboard) + } + + create(o: PaperlessSavedView) { + return super.create(o).pipe( + tap(() => this.reload()) + ) + } + + update(o: PaperlessSavedView) { + return super.update(o).pipe( + tap(() => this.reload()) + ) + } + + patchMany(objects: PaperlessSavedView[]): Observable { + return combineLatest(objects.map(o => super.patch(o))).pipe( + tap(() => this.reload()) + ) + } + + delete(o: PaperlessSavedView) { + return super.delete(o).pipe( + tap(() => this.reload()) + ) + } +} diff --git a/src-ui/src/app/services/rest/search.service.ts b/src-ui/src/app/services/rest/search.service.ts index b19a55769..c14a5c58f 100644 --- a/src-ui/src/app/services/rest/search.service.ts +++ b/src-ui/src/app/services/rest/search.service.ts @@ -2,8 +2,6 @@ import { HttpClient, HttpParams } from '@angular/common/http'; import { Injectable } from '@angular/core'; import { Observable } from 'rxjs'; import { map } from 'rxjs/operators'; -import { PaperlessDocument } from 'src/app/data/paperless-document'; -import { SearchResult } from 'src/app/data/search-result'; import { environment } from 'src/environments/environment'; import { DocumentService } from './document.service'; @@ -12,23 +10,10 @@ import { DocumentService } from './document.service'; providedIn: 'root' }) export class SearchService { - - constructor(private http: HttpClient, private documentService: DocumentService) { } - search(query: string, page?: number): Observable { - let httpParams = new HttpParams().set('query', query) - if (page) { - httpParams = httpParams.set('page', page.toString()) - } - return this.http.get(`${environment.apiBaseUrl}search/`, {params: httpParams}).pipe( - map(result => { - result.results.forEach(hit => this.documentService.addObservablesToDocument(hit.document)) - return result - }) - ) - } + constructor(private http: HttpClient) { } autocomplete(term: string): Observable { return this.http.get(`${environment.apiBaseUrl}search/autocomplete/`, {params: new HttpParams().set('term', term)}) } -} +} diff --git a/src-ui/src/app/services/rest/tag.service.ts b/src-ui/src/app/services/rest/tag.service.ts index b4151dbb9..7bc55b0c9 100644 --- a/src-ui/src/app/services/rest/tag.service.ts +++ b/src-ui/src/app/services/rest/tag.service.ts @@ -1,12 +1,12 @@ import { HttpClient } from '@angular/common/http'; import { Injectable } from '@angular/core'; import { PaperlessTag } from 'src/app/data/paperless-tag'; -import { AbstractPaperlessService } from './abstract-paperless-service'; +import { AbstractNameFilterService } from './abstract-name-filter-service'; @Injectable({ providedIn: 'root' }) -export class TagService extends AbstractPaperlessService { +export class TagService extends AbstractNameFilterService { constructor(http: HttpClient) { super(http, 'tags') diff --git a/src-ui/src/app/services/saved-view-config.service.spec.ts b/src-ui/src/app/services/saved-view-config.service.spec.ts deleted file mode 100644 index c67affead..000000000 --- a/src-ui/src/app/services/saved-view-config.service.spec.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { TestBed } from '@angular/core/testing'; - -import { SavedViewConfigService } from './saved-view-config.service'; - -describe('SavedViewConfigService', () => { - let service: SavedViewConfigService; - - beforeEach(() => { - TestBed.configureTestingModule({}); - service = TestBed.inject(SavedViewConfigService); - }); - - it('should be created', () => { - expect(service).toBeTruthy(); - }); -}); diff --git a/src-ui/src/app/services/saved-view-config.service.ts b/src-ui/src/app/services/saved-view-config.service.ts deleted file mode 100644 index 41c28216b..000000000 --- a/src-ui/src/app/services/saved-view-config.service.ts +++ /dev/null @@ -1,66 +0,0 @@ -import { Injectable } from '@angular/core'; -import { v4 as uuidv4 } from 'uuid'; -import { SavedViewConfig } from '../data/saved-view-config'; - -@Injectable({ - providedIn: 'root' -}) -export class SavedViewConfigService { - - constructor() { - let savedConfigs = localStorage.getItem('saved-view-config-service:savedConfigs') - if (savedConfigs) { - try { - this.configs = JSON.parse(savedConfigs) - } catch (e) { - this.configs = [] - } - } - } - - private configs: SavedViewConfig[] = [] - - getConfigs(): SavedViewConfig[] { - return this.configs - } - - getDashboardConfigs(): SavedViewConfig[] { - return this.configs.filter(sf => sf.showInDashboard) - } - - getSideBarConfigs(): SavedViewConfig[] { - return this.configs.filter(sf => sf.showInSideBar) - } - - getConfig(id: string): SavedViewConfig { - return this.configs.find(sf => sf.id == id) - } - - newConfig(config: SavedViewConfig) { - config.id = uuidv4() - this.configs.push(config) - - this.save() - } - - updateConfig(config: SavedViewConfig) { - let savedConfig = this.configs.find(c => c.id == config.id) - if (savedConfig) { - Object.assign(savedConfig, config) - this.save() - } - } - - private save() { - localStorage.setItem('saved-view-config-service:savedConfigs', JSON.stringify(this.configs)) - } - - deleteConfig(config: SavedViewConfig) { - let index = this.configs.findIndex(vc => vc.id == config.id) - if (index != -1) { - this.configs.splice(index, 1) - this.save() - } - - } -} diff --git a/src-ui/src/app/services/settings.service.spec.ts b/src-ui/src/app/services/settings.service.spec.ts new file mode 100644 index 000000000..359cb6b7a --- /dev/null +++ b/src-ui/src/app/services/settings.service.spec.ts @@ -0,0 +1,16 @@ +import { TestBed } from '@angular/core/testing'; + +import { SettingsService } from './settings.service'; + +describe('SettingsService', () => { + let service: SettingsService; + + beforeEach(() => { + TestBed.configureTestingModule({}); + service = TestBed.inject(SettingsService); + }); + + it('should be created', () => { + expect(service).toBeTruthy(); + }); +}); diff --git a/src-ui/src/app/services/settings.service.ts b/src-ui/src/app/services/settings.service.ts new file mode 100644 index 000000000..7b467f6ee --- /dev/null +++ b/src-ui/src/app/services/settings.service.ts @@ -0,0 +1,169 @@ +import { DOCUMENT } from '@angular/common'; +import { Inject, Injectable, LOCALE_ID, Renderer2, RendererFactory2 } from '@angular/core'; +import { Meta } from '@angular/platform-browser'; +import { CookieService } from 'ngx-cookie-service'; + +export interface PaperlessSettings { + key: string + type: string + default: any +} + +export interface LanguageOption { + code: string + name: string + englishName?: string + + /** + * A date format string for use by the date selectors. MUST contain 'yyyy', 'mm' and 'dd'. + */ + dateInputFormat?: string +} + +export const SETTINGS_KEYS = { + BULK_EDIT_CONFIRMATION_DIALOGS: 'general-settings:bulk-edit:confirmation-dialogs', + BULK_EDIT_APPLY_ON_CLOSE: 'general-settings:bulk-edit:apply-on-close', + DOCUMENT_LIST_SIZE: 'general-settings:documentListSize', + DARK_MODE_USE_SYSTEM: 'general-settings:dark-mode:use-system', + DARK_MODE_ENABLED: 'general-settings:dark-mode:enabled', + DARK_MODE_THUMB_INVERTED: 'general-settings:dark-mode:thumb-inverted', + USE_NATIVE_PDF_VIEWER: 'general-settings:document-details:native-pdf-viewer', + DATE_LOCALE: 'general-settings:date-display:date-locale', + DATE_FORMAT: 'general-settings:date-display:date-format', + NOTIFICATIONS_CONSUMER_NEW_DOCUMENT: 'general-settings:notifications:consumer-new-documents', + NOTIFICATIONS_CONSUMER_SUCCESS: 'general-settings:notifications:consumer-success', + NOTIFICATIONS_CONSUMER_FAILED: 'general-settings:notifications:consumer-failed', + NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD: 'general-settings:notifications:consumer-suppress-on-dashboard', +} + +const SETTINGS: PaperlessSettings[] = [ + {key: SETTINGS_KEYS.BULK_EDIT_CONFIRMATION_DIALOGS, type: "boolean", default: true}, + {key: SETTINGS_KEYS.BULK_EDIT_APPLY_ON_CLOSE, type: "boolean", default: false}, + {key: SETTINGS_KEYS.DOCUMENT_LIST_SIZE, type: "number", default: 50}, + {key: SETTINGS_KEYS.DARK_MODE_USE_SYSTEM, type: "boolean", default: true}, + {key: SETTINGS_KEYS.DARK_MODE_ENABLED, type: "boolean", default: false}, + {key: SETTINGS_KEYS.DARK_MODE_THUMB_INVERTED, type: "boolean", default: true}, + {key: SETTINGS_KEYS.USE_NATIVE_PDF_VIEWER, type: "boolean", default: false}, + {key: SETTINGS_KEYS.DATE_LOCALE, type: "string", default: ""}, + {key: SETTINGS_KEYS.DATE_FORMAT, type: "string", default: "mediumDate"}, + {key: SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_NEW_DOCUMENT, type: "boolean", default: true}, + {key: SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUCCESS, type: "boolean", default: true}, + {key: SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_FAILED, type: "boolean", default: true}, + {key: SETTINGS_KEYS.NOTIFICATIONS_CONSUMER_SUPPRESS_ON_DASHBOARD, type: "boolean", default: true}, +] + +@Injectable({ + providedIn: 'root' +}) +export class SettingsService { + + private renderer: Renderer2; + + constructor( + private rendererFactory: RendererFactory2, + @Inject(DOCUMENT) private document, + private cookieService: CookieService, + private meta: Meta, + @Inject(LOCALE_ID) private localeId: string + ) { + this.renderer = rendererFactory.createRenderer(null, null); + + this.updateDarkModeSettings() + } + + updateDarkModeSettings(): void { + let darkModeUseSystem = this.get(SETTINGS_KEYS.DARK_MODE_USE_SYSTEM) + let darkModeEnabled = this.get(SETTINGS_KEYS.DARK_MODE_ENABLED) + + if (darkModeUseSystem) { + this.renderer.addClass(this.document.body, 'color-scheme-system') + this.renderer.removeClass(this.document.body, 'color-scheme-dark') + } else { + this.renderer.removeClass(this.document.body, 'color-scheme-system') + darkModeEnabled ? this.renderer.addClass(this.document.body, 'color-scheme-dark') : this.renderer.removeClass(this.document.body, 'color-scheme-dark') + } + + } + + getLanguageOptions(): LanguageOption[] { + return [ + {code: "en-us", name: $localize`English (US)`, englishName: "English (US)", dateInputFormat: "mm/dd/yyyy"}, + {code: "en-gb", name: $localize`English (GB)`, englishName: "English (GB)", dateInputFormat: "dd/mm/yyyy"}, + {code: "de-de", name: $localize`German`, englishName: "German", dateInputFormat: "dd.mm.yyyy"}, + {code: "nl-nl", name: $localize`Dutch`, englishName: "Dutch", dateInputFormat: "dd-mm-yyyy"}, + {code: "fr-fr", name: $localize`French`, englishName: "French", dateInputFormat: "dd/mm/yyyy"}, + {code: "pt-pt", name: $localize`Portuguese`, englishName: "Portuguese", dateInputFormat: "dd/mm/yyyy"}, + {code: "pt-br", name: $localize`Portuguese (Brazil)`, englishName: "Portuguese (Brazil)", dateInputFormat: "dd/mm/yyyy"}, + {code: "it-it", name: $localize`Italian`, englishName: "Italian", dateInputFormat: "dd/mm/yyyy"}, + {code: "ro-ro", name: $localize`Romanian`, englishName: "Romanian", dateInputFormat: "dd.mm.yyyy"}, + {code: "ru-ru", name: $localize`Russian`, englishName: "Russian", dateInputFormat: "dd.mm.yyyy"}, + {code: "es-es", name: $localize`Spanish`, englishName: "Spanish", dateInputFormat: "dd/mm/yyyy"}, + {code: "pl-pl", name: $localize`Polish`, englishName: "Polish", dateInputFormat: "dd.mm.yyyy"}, + {code: "sv-se", name: $localize`Swedish`, englishName: "Swedish", dateInputFormat: "yyyy-mm-dd"}, + {code: "lb-lu", name: $localize`Luxembourgish`, englishName: "Luxembourgish", dateInputFormat: "dd.mm.yyyy"} + ] + } + + getDateLocaleOptions(): LanguageOption[] { + let isoOption: LanguageOption = {code: "iso-8601", name: $localize`ISO 8601`, dateInputFormat: "yyyy-mm-dd"} + return [isoOption].concat(this.getLanguageOptions()) + } + + private getLanguageCookieName() { + let prefix = "" + if (this.meta.getTag('name=cookie_prefix')) { + prefix = this.meta.getTag('name=cookie_prefix').content + } + return `${prefix || ''}django_language` + } + + getLanguage(): string { + return this.cookieService.get(this.getLanguageCookieName()) + } + + setLanguage(language: string) { + if (language) { + this.cookieService.set(this.getLanguageCookieName(), language) + } else { + this.cookieService.delete(this.getLanguageCookieName()) + } + } + + getLocalizedDateInputFormat(): string { + let dateLocale = this.get(SETTINGS_KEYS.DATE_LOCALE) || this.getLanguage() || this.localeId.toLowerCase() + return this.getDateLocaleOptions().find(o => o.code == dateLocale)?.dateInputFormat || "yyyy-mm-dd" + } + + get(key: string): any { + let setting = SETTINGS.find(s => s.key == key) + + if (!setting) { + return null + } + + let value = localStorage.getItem(key) + + if (value != null) { + switch (setting.type) { + case "boolean": + return JSON.parse(value) + case "number": + return +value + case "string": + return value + default: + return value + } + } else { + return setting.default + } + } + + set(key: string, value: any) { + localStorage.setItem(key, value.toString()) + } + + unset(key: string) { + localStorage.removeItem(key) + } +} diff --git a/src-ui/src/app/services/toast.service.ts b/src-ui/src/app/services/toast.service.ts index a3ce060a9..fc522e2df 100644 --- a/src-ui/src/app/services/toast.service.ts +++ b/src-ui/src/app/services/toast.service.ts @@ -1,30 +1,17 @@ import { Injectable } from '@angular/core'; import { Subject, zip } from 'rxjs'; -export class Toast { - - static make(title: string, content: string, classname?: string, delay?: number): Toast { - let t = new Toast() - t.title = title - t.content = content - t.classname = classname - if (delay) { - t.delay = delay - } - return t - } - - static makeError(content: string) { - return Toast.make("Error", content, null, 10000) - } +export interface Toast { title: string - classname: string - content: string - delay: number = 5000 + delay: number + + action?: any + + actionName?: string } @@ -39,11 +26,19 @@ export class ToastService { private toastsSubject: Subject = new Subject() - showToast(toast: Toast) { + show(toast: Toast) { this.toasts.push(toast) this.toastsSubject.next(this.toasts) } + showError(content: string, delay: number = 10000) { + this.show({title: $localize`Error`, content: content, delay: delay}) + } + + showInfo(content: string, delay: number = 5000) { + this.show({title: $localize`Information`, content: content, delay: delay}) + } + closeToast(toast: Toast) { let index = this.toasts.findIndex(t => t == toast) if (index > -1) { diff --git a/src-ui/src/app/utils/color.ts b/src-ui/src/app/utils/color.ts new file mode 100644 index 000000000..d320cc2b9 --- /dev/null +++ b/src-ui/src/app/utils/color.ts @@ -0,0 +1,48 @@ + +function componentToHex(c) { + var hex = Math.floor(c).toString(16) + return hex.length == 1 ? "0" + hex : hex +} + +/** + * https://axonflux.com/handy-rgb-to-hsl-and-rgb-to-hsv-color-model-c + * + * Converts an HSL color value to RGB. Conversion formula + * adapted from http://en.wikipedia.org/wiki/HSL_color_space. + * Assumes h, s, and l are contained in the set [0, 1] and + * returns r, g, and b in the set [0, 255]. + * + * @param Number h The hue + * @param Number s The saturation + * @param Number l The lightness + * @return Array The RGB representation + */ +function hslToRgb(h, s, l){ + var r, g, b + + if(s == 0){ + r = g = b = l // achromatic + }else{ + function hue2rgb(p, q, t){ + if(t < 0) t += 1 + if(t > 1) t -= 1 + if(t < 1/6) return p + (q - p) * 6 * t + if(t < 1/2) return q + if(t < 2/3) return p + (q - p) * (2/3 - t) * 6 + return p + } + + var q = l < 0.5 ? l * (1 + s) : l + s - l * s + var p = 2 * l - q + r = hue2rgb(p, q, h + 1/3) + g = hue2rgb(p, q, h) + b = hue2rgb(p, q, h - 1/3) + } + + return [r * 255, g * 255, b * 255] +} + +export function randomColor() { + let rgb = hslToRgb(Math.random(), 0.6, Math.random() * 0.4 + 0.4) + return `#${componentToHex(rgb[0])}${componentToHex(rgb[1])}${componentToHex(rgb[2])}` +} diff --git a/src-ui/src/app/utils/ngb-date-parser-formatter.ts b/src-ui/src/app/utils/ngb-date-parser-formatter.ts new file mode 100644 index 000000000..07d31a525 --- /dev/null +++ b/src-ui/src/app/utils/ngb-date-parser-formatter.ts @@ -0,0 +1,59 @@ +import { Injectable } from "@angular/core" +import { NgbDateParserFormatter, NgbDateStruct } from "@ng-bootstrap/ng-bootstrap" +import { SettingsService } from "../services/settings.service" + +@Injectable() +export class LocalizedDateParserFormatter extends NgbDateParserFormatter { + + constructor(private settings: SettingsService) { + super() + } + + private getDateInputFormat() { + return this.settings.getLocalizedDateInputFormat() + } + + /** + * This constructs a regular expression from a date input format which is then + * used to parse dates. + */ + private getDateParseRegex() { + return new RegExp( + "^" + this.getDateInputFormat() + .replace('dd', '(?[0-9]+)') + .replace('mm', '(?[0-9]+)') + .replace('yyyy', '(?[0-9]+)') + .split('.').join('\\.\\s*') + "$" // allow whitespace(s) after dot (specific for German) + ) + } + + parse(value: string): NgbDateStruct | null { + let match = this.getDateParseRegex().exec(value) + if (match) { + let dateStruct = { + day: +match.groups.day, + month: +match.groups.month, + year: +match.groups.year + } + if (dateStruct.year <= (new Date().getFullYear() - 2000)) { + dateStruct.year += 2000 + } else if (dateStruct.year < 100) { + dateStruct.year += 1900 + } + return dateStruct + } else { + return null + } + } + + format(date: NgbDateStruct | null): string { + if (date) { + return this.getDateInputFormat() + .replace('dd', date.day.toString().padStart(2, '0')) + .replace('mm', date.month.toString().padStart(2, '0')) + .replace('yyyy', date.year.toString().padStart(4, '0')) + } else { + return null + } + } +} diff --git a/src-ui/src/app/utils/ngb-iso-date-adapter.ts b/src-ui/src/app/utils/ngb-iso-date-adapter.ts new file mode 100644 index 000000000..e43602a16 --- /dev/null +++ b/src-ui/src/app/utils/ngb-iso-date-adapter.ts @@ -0,0 +1,27 @@ +import { Injectable } from "@angular/core"; +import { NgbDateAdapter, NgbDateStruct } from "@ng-bootstrap/ng-bootstrap"; + +@Injectable() +export class ISODateAdapter extends NgbDateAdapter { + + fromModel(value: string | null): NgbDateStruct | null { + if (value) { + let date = new Date(value) + return { + day : date.getDate(), + month : date.getMonth() + 1, + year : date.getFullYear() + } + } else { + return null + } + } + + toModel(date: NgbDateStruct | null): string | null { + if (date) { + return date.year.toString().padStart(4, '0') + "-" + date.month.toString().padStart(2, '0') + "-" + date.day.toString().padStart(2, '0') + } else { + return null + } + } +} diff --git a/src-ui/src/app/utils/ngb-iso-date-time-adapter.ts b/src-ui/src/app/utils/ngb-iso-date-time-adapter.ts new file mode 100644 index 000000000..21a97a19a --- /dev/null +++ b/src-ui/src/app/utils/ngb-iso-date-time-adapter.ts @@ -0,0 +1,23 @@ +import { Injectable } from "@angular/core"; +import { NgbDateAdapter, NgbDateStruct } from "@ng-bootstrap/ng-bootstrap"; + +@Injectable() +export class ISODateTimeAdapter extends NgbDateAdapter { + + fromModel(value: string | null): NgbDateStruct | null { + if (value) { + let date = new Date(value) + return { + day : date.getDate(), + month : date.getMonth() + 1, + year : date.getFullYear() + } + } else { + return null + } + } + + toModel(date: NgbDateStruct | null): string | null { + return date ? new Date(date.year, date.month - 1, date.day).toISOString() : null + } +} diff --git a/src-ui/src/apple-touch-icon.png b/src-ui/src/apple-touch-icon.png new file mode 100644 index 000000000..46388a873 Binary files /dev/null and b/src-ui/src/apple-touch-icon.png differ diff --git a/src-ui/src/assets/logo-dark-notext.svg b/src-ui/src/assets/logo-dark-notext.svg index 38ca9e700..74eb142c8 100644 --- a/src-ui/src/assets/logo-dark-notext.svg +++ b/src-ui/src/assets/logo-dark-notext.svg @@ -1,69 +1,19 @@ - - - - - - - - image/svg+xml - - - - - - - - - - - + + + + + + + + + + diff --git a/src-ui/src/assets/logo-dark.svg b/src-ui/src/assets/logo-dark.svg index bc8ba2b80..111d8d32f 100644 --- a/src-ui/src/assets/logo-dark.svg +++ b/src-ui/src/assets/logo-dark.svg @@ -1,93 +1,5 @@ - - - - - - image/svg+xml - - - - - - - - - - - - - - - - - - - - - + + + + diff --git a/src-ui/src/assets/logo-white-notext.svg b/src-ui/src/assets/logo-white-notext.svg new file mode 100644 index 000000000..38ca9e700 --- /dev/null +++ b/src-ui/src/assets/logo-white-notext.svg @@ -0,0 +1,69 @@ + + + + + + + + image/svg+xml + + + + + + + + + + + + diff --git a/src-ui/src/assets/save-filter.png b/src-ui/src/assets/save-filter.png index dcaa41714..0f011f812 100644 Binary files a/src-ui/src/assets/save-filter.png and b/src-ui/src/assets/save-filter.png differ diff --git a/src-ui/src/environments/environment.prod.ts b/src-ui/src/environments/environment.prod.ts index 09154dfca..67c4576aa 100644 --- a/src-ui/src/environments/environment.prod.ts +++ b/src-ui/src/environments/environment.prod.ts @@ -1,5 +1,12 @@ +const base_url = new URL(document.baseURI) + export const environment = { production: true, - apiBaseUrl: "/api/", - appTitle: "Paperless-ng" + apiBaseUrl: document.baseURI + "api/", + apiVersion: "2", + appTitle: "Paperless-ng", + version: "1.5.0", + webSocketHost: window.location.host, + webSocketProtocol: (window.location.protocol == "https:" ? "wss:" : "ws:"), + webSocketBaseUrl: base_url.pathname + "ws/", }; diff --git a/src-ui/src/environments/environment.ts b/src-ui/src/environments/environment.ts index 5e4b148dc..3b8a02e37 100644 --- a/src-ui/src/environments/environment.ts +++ b/src-ui/src/environments/environment.ts @@ -5,7 +5,12 @@ export const environment = { production: false, apiBaseUrl: "http://localhost:8000/api/", - appTitle: "DEVELOPMENT P-NG" + apiVersion: "2", + appTitle: "Paperless-ng", + version: "DEVELOPMENT", + webSocketHost: "localhost:8000", + webSocketProtocol: "ws:", + webSocketBaseUrl: "/ws/", }; /* diff --git a/src-ui/src/index.html b/src-ui/src/index.html index f82399ce6..4308a6b16 100644 --- a/src-ui/src/index.html +++ b/src-ui/src/index.html @@ -5,9 +5,13 @@ Paperless-ng + + + + - + diff --git a/src-ui/src/locale/messages.cs_CZ.xlf b/src-ui/src/locale/messages.cs_CZ.xlf new file mode 100644 index 000000000..dd55e772a --- /dev/null +++ b/src-ui/src/locale/messages.cs_CZ.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokumenty + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + View "" created successfully. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Select + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Select none + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Select page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Select all + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Save as... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Title + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Document type + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Created + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Added + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirm delete + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + The files for this document will be deleted permanently. This operation cannot be undone. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Delete document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Odstranit + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + More like this + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Close + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Content + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Discard + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Save + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + of + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archive serial number + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date created + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Date modified + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date added + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filename + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Original file size + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Original mime type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archive MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archive file size + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Original document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archived document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Save & next + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , welcome to Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welcome to Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Do you really want to delete the tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Create + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter by: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Color + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matching + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Document count + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Akce + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokumenty + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edit + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Do you really want to delete the document type ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Document types + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Settings saved successfully. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error while storing settings on server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Settings + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + General settings + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Saved views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Items per page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Bulk editing + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Show confirmation dialogs + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Deleting documents will always ask for confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Apply on close + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appears on + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Show on dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Show in sidebar + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No saved views defined. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Not Found + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Do you really want to delete the correspondent ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondents + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Last correspondence + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirm + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancel + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Create new correspondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edit correspondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matching algorithm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Case insensitive + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Create new tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edit tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tags are automatically assigned to all consumed documents. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Create new document type + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edit document type + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logout + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Manage + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documents + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Close all + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Title: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Title + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Title & content + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Advanced search + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + More like + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filter tags + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filter correspondents + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filter document types + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Not assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Apply + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Last 7 days + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Last month + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Last 3 months + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Last year + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + After + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Before + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Clear + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + View + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filter by correspondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filter by tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Score: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Created: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Added: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modified: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" and "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + and "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirm tags assignment + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirm correspondent assignment + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirm document type assignment + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Delete confirm + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + This operation cannot be undone. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Delete document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Select: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edit: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Save current view + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Show all + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistics + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Upload new documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Drop documents here or + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Browse files + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + First steps + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is running! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Select + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Please select an object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Searching document with asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Yes + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (no title) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Němčina + + + Dutch + + src/app/services/settings.service.ts + 93 + + Holandština + + + French + + src/app/services/settings.service.ts + 94 + + Francouzština + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portuguese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 99 + + Russian + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanish + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Document type + + + Created + + src/app/services/rest/document.service.ts + 21 + + Created + + + Added + + src/app/services/rest/document.service.ts + 22 + + Added + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modified + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Create new item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edit item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Could not save element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatický + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Do you really want to delete this element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Associated documents will not be deleted. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Odstranit + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Jakékoliv slovo + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + Všechna slova + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Přesná shoda + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regulární výraz + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy slovo + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/locale/messages.de_DE.xlf b/src-ui/src/locale/messages.de_DE.xlf new file mode 100644 index 000000000..fe004f15b --- /dev/null +++ b/src-ui/src/locale/messages.de_DE.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Dokument hinzugefügt + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Das Dokument wurde zu Paperless hinzugefügt. + + + Open document + + src/app/app.component.ts + 51 + + Dokument öffnen + + + Could not add : + + src/app/app.component.ts + 59 + + Konnte nicht hinzufügen: + + + New document detected + + src/app/app.component.ts + 65 + + Neues Dokument erkannt + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Dokument wird von Paperless verarbeitet. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokumente + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Ansicht "" erfolgreich gespeichert. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Ansicht "" erfolgreich erstellt. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Auswählen + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Nichts auswählen + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Seite auswählen + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Alles auswählen + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sortieren + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Ansicht + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Speichern als... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + "" speichern + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 { von 1 Dokumente ausgewählt} other { von Dokumente ausgewählt}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Ein Dokument} other { Dokumente}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (gefiltert) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Korrespondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titel + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Dokumenttyp + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Ausgestellt + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Hinzugefügt + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Löschen bestätigen + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Möchten Sie das Dokument "" wirklich löschen? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Die Dateien dieses Dokuments werden permanent gelöscht. Diese Aktion kann nicht rückgängig gemacht werden. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Dokument löschen + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Fehler beim Löschen des Dokuments: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Löschen + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Herunterladen + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Ähnliche Dokumente + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Schließen + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Inhalt + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadaten + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Verwerfen + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Speichern + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Seite + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + von + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Original herunterladen + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archiv-Seriennummer + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Ausgestellt am + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Geändert am + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Hinzugefügt am + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media-Dateiname + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + MD5-Prüfsumme Original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Dateigröße Original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + MIME-Typ Original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + MD5-Prüfsumme Archiv + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Dateigröße Archiv + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadaten Original + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadaten Archiv + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Speichern & weiter + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hallo , willkommen zu Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Willkommen zu Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Startseite + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Möchten Sie das Tag "" wirklich löschen? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Erstellen + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtern nach: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Farbe + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Zuweisung + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Anzahl Dokumente + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Aktionen + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokumente + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Bearbeiten + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Möchten Sie den Dokumenttyp "" wirklich löschen? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Dokumenttypen + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Protokoll + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Gespeicherte Ansicht "" gelöscht. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Einstellungen erfolgreich gespeichert. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Benutze Systemsprache + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Benutze Datumsformat der Anzeigesprache + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Fehler beim Speichern der Einstellungen auf dem Server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Einstellungen + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Allgemeine Einstellungen + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Benachrichtigungen + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Gespeicherte Ansichten + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Erscheinungsbild + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Anzeigesprache + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Nachdem Sie eine neue Sprache ausgewählt haben, müssen Sie die Seite neu laden. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Datumsanzeige + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Datumsformat + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Kurz: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Mittel: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Lang: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Dokumente pro Seite + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Dokumenteditor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Benutze PDF-Betrachter des Web Browsers + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Der integrierte PDF-Betrachter des Web-Browsers ist in der Regel schneller bei der Anzeige besonders großer Dokumente, funktioniert aber nicht in allen Browsern. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dunkler Modus + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Benutze Systemeinstellungen + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Dunklen Modus aktivieren + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Miniaturbilder im dunklen Modus invertieren + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Massenbearbeitung + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Bestätigungsdialoge anzeigen + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Beim Löschen von Dokumenten wird immer nach einer Bestätigung gefragt. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Anwenden beim Schließen + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Dokumentverarbeitung + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Zeige Benachrichtigungen wenn neue Dokumente erkannt werden + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Zeige Benachrichtigungen wenn neue Dokumente erfolgreich hinzugefügt wurden + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Zeige Benachrichtigungen wenn Dokumente nicht hinzugefügt werden konnten + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Unterdrücke Benachrichtigungen auf der Startseite + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Dadurch werden alle Benachrichtigungen über die Dokumentenverarbeitung auf der Startseite unterdrückt. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Erscheint auf + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Auf Startseite zeigen + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + In Seitenleiste zeigen + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Keine gespeicherten Ansichten vorhanden. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Nicht gefunden + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Möchten Sie den Korrespondenten "" wirklich löschen? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Korrespondenten + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Letzter Kontakt + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Bestätigung + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Bestätigen + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Abbrechen + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Neuen Korrespondenten erstellen + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Korrespondenten bearbeiten + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Zuweisungsalgorithmus + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Zuweisungsmuster + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Groß-/Kleinschreibung irrelevant + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Neues Tag erstellen + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Tag bearbeiten + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Posteingangs-Tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Ein Tag mit dieser Option wird automatisch allen neuen Dokumenten zugewiesen. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Neuen Dokumenttyp erstellen + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Dokumenttyp bearbeiten + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Dokumente suchen + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Abmelden + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Verwalten + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administration + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Dokumentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Eine Idee vorschlagen + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Angemeldet als + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Geöffnete Dokumente + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Alle schließen + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Korrespondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Ohne Korrespondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Typ: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Ohne Dokumenttyp + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Ohne Tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titel: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titel + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titel & Inhalt + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Erweiterte Suche + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Ähnlich zu + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Tags filtern + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Korrespondenten filtern + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Dokumenttypen filtern + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Filter zurücksetzen + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Nicht zugewiesen + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Anwenden + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Letzte 7 Tage + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Letzten Monat + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Letzte 3 Monate + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Letztes Jahr + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Nach + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Vor + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Zurücksetzen + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Anzeigen + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Nach Korrespondent filtern + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Nach Tag filtern + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Relevanz: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Ausgestellt: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Hinzugefügt: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Geändert: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Fehler beim Ausführung der Massenverarbeitung: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" und "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + und "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Tag-Zuweisung bestätigen + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Diese Aktion wird ausgewählten Dokumenten das Tag "" hinzufügen. + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Diese Aktion wird ausgewählten Dokumenten die Tags hinzufügen. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Diese Aktion wird das Tag "" von ausgewählten Dokumenten entfernen. + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Diese Aktion wird die Tags von ausgewählten Dokumenten entfernen. + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Diese Aktion wird die Tags den ausgewählten Dokumenten hinzufügen und die Tags entfernen. + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Korrespondent-Zuweisung bestätigen + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Diese Aktion wird ausgewählten Dokumenten den Korrespondent "" zuweisen. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Diese Aktion wird bei ausgewählten Dokumenten den Korrespondent entfernen. + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Dokumenttyp-Zuweisung bestätigen + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Diese Aktion wird ausgewählten Dokumenten den Dokumenttyp "" zuweisen. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Diese Aktion wird bei ausgewählten Dokumenten den Dokumenttyp entfernen. + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Löschen bestätigen + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Diese Aktion wird ausgewählte Dokumente unwiderruflich löschen. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Diese Aktion kann nicht rückgängig gemacht werden. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Dokument(e) löschen + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Auswählen: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Alle + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Bearbeiten: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Originale herunterladen + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Element hinzufügen + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Vorschläge: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Aktuelle Ansicht speichern + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Tag hinzufügen + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Alle anzeigen + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistiken + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Anzahl Dokumente gesamt: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Dokumente im Posteingang: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Verarbeite: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Fehlgeschlagen: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Hinzugefügt: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Verbinde... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Hochladen... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Datei hochgeladen, warte... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP-Fehler: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Neue Dokumente hochladen + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Dokumente hier ablegen oder + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Datei auswählen + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Fertiggestellte verbergen + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Ein weiteres Dokument} other { weitere Dokumente}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Dokument öffnen + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Erste Schritte + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless ist startbereit! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Sie können Ihre Dokumente hochladen, indem Sie sie in die Hochladebox auf der rechten Seite ziehen oder in den konfigurierten Ablageordner kopieren/verschieben. Danach werden sie in der Dokumentenliste erscheinen. Nachdem Sie den Dokumenten einige Metadaten zugewiesen haben, können Sie die Filtermechanismen zur Erstellung individueller Ansichten benutzen (zum Beispiel "Kürzlich hinzugefügt" oder "Mit TODO markiert") und die Ansichten werden auf der Startseite anstelle dieser Nachricht erscheinen. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless bietet darüber hinaus einige erweiterte Funktionen: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Nachdem Sie einige Dokumente hochgeladen und diesen Korrespondenten, Tags und Dokumenttypen zugewiesen haben, kann Paperless-ng diese Metadaten automatisch passenden neuen Dokumenten zuweisen. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Sie können in Paperless Ihre E-Mail-Konten einrichten und Paperless wird automatisch Dokumente aus Dateianhängen hinzufügen. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Die Dokumentation enthält mehr Informationen, wie diese Funktionen benutzt werden. Der Abschnitt zur allgemeinen Verwendung enthält zudem wertvolle Informationen zur generellen Verwendung von Paperless. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadaten + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Auswählen + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Bitte wählen Sie ein Element aus + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Ungültiges Datum. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Dokument mit ASN wird gesucht + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Ja + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nein + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (kein Titel) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Englisch (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Englisch (UK) + + + German + + src/app/services/settings.service.ts + 92 + + Deutsch + + + Dutch + + src/app/services/settings.service.ts + 93 + + Niederländisch + + + French + + src/app/services/settings.service.ts + 94 + + Französisch + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugiesisch + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugiesisch (Brasilien) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italienisch + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumänisch + + + Russian + + src/app/services/settings.service.ts + 99 + + Russisch + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanisch + + + Polish + + src/app/services/settings.service.ts + 101 + + Polnisch + + + Swedish + + src/app/services/settings.service.ts + 102 + + Schwedisch + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Dokument existiert bereits. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Datei nicht gefunden. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-Consume-Skript existiert nicht. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fehler beim Ausführen des Pre-Consume-Skripts. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-Consume-Skript existiert nicht. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fehler beim Ausführen des Post-Consume-Skripts. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Neue Datei erhalten. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Dateityp wird nicht unterstützt. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Verarbeite Dokument... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Erzeuge Miniaturbild... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Ermittle Datum des Dokuments... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Speichere Dokument... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Abgeschlossen. + + + Error + + src/app/services/toast.service.ts + 35 + + Fehler + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Korrespondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Dokumenttyp + + + Created + + src/app/services/rest/document.service.ts + 21 + + Ausgestellt am + + + Added + + src/app/services/rest/document.service.ts + 22 + + Hinzugefügt am + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Geändert am + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Relevanz + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Neues Element erstellen + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Element bearbeiten + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Konnte Element nicht speichern: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatisch + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Möchten Sie dieses Element wirklich löschen? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Assoziierte Dokumente werden nicht gelöscht. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Löschen + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Fehler beim Löschen des Elements: + + + Any word + + src/app/data/matching-model.ts + 12 + + Irgendein Wort + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Irgendein Wort: Dokument enthält eins der folgenden Wörter + + + All words + + src/app/data/matching-model.ts + 13 + + Alle Wörter + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Alle: Dokument enthält alle folgenden Wörter + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exakte Übereinstimmung + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exakt: Dokument enthält die folgende Zeichenkette + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regulärer Ausdruck + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regulärer Ausdruck: Dokument passt zum folgenden Ausdruck + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Ungenaues Wort + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Ungenau: Dokument enthält ein zum folgenden Wort ähnliches Wort + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Zuweisung automatisch erlernen + + + + diff --git a/src-ui/src/locale/messages.en_GB.xlf b/src-ui/src/locale/messages.en_GB.xlf new file mode 100644 index 000000000..5aa1c8d9f --- /dev/null +++ b/src-ui/src/locale/messages.en_GB.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documents + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + View "" created successfully. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Select + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Select none + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Select page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Select all + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Save as... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Title + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Document type + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Created + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Added + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirm delete + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + The files for this document will be deleted permanently. This operation cannot be undone. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Delete document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Delete + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + More like this + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Close + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Content + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Discard + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Save + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + of + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archive serial number + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date created + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Date modified + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date added + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filename + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Original file size + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Original mime type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archive MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archive file size + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Original document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archived document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Save & next + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , welcome to Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welcome to Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Do you really want to delete the tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Create + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter by: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Colour + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matching + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Document count + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Actions + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documents + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edit + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Do you really want to delete the document type ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Document types + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Settings saved successfully. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error while storing settings on server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Settings + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + General settings + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Saved views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Items per page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Bulk editing + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Show confirmation dialogs + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Deleting documents will always ask for confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Apply on close + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appears on + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Show on dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Show in sidebar + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No saved views defined. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Not Found + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Do you really want to delete the correspondent ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondents + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Last correspondence + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirm + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancel + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Create new correspondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edit correspondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matching algorithm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Case insensitive + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Create new tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edit tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tags are automatically assigned to all consumed documents. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Create new document type + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edit document type + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logout + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Manage + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documents + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Close all + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Title: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Title + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Title & content + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Advanced search + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + More like + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filter tags + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filter correspondents + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filter document types + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Not assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Apply + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Last 7 days + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Last month + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Last 3 months + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Last year + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + After + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Before + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Clear + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + View + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filter by correspondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filter by tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Score: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Created: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Added: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modified: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" and "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + and "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirm tags assignment + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirm correspondent assignment + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirm document type assignment + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Delete confirm + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + This operation cannot be undone. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Delete document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Select: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edit: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Save current view + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Show all + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistics + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Upload new documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Drop documents here or + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Browse files + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + First steps + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is running! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Select + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Please select an object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Searching document with asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Yes + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (no title) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + German + + + Dutch + + src/app/services/settings.service.ts + 93 + + Dutch + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portuguese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 99 + + Russian + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanish + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Document type + + + Created + + src/app/services/rest/document.service.ts + 21 + + Created + + + Added + + src/app/services/rest/document.service.ts + 22 + + Added + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modified + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Create new item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edit item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Could not save element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatic + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Do you really want to delete this element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Associated documents will not be deleted. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Delete + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Any word + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + All words + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exact match + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy word + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/locale/messages.es_ES.xlf b/src-ui/src/locale/messages.es_ES.xlf new file mode 100644 index 000000000..e740d54c1 --- /dev/null +++ b/src-ui/src/locale/messages.es_ES.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Documento añadido + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Documento agregado a paperless. + + + Open document + + src/app/app.component.ts + 51 + + Abrir documento + + + Could not add : + + src/app/app.component.ts + 59 + + No se pudo añadir: + + + New document detected + + src/app/app.component.ts + 65 + + Nuevo documento detectado + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + El documento está siendo procesado por paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documentos + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Ver "" guardado satisfactoriamente. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Ver "" creado satisfactoriamente. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selecciona + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Seleccionar ninguno + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Seleccionar página + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Seleccionar todo + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Ordenar + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Vistas + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Guardar como... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Guardar "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Seleccionado de un documento} other {Seleccionados de documentos}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Un documento} other { documentos}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrado) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + NSA + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondencia + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Título + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Tipo de documento + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Creado + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Añadido + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirmar borrado + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + ¿Estás seguro de querer borrar el documento ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Los archivos para este documento serán borrados permanentemente. Esta operación no se puede deshacer. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Borrar documento + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Error borrando el documento: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Borrar + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Descargar + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Similares + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Cerrar + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detalles + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Contenido + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadatos + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Descartar + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Salvar + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Página + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + de + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Documento original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Número de serie del archivo + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Fecha de creación + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Fecha de modificación + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Fecha de subida + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nombre del fichero + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Comprobación MD5 original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Tamaño del fichero original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Tipo MIME original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Comprobación MD5 del archivo + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Tamaño del archivo + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadatos originales + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadatos archivados + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Guardar y continuar + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hola , bienvenido a Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Bienvenido a Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Panel de control + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + ¿Estás seguro de querer borrar la etiqueta ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etiquetas + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Crear + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrado por: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nombre + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Color + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Condición + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Cantidad de documentos + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Acciones + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documentos + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Editar + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + ¿Estás seguro de querer borrar el tipo de documento ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Tipos de documento + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Vista guardada "" borrada. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Ajustes guardados satisfactoriamente. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Usar idioma del sistema + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Usar formato de fechas del idioma seleccionado + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error guardando los ajustes: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Ajustes + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Ajustes generales + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notificaciones + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Vistas guardadas + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Apariencia + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Idioma + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Necesitas recargar la página tras aplicar el nuevo idioma. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Formato de fechas + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Formato de fechas + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Corto: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medio: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Largo: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Elementos por página + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Editor de documentos + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Usar el visor PDF proporcionado por el navegador + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Habitualmente esto es mas rápido para mostrar archivos PDF largos, pero puede no funcionar en algunos navegadores. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Modo oscuro + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Usar ajustes del sistema + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Habilitar modo oscuro + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invertir miniaturas en modo oscuro + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Editor multiple + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Mostrar diálogos de confirmación + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Borrar los documentos siempre pedirá confirmación. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Aplicar al cerrar + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Procesado de documentos + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Mostrar notificaciones cuando nuevos documentos sean borrados + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Mostrar notificaciones cuando el procesado se complete satisfactoriamente + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Mostrar notificaciones cuando el procesado falle + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + No mostrar notificaciones en el panel de control + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Esto suprimirá todos los mensajes de estado de procesamiento en el panel de control. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Aparece en + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Mostrar en el panel de control + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Mostrar barra lateral + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No hay ninguna vista guardada definida + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 No encontrado + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + ¿Estás seguro de querer borrar el tipo de documento ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Tipos de documento + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Último tipo de documento + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmación + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirmar + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancelar + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Crear nuevo tipo de documento + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Editar tipo de documento + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritmo del patrón de búsqueda + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Patrón de busqueda + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Insensible a mayúsculas y minusculas + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Crear nueva etiqueta + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Editar etiqueta + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Etiqueta de entrada + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Las etiquetas de la bandeja son asignadas automáticamente a todos los documentos consumidos. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Crear nuevo tipo de documento + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Editar tipo de documento + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Buscar documentos + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Cerrar sesión + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Organizar + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administrar + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Información + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentación + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Sugerir una idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Sesión iniciada como + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Abrir documentos + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Cerrar todos + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Tipo: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Sin tipo de documento + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Tipo: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Sin tipo de documento + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Etiqueta: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Sin ninguna etiqueta + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Título: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + NSA: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Título + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titulo y contenido + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + NSF + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Búsqueda avanzada + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Más parecido + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrar etiquetas + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrar tipos de documento + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrar tipos de documento + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Quitar filtros + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Sin asignar + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Aplicar + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Últimos 7 días + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Último mes + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Últimos 3 meses + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Último año + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Después + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Antes + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Limpiar + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Vista + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrar por tipo de documento + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrar por etiqueta + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Puntuación: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Creado: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Añadido: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modificado: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error ejecutando operación múltiple: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" y "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + y "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirmar etiquetas asignadas + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Esta operación agregará la etiqueta "" a documento(s) seleccionado(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Esta operación agregará las etiquetas a documento(s) seleccionado(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Esta operacion eliminará la etiqueta "" de documento(s) seleccionado(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Esta operacion eliminará las etiquetas de documento(s) seleccionado(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Esta operacion agregará las etiquetas y eliminará las etiquetas en documento(s) seleccionado(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirmar asignación correspondiente + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Esta operacion asignará el tipo de documento "" a documento(s) seleccionado(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Esta operación eliminará el tipo de documento de documento(s) seleccionado(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirmar asignación de tipo de documento + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Esta operacion asignará el tipo de documento "" a documento(s) seleccionado(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Esta operación eliminará el tipo de documento de documento(s) seleccionado(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Confirmar borrado + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Esta operación borrará permanentemente documento(s) seleccionado(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Esta operación no se puede deshacer. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Borrar documento(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Seleccionar: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Todos + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Editar: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Descargar originales + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Añadir elemento + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Sugerencias: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Guardar la vista actual + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Añadir etiqueta + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Mostrar todos + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Estadisticas + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total de documentos: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documentos en la bandeja de entrada: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Procesando: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Fallidos: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Agregado: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Conectando... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Subiendo... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Subida completa, esperando... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Error HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Subir nuevos documentos + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Arrastra aquí los documentos o + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Explorar ficheros + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Descartar completados + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Un documento más} other { más documentos}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Abrir documento + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Primeros pasos + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless está ejecutándose :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Puede empezar a subir documentos soltándolos en la caja de subida de archivos a la derecha o colocándolos en la carpeta de consumo configurada y comenzarán a aparecer en la lista de documentos. Después de añadir algunos metadatos a sus documentos, utilice los mecanismos de filtrado de paperless para crear vistas personalizadas (como 'Añadidas recientes', 'Pendiente de etiquetar') y aparecerán en el panel de control en lugar de este mensaje. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless tiene algunas características para hacer tu vida mas sencilla: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Una vez tengas un par de documentos en paperless y añadido metadatos, paperless puede asignar metadatos a los nuevos documentos automáticamente. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Puedes configurar paperless para leer tus correos y añadir documentos de los ficheros adjuntos. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consulta la documentación sobre como usar estas características . La sección de uso básico también tiene algo de información de como usar Paperless en general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadatos + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Seleccionar + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Por favor, selecciona un objeto + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Fecha no válida. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Buscando documento con NSA + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Si + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (sin título) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Inglés (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Inglés (Gran Bretaña) + + + German + + src/app/services/settings.service.ts + 92 + + Alemán + + + Dutch + + src/app/services/settings.service.ts + 93 + + Alemán + + + French + + src/app/services/settings.service.ts + 94 + + Francés + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugués + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugués (Brasil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiano + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumano + + + Russian + + src/app/services/settings.service.ts + 99 + + Ruso + + + Spanish + + src/app/services/settings.service.ts + 100 + + Español + + + Polish + + src/app/services/settings.service.ts + 101 + + Polaco + + + Swedish + + src/app/services/settings.service.ts + 102 + + Sueco + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + El documento ya existe. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Archivo no encontrado + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + No existe script pre-consumo. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error al ejecutar el script pre-consumo. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + No existe script post-consumo. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error al ejecutar el script post-consumo. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Nuevo archivo recibido. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Tipo de fichero no soportado. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Procesando documento... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generando miniatura... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Obteniendo fecha del documento... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Guardando documento... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Completado. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Información + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondencia + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Tipo de documento + + + Created + + src/app/services/rest/document.service.ts + 21 + + Creado + + + Added + + src/app/services/rest/document.service.ts + 22 + + Agregado + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modificado + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Puntuación de búsqueda + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Crear nuevo elemento + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Editar elemento + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + No se pudo guardar el elemento: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automático + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + ¿Estás seguro de querer borrar este elemento? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Los documentos asociados no serán borrados. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Borrar + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error borrando el elemento: + + + Any word + + src/app/data/matching-model.ts + 12 + + Cualquier palabra + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Cualquiera: El documento contiene cualquiera de estas palabras (separadas por espacios) + + + All words + + src/app/data/matching-model.ts + 13 + + Todas las palabras + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Todas: El documento contiene todas estas palabras (separadas por espacios) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Coincidencia exacta + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exacto: El documento contiene este texto + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Expresión regular + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Expresión regular: El documento coincide con esta expresión regular + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Palabra borrosa + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Difuso: El documento contiene una palabra similar a esta + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Automático: Aprender automáticamente + + + + diff --git a/src-ui/src/locale/messages.fr_FR.xlf b/src-ui/src/locale/messages.fr_FR.xlf new file mode 100644 index 000000000..973d1ffaf --- /dev/null +++ b/src-ui/src/locale/messages.fr_FR.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document ajouté + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Le document a été ajouté à Paperless-ng. + + + Open document + + src/app/app.component.ts + 51 + + Ouvrir le document + + + Could not add : + + src/app/app.component.ts + 59 + + Impossible d'ajouter : + + + New document detected + + src/app/app.component.ts + 65 + + Nouveau document détecté + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Le document est en cours de traitement par Paperless-ng. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documents + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Vue "" enregistrée avec succès. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Vue "" créée avec succès. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Sélectionner + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Sélectionner aucun + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Sélectionner la page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Sélectionner tout + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Trier + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Vues + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Enregistrer sous... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Enregistrer "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 { document sélectionné sur 1} other { documents sélectionnés sur }} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Un document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtré) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + NSA + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondant + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titre + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Type de document + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Date de création + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Date d'ajout + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirmer la suppression + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Voulez-vous vraiment supprimer le document "" ? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Les fichiers liés à ce document seront supprimés définitivement. Cette action est irréversible. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Supprimer le document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Une erreur s'est produite lors de la suppression du document : + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Supprimer + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Télécharger + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Documents relatifs + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Fermer + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Détails + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Contenu + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Métadonnées + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Abandonner + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Enregistrer + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + sur + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Télécharger l'original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Numéro de série de l'archive + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date de création + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Modifié le + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date d'ajout + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nom de fichier du média + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Somme de contrôle MD5 de l'original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Taille de fichier de l'original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Type mime de l'original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Somme de contrôle MD5 de l'archive + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Taille de fichier de l'archive + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Métadonnées du document original + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Métadonnées du document archivé + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Enregistrer & suivant + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Bonjour , bienvenue dans Paperless-ng ! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Bienvenue dans Paperless-ng ! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Tableau de bord + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Voulez-vous vraiment supprimer l'étiquette "" ? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Étiquettes + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Créer + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrer par : + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nom + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Couleur + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Rapprochement + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Nombre de documents + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Actions + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documents + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Éditer + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Voulez-vous vraiment supprimer le type de document "" ? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Types de document + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Journaux + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Vue "" supprimée. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Paramètres enregistrés avec succès. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Utiliser la langue du système + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Utiliser le format de date de la langue d'affichage + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Une erreur s'est produite lors de l'enregistrement des paramètres sur le serveur : + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Paramètres + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Paramètres généraux + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Vues enregistrées + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Affichage + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Langue d'affichage + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Vous devez recharger la page après avoir sélectionné une nouvelle langue. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Affichage de la date + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Format de date + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Court : + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Moyen : + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long : + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Éléments par page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Visionneuse de documents + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Utiliser la visionneuse PDF fournie par le navigateur + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Cette option est généralement plus rapide pour l'affichage de grands documents PDF, mais elle peut ne pas fonctionner sur certains navigateurs. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Mode sombre + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Utiliser les paramètres du système + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Activer le mode sombre + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Inverser les vignettes en mode sombre + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Edition en masse + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Afficher les messages de confirmation + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + La suppression de documents requiert toujours une confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Appliquer lors de la fermeture + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Traitement de documents + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Afficher des notifications lorsque de nouveaux documents sont détectés + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Afficher des notifications lorsque le traitement des documents se termine avec succès + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Afficher des notifications en cas d'échec du traitement des documents + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Supprimer les notifications du tableau de bord + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Cela supprimera tous les messages liés au traitement de documents sur le tableau de bord. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Apparaît sur + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Montrer sur le tableau de bord + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Montrer dans la barre latérale + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Aucune vue sauvegardée n'est définie. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Non trouvé + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Voulez-vous vraiment supprimer le correspondant "" ? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondants + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Dernière correspondance + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirmer + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Annuler + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Créer un nouveau correspondant + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Éditer le correspondant + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algorithme de rapprochement + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Modèle de rapprochement + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Insensible à la casse + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Créer une nouvelle étiquette + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Éditer l'étiquette + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Étiquette de boîte de réception + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Les étiquettes de boîte de réception sont automatiquement affectées à tous les documents traités. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Créer un nouveau type de document + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Éditer le type de document + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Rechercher des documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Déconnexion + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Gestion + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administration + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggérer une idée + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Connexion en tant que + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Documents ouverts + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Fermer tout + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondant : + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Sans correspondant + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Type : + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Sans type de document + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Étiquette : + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Sans étiquette + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titre : + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + NSA : + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titre + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titre & contenu + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + NSA + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Recherche avancée + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Plus comme + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrer les étiquettes + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrer les correspondants + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrer les types de documents + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Réinitialiser les filtres + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Non affecté + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Appliquer + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Les 7 derniers jours + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Le mois dernier + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Les 3 derniers mois + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + L'année passée + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Après + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Avant + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Réinitialiser + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Vue + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrer par correspondant + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrer par étiquette + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Score : + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Date de création : + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Date d'ajout : + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Date de modification : + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Une erreur s'est produite lors de l'exécution de l'opération de masse : + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" et "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + et "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirmer l'affectation des étiquettes + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Cette action affectera l'étiquette "" au(x) document(s) sélectionné(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Cette action affectera les étiquettes au(x) document(s) sélectionné(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Cette action supprimera l'étiquette "" de(s) document(s) sélectionné(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Cette action supprimera les étiquettes de(s) document(s) sélectionné(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Cette action affectera les étiquettes et supprimera les étiquettes de(s) document(s) sélectionné(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirmer l'affectation du correspondant + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Cette action affectera le correspondant "" au(x) document(s) sélectionné(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Cette action supprimera le correspondant de(s) document(s) sélectionné(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirmer l'affectation du type de document + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Cette action affectera le type de document "" au(x) document(s) sélectionné(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Cette action supprimera le type de document de(s) document(s) sélectionné(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Confirmer la suppression + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Cette action supprimera définitivement document(s) sélectionné(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Cette action est irréversible. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Supprimer le(s) document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Sélectionner : + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Tout + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Éditer : + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Télécharger les originaux + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Ajouter un élément + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggestions : + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Enregistrer la vue actuelle + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Ajouter une étiquette + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Tout afficher + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistiques + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Nombre total de documents : + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents dans la boîte de réception : + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Traitement : + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Échec : + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Ajout : + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connexion... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Téléchargement... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Chargement terminé, en attente... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Erreur HTTP : + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Charger de nouveaux documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Déposer des documents ici ou + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Parcourir les fichiers + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Masquer lorsque terminé + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Un document supplémentaire} other { documents supplémentaires}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Ouvrir le document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Premiers pas + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless-ng fonctionne ! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Vous pouvez commencer par télécharger des documents en les déposant dans la boîte de téléchargement de fichiers à droite ou en les déposant dans le dossier de traitement configuré, et ils commenceront à apparaître dans la liste des documents. Une fois que vous avez ajouté des métadonnées à vos documents, utilisez les mécanismes de filtrage de Paperless-ng pour créer des vues personnalisées (telles que "Récemment ajouté", "Marqué À FAIRE") et elles apparaîtront sur le tableau de bord à la place de ce message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless-ng offre quelques fonctionnalités supplémentaires qui tendent à vous faciliter la vie : + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Une fois que vous avez quelques documents dans Paperless-ng et que vous y avez ajouté des métadonnées, Paperless-ng peut attribuer automatiquement ces métadonnées à de nouveaux documents. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Vous pouvez configurer Paperless-ng pour lire vos courriels et ajouter des documents à partir des pièces jointes. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consultez la documentation sur l'utilisation de ces fonctionnalités. La section sur l'utilisation de base contient également des informations sur la manière d'utiliser Paperless-ng en général. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Métadonnées + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Sélectionner + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Veuillez sélectionner un objet + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Date incorrecte. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Recherche de document avec NSA + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Oui + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Non + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (sans titre) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Anglais (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Anglais (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Allemand + + + Dutch + + src/app/services/settings.service.ts + 93 + + Néerlandais + + + French + + src/app/services/settings.service.ts + 94 + + Français + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugais + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugais (Brésil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italien + + + Romanian + + src/app/services/settings.service.ts + 98 + + Roumain + + + Russian + + src/app/services/settings.service.ts + 99 + + Russe + + + Spanish + + src/app/services/settings.service.ts + 100 + + Espagnol + + + Polish + + src/app/services/settings.service.ts + 101 + + Polonais + + + Swedish + + src/app/services/settings.service.ts + 102 + + Suédois + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Le document existe déjà. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Fichier non trouvé. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Le script de pré-traitement n'existe pas. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erreur lors de l'exécution du script de pré-traitement. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Le script de post-traitement n'existe pas. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erreur lors de l'exécution du script de post-traitement. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Réception d'un nouveau fichier. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Type de fichier non pris en charge. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Traitement du document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Génération de vignette... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Extraction de la date du document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Enregistrement du document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Terminé. + + + Error + + src/app/services/toast.service.ts + 35 + + Erreur + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondant + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Type de document + + + Created + + src/app/services/rest/document.service.ts + 21 + + Date de création + + + Added + + src/app/services/rest/document.service.ts + 22 + + Date d'ajout + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Date de modification + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Pertinence + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Créer un nouvel élément + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Éditer l'élément + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Impossible d'enregistrer l'élément : + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatique + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Voulez-vous vraiment supprimer cet élément ? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Les documents associés ne seront pas supprimés. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Supprimer + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Une erreur s'est produite lors de la suppression de l'élément : + + + Any word + + src/app/data/matching-model.ts + 12 + + Un des mots + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Un des mots : contient l'un de ces mots (séparés par des espaces) + + + All words + + src/app/data/matching-model.ts + 13 + + Tous les mots + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Tous les mots : contient tous ces mots (séparés par des espaces) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Concordance exacte + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Concordance exacte : contient cette chaîne de caractères + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Expression régulière + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Expression régulière : correspond à cette expression régulière + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Mot approximatif + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Mot approximatif : contient un mot similaire à ce mot + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Automatique : apprentissage automatique du rapprochement + + + + diff --git a/src-ui/src/locale/messages.hu_HU.xlf b/src-ui/src/locale/messages.hu_HU.xlf new file mode 100644 index 000000000..37cf6d89d --- /dev/null +++ b/src-ui/src/locale/messages.hu_HU.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokumentumok + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Nézet "" sikeresen létrehozva. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Kijelölés + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Kijelölés törlése + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Oldal kijelölése + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Mindent kijelöl + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Nézet + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Mentés másként + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Partner + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Cím + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Dokumentum típus + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Létrehozva + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Hozzáadva + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Törlés megerősítése + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + A dokumentumhoz tartozó fájlok véglegesen törlésre kerülnek. Ez a művelet nem visszafordítható. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Dokumentum törlése + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Törlés + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Letöltés + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Ehhez hasonlók + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Bezárás + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Részletek + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Tartalom + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metaadat + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Elvetés + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Mentés + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Oldal + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + / + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Eredeti letöltése + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archívum sorozat szám + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Létrehozás dátuma + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Módosítás dátuma + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Hozzáadás dátuma + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Média fájlnév + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Eredeti MD5 ellenőrző szám + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Eredeti fájl méret + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Eredeti MIME típus + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archívum MD5 ellenőrző szám + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archív fájl mérete + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Eredeti dokumentum metaadat + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archivált dokumentum metaadat + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Mentés & Következő + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , a Paperless-ng üdvözöl Téged! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Üdvözöl a Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Vezérlőpult + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Valóban törölni akarod a "" címkét? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Cimkék + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Létrehozás + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Szűrés: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Név + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Szín + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Egyezés + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Dokumentum szám + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Műveletek + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokumentumok + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Szerkesztés + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Valóban törölni akarod a "" dokumentum típust? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Dokumentum típusok + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Napló + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Beállítások elmentve. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Hiba történt a beállitások mentése közben: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Beállitások + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Általános beállitások + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Mentett nézetek + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Oldalankénti elemek száma + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Csoportos szerkesztés + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Jóváhagyó dialógusok mutatása + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Dokumentumok törlése mindig jóváhagyást igényel + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Alkalmazás bezárás után + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Megjelenés + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Vezérlőpulton megjelenítés + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Oldal menüben megjelenítés + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nincs mentett nézet definiálva. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Nem található + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Valóban törölni szerenéd a következő partnert: "" ? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Partnerek + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Utolsó partner + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Jóváhagyás + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Jóváhagy + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Megszakítás + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Új partner létrehozása + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Partner szerkesztése + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Egyeztető algoritmus + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Kisbetű-nagybetű érzékenység + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Új cimke készítése + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Cimke szerkesztés + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox cimke + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Az "inbox" cimkék automatikusan hozzárendelődnek a feldolgozott dokumentumokhoz. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Új dokumentum tipus létrehozása + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Dokumentum típus szerkesztése + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Kilépés + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Menedzselés + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Adminisztráció + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Dokumentáció + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Dokumentum megnyitása + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Összes bezárása + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Partner: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Típus: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Cimke: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Title: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Cím + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Title & content + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Advanced search + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + More like + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Cimkék szerinti szűrés + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Partnerek szerinti szűrés + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Dokumentum típus szerinti szűrés + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Nincs hozzárendelés + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Alkalmaz + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Utolsó 7 nap + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Előző hónap + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Előző 3 hónap + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Előző év + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Után + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Előtte + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Törlés + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Megtekint + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Partner szerinti szűrés + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Cimke szerinti szűrés + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Pont: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Created: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Added: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modified: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" és "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + és "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Cimke hozzárendelés jóváhagyása + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Partner hozzárendelés megerősítése + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Dokumentum típus hozzárendelés megerősítése + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Törlés megerősítése + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Ez a művelet nem visszafordítható. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Dokumentum(ok) törlése + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Kijelöl + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Mind + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Szerkeszt: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Jelenlegi nézet mentése + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Mindent megjelenít + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statisztika + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Új dokumentum feltöltése + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Húzz ide dokumentumokat, vagy + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Fájl böngészése + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Első lépések + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + A Paperless üzemel! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metaadat + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Kijelöl + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Kérem válaszon egy elemet + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Searching document with asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Igen + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nem + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (nincs cím) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Angol (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Német + + + Dutch + + src/app/services/settings.service.ts + 93 + + Dutch + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portuguese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 99 + + Russian + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanish + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Hiba + + + Information + + src/app/services/toast.service.ts + 39 + + Információ + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Partner + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Dokumentum típus + + + Created + + src/app/services/rest/document.service.ts + 21 + + Létrehozva + + + Added + + src/app/services/rest/document.service.ts + 22 + + Hozzáadva + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Módosítva + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Új elem létrehozása + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Elem szerkesztése + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Hiba az elem mentése során: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatikus + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Valóban törölni szeretnéd ezt az elemet? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + A kapcsolódó dokumentumok nem kerülnek törlésre. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Törlés + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Any word + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + All words + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exact match + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regexp + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy word + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/locale/messages.it_IT.xlf b/src-ui/src/locale/messages.it_IT.xlf new file mode 100644 index 000000000..3b2271a9b --- /dev/null +++ b/src-ui/src/locale/messages.it_IT.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Documento aggiunto + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Il documento è stato aggiunto a paperless. + + + Open document + + src/app/app.component.ts + 51 + + Apri documento + + + Could not add : + + src/app/app.component.ts + 59 + + Impossibile aggiungere : + + + New document detected + + src/app/app.component.ts + 65 + + Nuovo documento trovato + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Paperless sta elaborando il documento . + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documenti + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + La vista "" è stata salvata. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + La vista "" è stata creata. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selezione + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Seleziona nessuno + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Seleziona pagina + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Seleziona tutti + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Ordina + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Viste + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Salva come... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Salva "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selezionato di un documento} other {Selezionati di documenti}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Un documento} other { documenti}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrato) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Corrispondente + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titolo + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Tipo di documento + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Creato + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Aggiunto + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Conferma eliminazione + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Vuoi eliminare il documento ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + I file di questo documento saranno eliminati permanentemente. Questa operazione è irreversibile. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Elimina documento + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Errore nell'eliminazione del documento: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Elimina + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Scarica + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Altro come questo + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Chiudi + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Dettagli + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Contenuto + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadati + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Scarta + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Salva + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Pagina + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + di + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Scarica originale + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Numero seriale di archivio + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Data creazione + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Data modifica + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Data aggiunta + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nome file + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Checksum MD5 originale + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Dimensione file originale + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Tipo mime originale + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Checksum MD5 archivio + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Dimensione file archivio + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadati del documento originale + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadati del documento archiviato + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Salva e vai al prossimo + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Ciao , benvenuto su Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Benvenuto su Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Vuoi eliminare il tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etichette + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Crea + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtra per: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nome + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Colore + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Assegnazione + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Numero di documenti + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Azioni + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documenti + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Modifica + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Vuoi eliminare il tipo di documento ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Tipi di documento + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Log + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + La vista "" è stata eliminata. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Le impostazioni sono state salvate. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Usa lingua di sistema + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Usa il formato data della lingua + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Errore durante il salvataggio delle impostazioni sul server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Impostazioni + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Impostazioni generali + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifiche + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Viste salvate + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Aspetto + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Lingua + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Riavviare la pagina dopo avere impostato la nuova lingua. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Visualizzazione data + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Formato data + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Breve: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Media: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Lunga: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Oggetti per pagina + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Editor dei documenti + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Usa il visualizzatore PDF fornito dal browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Di solito è più rapido nel visualizzare documenti PDF grandi, ma potrebbe non funzionare su tutti i browser. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Modalità notte + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Usa impostazioni di sistema + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Abilita modalità notte + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Inverti anteptime in modalità notte + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Modifica in blocco + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Mostra dialoghi di conferma + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Verrà sempre chiesta una conferma prima di eliminare documenti. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Applica in chiusura + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Elaborazione del documento + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Notifica quando vengono trovati nuovi documenti + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Notifica quando l'elaborazione del documento viene completata con successo. + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Notifica quando l'elaborazione del documento fallisce + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Non mostrare notifiche nella dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Verranno interrotte tutte le notifiche nella dashboard riguardo lo stato dell'elaborazione dei documenti. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appare in + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Mostra nella dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Mostra nella barra laterale + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nessuna vista salvata. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Non trovato + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Vuoi eliminare il corrispondente ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Corrispondenti + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Ultima assegnazione + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Conferma + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Conferma + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Annulla + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Crea nuovo corrispondente + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Modifica corrispondente + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritmo di assegnazione + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Criterio di assegnazione + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Senza distinzione tra maiuscole e minuscole + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Crea nuovo tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Modifica tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Tag di arrivo + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + I tag di arrivo vengono assegnati automaticamente a tutti i documenti elaborati. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Crea nuovo tipo di documento + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Modifica tipo di documento + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Cerca documenti + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Esci + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Gestisci + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Amministratore + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informazioni + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentazione + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggerisci un'idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Accesso effettuato come + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Apri documenti + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Chiudi tutti + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Corrispondente: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Senza corrispondente + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Tipo: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Senza tipo di documento + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Senza alcun tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titolo: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titolo + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titolo & contenuto + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Ricerca avanzata + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Più come + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtra tag + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtra corrispondenti + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtra tipi di documento + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Ripristina filtri + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Non assegnato + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Applica + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Ultimi 7 giorni + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Ultimo mese + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Ultimi 3 mesi + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Ultimo anno + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Dopo + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Prima + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Pulisci + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Mostra + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtra per corrispondente + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtra per tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Rilevanza: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Creato il: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Aggiunto il: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modificato il: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Errore durante l'operazione di modifica in blocco: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" e "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + e "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Conferma assegnazione tag + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Questa operazione aggiungerà il tag "" a documento/i selezionato/i. + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Questa operazione aggiungerà i tag a documento/i selezionato/i. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Questa operazione rimuoverà il tag "" da documento/i selezionato/i. + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Questa operazione rimuoverà i tag da documento/i selezionato/i. + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Questa operazione aggiungerà i tag e rimuoverà i tag a documento/i selezionato/i. + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Conferma assegnazione corrispondente + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Questa operazione assegnerà il corrispondente "" a documento/i selezionato/i. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Questa operazione rimuoverà il corrispondente da documento/i selezionato/i. + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Conferma assegnazione tipo di documento + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Questa operazione assegnerà il tipo di documento "" a documento/i selezionato/i. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Questa operazione eliminerà il tipo di documento da documento/i selezionato/i. + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Conferma eliminazione + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Questa operazione eliminerà definitivamente documento/i selezionato/i. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Questa operazione non può essere annullata. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Elimina documento/i + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Seleziona: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Tutto + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Modifica: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Scarica originali + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Aggiungi elemento + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggerimenti: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Salva vista corrente + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Aggiungi tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Mostra tutto + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistiche + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Documenti totali: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documenti taggati in arrivo: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Elaborazione in corso: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Falliti: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Aggiunti: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connessione in corso... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Caricamento in corso... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Caricamento completato, in attesa... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Errore HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Carica nuovi documenti + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Trascina qui i documenti oppure + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Sfoglia i file + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Ignora completati + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Un altro documento} other {altri documenti}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Apri documento + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Primi passi + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless è in esecuzione! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Puoi iniziare a caricare documenti trascinandoli nel riquadro per il caricamento posizionato sulla destra, oppure spostarli nella cartella di elaborazione e appariranno nella lista dei documenti. Dopo aver aggiunto i metadati ai tuoi documenti, usa i meccanismi di filtraggio per creare viste personalizzate (come 'Aggiunti di recente' o 'Da taggare') e appariranno sulla dashboard invece di questo messaggio. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offre alcune funzionalità che cercano di rendere migliore la tua vita: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Quando avrai caricato qualche documento e aggiunto dei metadati, Paperless potrà assegnare automaticamente quei metadati ai nuovi documenti. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Puoi configurare Paperless per leggere le tue mail e aggiungere i documenti che trova negli allegati. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Controlla la documentazione per capire come usare queste funzioni. La sezione per l'utilizzo base ha anche qualche suggerimento su come usare Paperless in generale. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadati + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Seleziona + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Seleziona un elemento + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Data non valida. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Cercando documento con asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (nessun titolo) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Inglese (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Inglese (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Tedesco + + + Dutch + + src/app/services/settings.service.ts + 93 + + Olandese + + + French + + src/app/services/settings.service.ts + 94 + + Francese + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portoghese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portoghese (Brasile) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiano + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumeno + + + Russian + + src/app/services/settings.service.ts + 99 + + Russo + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spagnolo + + + Polish + + src/app/services/settings.service.ts + 101 + + Polacco + + + Swedish + + src/app/services/settings.service.ts + 102 + + Svedese + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Il documento esiste già. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File non trovato. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Lo script di pre-consume (pre elaborazione) non esiste. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Errore durante l'esecuzione dello script di pre-consume (pre elaborazione). + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Lo script di post-consume (post elaborazione) non esiste. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Errore durante l'esecuzione dello script di post-consume (post elaborazione). + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Nuovo file ricevuto. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Tipo di file non supportato. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Elaborazione documento in corso... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generazione anteprima in corso... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Recupero della data del documento in corso... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Salvataggio documento in corso... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Completato. + + + Error + + src/app/services/toast.service.ts + 35 + + Errore + + + Information + + src/app/services/toast.service.ts + 39 + + Informazione + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Corrispondente + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Tipo di documento + + + Created + + src/app/services/rest/document.service.ts + 21 + + Creato + + + Added + + src/app/services/rest/document.service.ts + 22 + + Aggiunto + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modificato + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Rilevanza ricerca + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Crea nuovo elemento + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Modifica elemento + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Non è possibile salvare l'elemento: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatico + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Vuoi eliminare questo elemento? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + I documenti associati non verranno eliminati. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Elimina + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Errore durante l'eliminazione dell'elemento: + + + Any word + + src/app/data/matching-model.ts + 12 + + Qualsiasi parola + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Qualsiasi: il documento contiene una qualunque di queste parole (separate dallo spazio) + + + All words + + src/app/data/matching-model.ts + 13 + + Tutte le parole + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Tutto: il documento contiene tutte queste parole (separate dallo spazio) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Corrispondenza esatta + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Puntuale: il documento contiene questa stringa + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Espressione regolare + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Espressione regolare: il documento corrisponde a questa espressione regolare + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Parola fuzzy + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: il documento contiene una parola simile a questa + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Automatico: apprende l'assegnazione automaticamente + + + + diff --git a/src-ui/src/locale/messages.la_LA.xlf b/src-ui/src/locale/messages.la_LA.xlf new file mode 100644 index 000000000..149c26513 --- /dev/null +++ b/src-ui/src/locale/messages.la_LA.xlf @@ -0,0 +1,2290 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 49 + + Documents + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 115 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 136 + + View "" created successfully. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Select + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Select none + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Select page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Select all + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Save as... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 105 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 111 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 117 + + Title + + + Document type + + src/app/components/document-list/document-list.component.html + 123 + + Document type + + + Created + + src/app/components/document-list/document-list.component.html + 129 + + Created + + + Added + + src/app/components/document-list/document-list.component.html + 135 + + Added + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 203 + + Confirm delete + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 204 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 205 + + The files for this document will be deleted permanently. This operation cannot be undone. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 207 + + Delete document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 214 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Delete + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + More like this + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Close + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Content + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Discard + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Save + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + of + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archive serial number + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date created + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Date modified + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date added + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filename + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Original file size + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Original mime type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archive MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archive file size + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Original document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archived document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Save & next + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , welcome to Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welcome to Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Do you really want to delete the tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Create + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter by: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Color + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matching + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Document count + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Actions + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documents + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edit + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Do you really want to delete the document type ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Document types + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Settings saved successfully. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error while storing settings on server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Settings + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + General settings + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Saved views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Items per page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Bulk editing + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Show confirmation dialogs + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Deleting documents will always ask for confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Apply on close + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appears on + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Show on dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Show in sidebar + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No saved views defined. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Not Found + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Do you really want to delete the correspondent ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondents + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Last correspondence + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirm + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancel + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Create new correspondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edit correspondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matching algorithm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Case insensitive + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Create new tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edit tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tags are automatically assigned to all consumed documents. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Create new document type + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edit document type + + + Search results + + src/app/components/search/search.component.html + 1 + + Search results + + + Invalid search query: + + src/app/components/search/search.component.html + 4 + + Invalid search query: + + + Showing documents similar to + + src/app/components/search/search.component.html + 7 + + Showing documents similar to + + + Search query: + + src/app/components/search/search.component.html + 11 + + Search query: + + + Did you mean ""? + + src/app/components/search/search.component.html + 13 + + Did you mean ""? + + + {VAR_PLURAL, plural, =0 {No results} =1 {One result} other { results}} + + src/app/components/search/search.component.html + 18 + + {VAR_PLURAL, plural, =0 {No results} =1 {One result} other { results}} + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logout + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Manage + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documents + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Close all + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 73 + + Title + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 74 + + Title & content + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 32 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 34 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 41 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 45 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 49 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 53 + + Title: + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 20 + + Filter tags + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 28 + + Filter correspondents + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 35 + + Filter document types + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 58 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Not assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Apply + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Last 7 days + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Last month + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Last 3 months + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Last year + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + After + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Before + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Clear + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 50 + + View + + + Created: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 67 + + Created: + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filter by correspondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filter by tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 62 + + Score: + + + View in browser + + src/app/components/document-list/document-card-small/document-card-small.component.html + 40 + + View in browser + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" and "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + and "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirm tags assignment + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirm correspondent assignment + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirm document type assignment + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Delete confirm + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + This operation cannot be undone. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Delete document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Select: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edit: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 26 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Save current view + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Show all + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistics + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Upload new documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Drop documents here or + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Browse files + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + First steps + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is running! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Select + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Please select an object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Yes + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (no title) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + German + + + Dutch + + src/app/services/settings.service.ts + 93 + + Dutch + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 95 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 96 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 97 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 98 + + Russian + + + ISO 8601 + + src/app/services/settings.service.ts + 103 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + ASN + + src/app/services/rest/document.service.ts + 17 + + ASN + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Document type + + + Created + + src/app/services/rest/document.service.ts + 21 + + Created + + + Added + + src/app/services/rest/document.service.ts + 22 + + Added + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modified + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Create new item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edit item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Could not save element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatic + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Do you really want to delete this element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Associated documents will not be deleted. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Delete + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Any word + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + All words + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exact match + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy word + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/locale/messages.lb_LU.xlf b/src-ui/src/locale/messages.lb_LU.xlf new file mode 100644 index 000000000..2b6f191a9 --- /dev/null +++ b/src-ui/src/locale/messages.lb_LU.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Dokument dobäigesat + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Dokument gouf a Paperless-ng dobäigesat. + + + Open document + + src/app/app.component.ts + 51 + + Dokument opmaachen + + + Could not add : + + src/app/app.component.ts + 59 + + Konnt de Fichier net dobäisetzen: + + + New document detected + + src/app/app.component.ts + 65 + + Neit Dokument detektéiert + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Dokument gëtt vu Paperless-ng veraarbecht. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokumenter + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Usiicht "" gouf erfollegräich gespäichert. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Vue "" gouf erfollegräich erstallt. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Auswielen + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Näischt auswielen + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Säit auswielen + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Alles auswielen + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Zortéieren + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Usiichten + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Späicheren als... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + "" späicheren + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 { vun 1 Dokument ausgewählt} other { vu(n) Dokumenter ausgewielt}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Een Dokument} other { Dokumenter}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (gefiltert) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Korrespondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titel + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Dokumententyp + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Erstallt + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Dobäigesat + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Läsche bestätegen + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Wëllt Dir d'Dokument "" wierklech läschen? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + D'Fichiere fir dëst Dokument gi permanent geläscht. Dës Operatioun kann net réckgängeg gemaach ginn. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Dokument läschen + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Feeler beim Läsche vum Dokument: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Läschen + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Eroflueden + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Ähnlech Dokumenter + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Zoumaachen + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detailer + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Inhalt + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadaten + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Verwerfen + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Späicheren + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Säit + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + vu(n) + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Original eroflueden + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archiv-Seriennummer + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Erstellungsdatum + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Verännert um + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Dobäigesat um + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Dateinumm vum Mediefichier + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + MD5-Préifzomm vum Original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Dateigréisst vum Original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Urspréngleche MIME-Typ + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + MD5-Préifzomm vum Archiv + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archiv-Dateigréisst + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadate vum Original-Dokument + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadate vum Archiv-Dokument + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Späicheren a weider + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hallo , wëllkomm bei Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Wëllkomm bei Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Startsäit + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Soll d'Etikett "" wierklech geläscht ginn? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etiketten + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Erstellen + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filteren no: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Numm + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Faarf + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Zouweisung + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Unzuel Dokumenter + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Aktiounen + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokumenter + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Editéieren + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Soll den Dokumententyp "" wierklech geläscht ginn? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Dokumententypen + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Protokoller + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Gespäichert Usiicht "" geläscht. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Astellungen erfollegräich gespäichert. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Systemsprooch benotzen + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Datumsformat vun der Sprooch vum Interface notzen + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Feeler beim Späichere vun den Astellungen um Server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Astellungen + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Allgemeng Astellungen + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifikatiounen + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Gespäichert Usiichten + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Ausgesinn + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Sprooch vum Interface + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Nodeem s eng nei Sprooch gewielt gouf muss d'Säit nei geluede ginn. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Uweise vum Datum + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Datumsformat + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Kuerz: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Mëttel: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Laang: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Elementer pro Säit + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Dokumenteneditor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + PDF-Betruechter vum Web-Browser notzen + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Dëst ass normalerweis méi séier fir grouss PDF-Dokumenter, mee kéint op verschiddene Browseren net goen. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Nuetsmodus + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Systemastellungen notzen + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Nuetsmodus aktivéieren + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Virschaubiller an Nuetsmodus invertéieren + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Massebeaarbechtung + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Bestätegungsmessagë weisen + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Beim Läsche vun Dokumenter gëtt ëmmer no enger Bestätegung gefrot. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Späichere beim Zoumaachen + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Dokumenteveraarbechtung + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Notifikatiounen uweise wann nei Dokumenter detektéiert ginn + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Notifikatiounen uweise wann d'Dokumentveraarbechtung erfollegräich ofgeschloss ass + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Notifikatiounen uweise wann d'Dokumenteveraarbechtung feelschléit + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Notifikatiounen op der Startsäit ënnerdrécken + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Doduerch ginn all Messagen iwwer Dokumenteveraarbechtung op der Startsäit ënnerdréckt. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Erschéngt op + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Op der Startsäit uweisen + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + An der Säiteleescht uweisen + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Keng gespäichert Usiicht definéiert. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Net fonnt + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Soll de Korrespondent "" wierklech geläscht ginn? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Korrespondenten + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Lescht Korrespondenten + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Bestätegung + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Bestätegen + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Ofbriechen + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Neie Korrespondent erstellen + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Korrespondent editéieren + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Zouweisungsalgorithmus + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Zouweisungsmuster + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Grouss-/Klengschreiwung ignoréieren + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Nei Etikett erstellen + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Etikett editéieren + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Postaganks-Etikett + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Postaganks-Etikette ginn automatesch allen nei veraarbechten Dokumenter zougewisen. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Neien Dokumententyp erstellen + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Dokumententyp änneren + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Dokumenter sichen + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Ofmellen + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Verwalten + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administratioun + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informatiounen + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Dokumentatioun + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Eng Iddi virschloen + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Ugemellt als + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Oppen Dokumenter + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + All zoumaachen + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Korrespondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Ouni Korrespondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Typ: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Ouni Dokumententyp + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Etikett: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Ouni Etikett + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titel: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titel + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titel an Inhalt + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Erweidert Sich + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Méi ähnleches + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Etikette filteren + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Korrespondente filteren + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Dokumententype filteren + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Filteren zrécksetzen + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Net zougewisen + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Applizéieren + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Lescht 7 Deeg + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Leschte Mount + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Lescht 3 Méint + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Lescht Joer + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + No + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Virun + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Läschen + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Usiicht + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + No Korrespondent filteren + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + No Etikett filteren + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Relevanz: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Erstallt: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Dobäigesat: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Verännert: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Feeler beim Ausféiere vun der Masseveraarbechtung: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" a(n) "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + a(n) "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Zouweisung vun der Etikett bestätegen + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Dës Operatioun setzt d'Etikett "" bei ausgewielt Dokument(er). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Dës Operatioun setzt d'Etiketten bei ausgewielt Dokument(er). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Dës Operatioun läscht d'Etikett "" bei ausgewielt Dokument(er). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Dës Operatioun läscht d'Etiketten bei ausgewielt Dokument(er). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Dës Operatioun setzt d'Etiketten dobäi a läscht d'Etiketten bei ausgewielt Dokument(er). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + D'Zouweisung vum Korrespondent bestätegen + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Dës Operatioun weist de Korrespondent "" ausgewielt Dokument(er) zou. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Dës Operatioun läscht de Korrespondent bei ausgewielt Dokument(er). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Zouweisung vum Dokumententyp bestätegen + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Dës Operatioun weist den Dokumententyp "" ausgewielt Dokument(er) zou. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Dës Operatioun läscht den Dokumententyp bei ausgewielt Dokument(er). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Läsche bestätegen + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Dës Operatioun wäert ausgewielt Dokument(er) permanent läschen. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Dës Operatioun kann net réckgängeg gemaach ginn. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Dokument(er) läschen + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Auswielen: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Editéieren: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Originaler eroflueden + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Element dobäisetzen + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Virschléi: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Aktuell Usiicht späicheren + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Etikett dobäisetzen + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Alles uweisen + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistiken + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Unzuel Dokumenter insgesamt: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Unzuel Dokumenter am Postagank: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Veraarbechten: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Feelgeschloen: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Dobäigesat: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Gëtt verbonnen... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Gëtt eropgelueden... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Fichier eropgelueden. Waarden... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP-Feeler: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Nei Dokumenter eroplueden + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Dokumenter hei ofleeën oder + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Fichieren duerchsichen + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Déi fäerdeg traitéiert verstoppen + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Ee weidert Dokument} other { weider Dokumenter}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Dokument opmaachen + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Éischt Schrëtter + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless-ng leeft! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Nei Dokumenter kënnen eropgeluede ginn andeem s se an d'Eroplued-Këscht op der rietser Säit ofgeluecht oder an de konfiguréierte Veraarbechtungs-Dossier kopéiert ginn. Se erschéngen dann an der Dokumentelëscht. Nodeem s d'Dokumenter mat e puer Metadaten markéiert sinn, kann de Filtermechanismus benotzt gi fir personaliséiert Vuen ze konfiguréieren (wéi beispillsweis "Rezent dobäikomm", "Als To Do markéiert") an déi erschéngen op der Startsäit amplaz vun dësem Message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless bitt e puer weider Funktioune fir d'Liewe méi einfach ze maachen: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Nodeem s de puer Dokumenter dobäigesat goufen a mat Metadate markéiert sinn, ka paperless dës Metadaten bei neien Dokumenter automatesch dobäisetzen. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Paperless ka konfiguréiert ginn, fir E-Mailen ze liesen an Dokumenter aus ugehaangene Fichieren anzeliesen. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consultéiert d'Dokumentatioun fir méi Detailer zu der Verwendung vun dëse Funktiounen. De Beräich iwwer déi allgemeng Benotzung huet och e puer Informatiounen, wéi paperless am Allgemenge benotzt ka ginn. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadaten + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Auswielen + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + W. e. g. en Objet auswielen + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Ongëltegen Datum. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Dokument mat ASN gëtt gesicht + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Jo + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nee + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (keen Titel) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Englesch (USA) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Englesch (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Däitsch + + + Dutch + + src/app/services/settings.service.ts + 93 + + Hollännesch + + + French + + src/app/services/settings.service.ts + 94 + + Franséisch + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugisesch + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugisesch (Brasilien) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italienesch + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumänesch + + + Russian + + src/app/services/settings.service.ts + 99 + + Russesch + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spuenesch + + + Polish + + src/app/services/settings.service.ts + 101 + + Polnesch + + + Swedish + + src/app/services/settings.service.ts + 102 + + Schwedesch + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Dokument existéiert schonn. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Fichier net fonnt. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Skript fir de Virtraitement existéiert net. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Feeler beim ausféiere vum Skript fir d'Virbehandlung. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Skript fir d'Nobehandlung existéiert net. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Feeler beim ausféiere vum Skript fir d'Nobehandlung. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Neie Fichier emfaangen. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Fichierstyp net ënnerstëtzt. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Dokument gëtt veraarbecht... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Virschaubild gëtt generéiert... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Datum vum Dokument gëtt ermëttelt... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Dokument gëtt gespäichert... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Ofgeschloss. + + + Error + + src/app/services/toast.service.ts + 35 + + Feeler + + + Information + + src/app/services/toast.service.ts + 39 + + Informatioun + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Korrespondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Dokumententyp + + + Created + + src/app/services/rest/document.service.ts + 21 + + Erstallt + + + Added + + src/app/services/rest/document.service.ts + 22 + + Dobäigesat + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Verännert + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Pertinenz + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Neit Element erstellen + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Element beaarbechten + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Element konnt net gespäichert ginn: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatesch + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Soll d'Element wierklech geläscht ginn? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Assoziéiert Dokumenter ginn net geläscht. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Läschen + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Feeler beim Läsche vum Element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Iergendee Wuert + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Iergendee Wuert: Dokument enthält eent vun dëse Wierder (getrennt mat engem Espace) + + + All words + + src/app/data/matching-model.ts + 13 + + All d'Wierder + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Iergendee Wuert: Dokument enthält all dës Wierder (getrennt mat engem Espace) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exakten Treffer + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exakt: Dokument enthält dësen Text + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regulären Ausdrock + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regulären Ausdrock: D'Dokument passt zu dësem regulären Ausdrock + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Ongenaut Wuert + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Ongenau: Dokument enthält e Wuert dat ähnlech ass wéi dëst Wuert + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Zouweisung automatesch léieren + + + + diff --git a/src-ui/src/locale/messages.nl_NL.xlf b/src-ui/src/locale/messages.nl_NL.xlf new file mode 100644 index 000000000..e901e802b --- /dev/null +++ b/src-ui/src/locale/messages.nl_NL.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document toegevoegd + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document werd toegevoegd aan paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Kon niet toevoegen : + + + New document detected + + src/app/app.component.ts + 65 + + Nieuw document gedetecteerd + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document wordt verwerkt door paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documenten + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + View "" met succes opgeslagen. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + View "" met succes gemaakt. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selecteer + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Niets selecteren + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Selecteer pagina + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Alles selecteren + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sorteren + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Opslaan als... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Opslaan "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 { van één document geselecteerd} other { van documenten geselecteerd}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Eén document} other { documenten}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (gefilterd) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titel + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Documenttype + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Aangemaakt + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Toegevoegd + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Bevestig het verwijderen + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Wilt u het document echt verwijderen ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + De bestanden voor dit document worden definitief verwijderd. Deze bewerking kan niet ongedaan worden gemaakt. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Verwijder document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Fout bij het verwijderen van het document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Verwijderen + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Meer zoals dit + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Sluiten + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Inhoud + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Negeren + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Opslaan + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Pagina + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + van + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download origineel + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archief serienummer + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Aanmaakdatum + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Wijzigingsdatum + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Datum toegevoegd + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media bestandsnaam + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Originele MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Originele bestandsgrootte + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Oorspronkelijke mime-type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archief MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archief bestandsgrootte + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Originele document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Gearchiveerde document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Opslaan & volgende + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hallo , welkom bij Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welkom bij Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Wil je dit label echt verwijderen ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Labels + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Maak + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter op: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Naam + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Kleur + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Overeenkomend + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Aantal documenten + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Acties + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documenten + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Bewerk + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Wilt u het documenttype echt verwijderen ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Documenttypen + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Opgeslagen view "" verwijderd. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Instellingen succesvol opgeslagen. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Gebruik de systeemtaal + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Datumopmaak van weergavetaal gebruiken + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Fout bij het opslaan van de instellingen: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Instellingen + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Algemene instellingen + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Meldingen + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Opgeslagen views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Uiterlijk + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Taal + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + De pagina moet herladen worden nadat een andere taal werd gekozen. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Datumweergave + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Datumopmaak + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Kort: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Lang: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Resultaten per pagina + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Documentbewerker + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Gebruik de PDF lezer ingebouwd in uw browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Dit is gewoonlijk sneller om grote PDF's te tonen, maar werkt niet bij elke browser. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Nachtmodus + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Gebruik systeeminstellingen + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Gebruik nachtmodus + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Toon negatief van voorbeeldweergave in nachtmodus + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Massabewerking + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Toon dialoogvenster ter bevestiging + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Er zal altijd om bevestiging gevraagd worden voor het verwijderen van documenten. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Toepassen bij het sluiten + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Verwerking van documenten + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Toon meldingen wanneer nieuwe documenten worden gedetecteerd + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Toon een melding wanneer documenten met succes zijn verwerkt + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Toon meldingen wanneer het verwerken van documenten faalt + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Onderdruk meldingen op het dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Dit verbergt alle statusberichten op het dashboard die met het verwerken van documenten te maken hebben. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Komt voor bij + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Toon op het dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Toon in de zijbalk + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Geen opgeslagen views gedefinieerd. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Niet Gevonden + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Bent u zeker dat u correspondent "" wilt verwijderen? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondenten + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Laatste correspondentie + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Bevestiging + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Bevestig + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Annuleren + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Nieuwe correspondent maken + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Correspondent bewerken + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritme voor het bepalen van de overeenkomst + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Patroon voor overeenkomst + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Niet hoofdlettergevoelig + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Nieuw label aanmaken + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Etiket bewerken + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + "Postvak in"-label + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + "Postvak in"-labels worden automatisch toegewezen aan alle verwerkte documenten. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Nieuw documenttype maken + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Documenttype bewerken + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Documenten doorzoeken + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Afmelden + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Beheren + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Beheer + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informatie + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Handleiding + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Ideeënbus + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Aangemeld als + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documenten + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Alles sluiten + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Zonder correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Zonder documenttype + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Label: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Zonder label + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titel: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titel + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titel en inhoud + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Geavanceerd zoeken + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Meer zoals + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Labels filteren + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Correspondenten filteren + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Documenttypes filteren + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Filters terug zetten + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Zonder toewijzing + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Toepassen + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Afgelopen 7 dagen + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Afgelopen maand + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Afgelopen 3 maanden + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Afgelopen jaar + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Na + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Voor + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Leegmaken + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Bekijken + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Op correspondent filteren + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Op label filteren + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Score: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Aangemaakt op: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Toegevoegd op: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Gewijzigd op: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Fout bij het uitvoeren van een massabewerking: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" en "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + en "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Bevestig toewijzen van labels + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Het label "" zal aan geselecteerd(e) document(en) worden toegewezen. + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + De labels zullen aan geselecteerd(e) document(en) worden toegewezen. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Het label "" zal verwijderd worden van geselecteerd(e) document(en). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + De labels zullen verwijderd worden van geselecteerd(e) document(en). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + De labels zullen toegevoegd worden aan, en de labels zullen verwijderd worden van geselecteerd(e) document(en). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Bevestig toewijzen van correspondent + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + De correspondent "" zal aan geselecteerd(e) document(en) worden toegewezen. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + De correspondent zal verwijderd worden van geselecteerd(e) document(en). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Bevestig toewijzen van documenttype + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Het documenttype "" zal aan geselecteerd(e) document(en) worden toegewezen. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Het documenttype zal verwijderd worden van geselecteerd(e) document(en). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Bevestig verwijderen + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + geselecteerd(e) document(en) zullen permanent worden verwijderd. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Deze actie kan niet ongedaan worden gemaakt. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Verwijder document(en) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Selecteer: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Alles + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Bewerk: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Originelen downloaden + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Item toevoegen + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggesties: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Huidige view opslaan + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Label toevoegen + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Alles weergeven + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistieken + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Totaal aantal documenten: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documenten in "Postvak in": + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Bezig met verwerken: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Mislukt: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Toegevoegd: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Bezig met verbinden... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Bezig met uploaden... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload voltooid, wachten op verwerking... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP fout: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Nieuwe documenten uploaden + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Sleep documenten hierheen, of + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Kies bestanden + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Verberg verwerkte documenten + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Nog één document} other {Nog documenten}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Laten we aan de slag gaan + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is gestart! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Je kunt documenten uploaden door bestanden in het upload vak rechts te slepen of door ze in de map te kopiëren die geconfigureerd werd als consumptiemap. Ze zullen vervolgens verschijnen in de lijst met documenten. Nadat je metadata hebt toegevoegd aan je documenten, kun je de filters in Paperless gebruiken om eigen views te maken (zoals 'Recent toegevoegd', 'Te bekijken'). Deze views zullen hier verschijnen in plaats van dit bericht. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless heeft nog meer in petto om jouw leven makkelijker te maken: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Zodra je documenten hebt toegevoegd en metadata hebt toegewezen, kan Paperless die metadata automatisch aan nieuwe documenten toewijzen. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Je kunt Paperless configureren om mails te lezen en documenten toe te voegen vanuit bijlagen. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Raadpleeg de documentatie over het gebruik van deze functies. De sectie over basisgebruik bevat ook enige informatie over het gebruik van Paperless in het algemeen. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Selecteer + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Selecteer een object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Ongeldige datum. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Document zoeken met asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Ja + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nee + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (geen titel) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Engels (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Engels (Brits) + + + German + + src/app/services/settings.service.ts + 92 + + Duits + + + Dutch + + src/app/services/settings.service.ts + 93 + + Nederlands + + + French + + src/app/services/settings.service.ts + 94 + + Frans + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugees + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugees (Brazilië) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiaans + + + Romanian + + src/app/services/settings.service.ts + 98 + + Roemeens + + + Russian + + src/app/services/settings.service.ts + 99 + + Russisch + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spaans + + + Polish + + src/app/services/settings.service.ts + 101 + + Pools + + + Swedish + + src/app/services/settings.service.ts + 102 + + Zweeds + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document bestaat al. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Bestand niet gevonden. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-verwerkingsscript bestaat niet. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fout tijdens het uitvoeren van het pre-verwerkingsscript + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-verwerkingsscript bestaat niet. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fout tijdens het uitvoeren van het post-verwerkingsscript + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Nieuw bestand ontvangen. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Bestandstype niet ondersteund. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Document wordt verwerkt... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Voorbeeldweergave wordt gemaakt... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Datum wordt gezocht in document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Document wordt opgeslagen... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Klaar. + + + Error + + src/app/services/toast.service.ts + 35 + + Fout + + + Information + + src/app/services/toast.service.ts + 39 + + Informatie + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Documenttype + + + Created + + src/app/services/rest/document.service.ts + 21 + + Aangemaakt + + + Added + + src/app/services/rest/document.service.ts + 22 + + Toegevoegd + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Gewijzigd + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Zoekscore + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Maak nieuw item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Item bewerken + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Kon het element niet opslaan: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatisch + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Wil je dit element echt verwijderen? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Gekoppelde documenten zullen niet worden verwijderd. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Verwijderen + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Fout bij het verwijderen van het element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Eender welk woord + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Een of meer: Documenten die een of meer van deze woorden bevatten (gescheiden door spaties) + + + All words + + src/app/data/matching-model.ts + 13 + + Alle woorden + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Alle: Documenten die elk van deze woorden bevatten (gescheiden door spaties) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exacte overeenkomst + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document bevat deze tekenreeks + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Reguliere expressie + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Reguliere expressie: Document voldoet aan deze reguliere expressie + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Gelijkaardig woord + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Gelijkaardig: Document bevat een woord dat lijkt op dit woord + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Automatisch overeenkomsten aanleren + + + + diff --git a/src-ui/src/locale/messages.pl_PL.xlf b/src-ui/src/locale/messages.pl_PL.xlf new file mode 100644 index 000000000..207523b0b --- /dev/null +++ b/src-ui/src/locale/messages.pl_PL.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Dokument dodano + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Dokument został dodany do paperless. + + + Open document + + src/app/app.component.ts + 51 + + Otwórz dokument + + + Could not add : + + src/app/app.component.ts + 59 + + Nie można dodać : + + + New document detected + + src/app/app.component.ts + 65 + + Wykryto nowy dokument + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Dokument jest przetwarzany przez paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokumenty + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Widok "" został zapisany. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Widok "" został utworzony pomyślnie. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Wybierz + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Odznacz wszystko + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Zaznacz wszystko na tej stronie + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Zaznacz wszystko + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sortowanie + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Widoki + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Zapisz jako... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Zapisz "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, one {} few {Wybrano z dokumentów} many {Wybrano z dokumentów} =1 {Wybrano z jednego dokumentu} other {Wybrano z dokumentów}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, one {} few { dokumenty} many { dokumentów} =1 {Jeden dokument} other { dokumenty}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + Filtrowane + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Korespondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Tytuł + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Typ dokumentu + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Utworzony + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Dodano + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Potwierdź usunięcie + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Czy na pewno chcesz usunąć dokument""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Pliki tego dokumentu zostaną trwale usunięte. Tej operacji nie można cofnąć. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Usuń dokument + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Błąd podczas usuwania dokumentu: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Usuń + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Pobierz + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Podobne dokumenty + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Zamknij + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Szczegóły + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Zawartość + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadane + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Zaniechaj + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Zapisz + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Strona + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + z + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Pobierz oryginał + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Numer seryjny archiwum + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Data utworzenia + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Data modyfikacji + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Data dodania + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nazwa pliku + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + MD5 - Suma kontrolna Oryginału + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Rozmiar oryginalnego pliku + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Typ mime oryginału + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Suma kontrolna archiwum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Rozmiar pliku archiwalnego + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadane oryginalnego dokumentu + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadane zarchiwizowanego dokumentu + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Zapisz & następny + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Cześć , witaj w Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Witaj w Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Kokpit + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Czy na pewno chcesz usunąć tag "? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tagi + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Utwórz + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtruj według: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nazwa + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Kolor + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Dopasowanie + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Liczba dokumentów + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Akcje + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokumenty + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edytuj + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Czy na pewno chcesz usunąć typ dokumentu ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Typy dokumentów + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logi + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Zapisany widok "" został usunięty. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Ustawienia zostały zapisane. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Użyj języka systemowego + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Użyj formatu daty według wyświetlanego języka + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Błąd podczas zapisywania ustawień na serwerze: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Ustawienia + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Ustawienia ogólne + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Powiadomienia + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Zapisane widoki + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Wygląd + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Język + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Musisz przeładować stronę po wybraniu nowego języka. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Wyświetlanie daty + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Format daty + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Krótki: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Średni: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Długi: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Ilość elementów na stronie + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Edytor dokumentów + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Wyświetl PDF przez wbudowaną przeglądarkę + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Zazwyczaj działa szybciej przy wyświetlaniu dużych dokumentów w formacie PDF, ale może nie działać na niektórych przeglądarkach. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Tryb ciemny + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Użyj ustawień systemowych + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Użyj ciemnego motywu + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Odwróć kolory miniaturki w trybie ciemnym + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Masowa edycja + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Używaj dialogów potwierdzenia + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Usuwanie dokumentów zawsze będzie wymagało potwierdzenia. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Zamknięcie zapisuje zmiany + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Przetwarzanie dokumentów + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Pokaż powiadomienia, gdy zostaną wykryte nowe dokumenty + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Pokaż powiadomienia, gdy przetwarzanie dokumentu zakończy się pomyślnie + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Pokaż powiadomienia, gdy przetwarzanie dokumentu nie powiedzie się + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Wyłącz powiadomienia na pulpicie + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Wyłącz pokazywanie wszystkich wiadomości o statusie przetwarzania dokumentów na pulpicie. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Wyświetlanie + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Pokaż na pulpicie + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Pokaż w panelu bocznym + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nie zdefiniowano zapisanych widoków. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Nie znaleziono + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Czy na pewno chcesz usunąć nadawcę "? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Nadawcy + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Ostatnia korespondencja + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Potwierdzenie + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Potwierdź + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Anuluj + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Dodaj nowego nadawcę + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edytuj nadawcę + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algorytm dopasowania + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Wzorzec dopasowania + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + (Nieuwzględniający wielkości liter) + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Utwórz tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edytuj tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Tag skrzynki odbiorczej + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Tagi skrzynki odbiorczej są automatycznie przypisane do wszystkich przetworzonych dokumentów. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Utwórz nowy dokument + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edytuj typ dokumentu + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Szukaj dokumentów + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Wyloguj + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Zarządzaj + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Dokumentacja + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Zaproponuj pomysł + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Zalogowano jako + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Otwarte dokumenty + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Zamknij wszystkie + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Nadawca: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Bez nadawcy + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Rodzaj: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Bez typu dokumentu + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Bez żadnego tagu + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Tytuł: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + Numer archiwum: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Tytuł + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Tytuł & treść + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Wyszukiwanie zaawansowane + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Podobne do + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtruj tagi + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtruj nadawców + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtruj typy dokumentów + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Zresetuj filtry + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Nieprzypisane + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Zastosuj + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Ostatnie 7 dni + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Ostatni miesiąc + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Ostatnie 3 miesiące + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Ostatni rok + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Po + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Przed + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Wyczyść + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Pokaż + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtruj według nadawcy + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtruj po tagach + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Wynik: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Utworzony: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Dodany: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Zmodyfikowany: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Błąd podczas wykonywania operacji zbiorczej: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" i "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + i "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Potwierdź przypisanie tagów + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Ta operacja doda tag "" do wybranych dokumentów. + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Ta operacja doda tagi do wybranych dokumentów. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Ta operacja usunie tag "" z wybranych dokumentów. + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Ta operacja usunie tagi z wybranych dokumentów. + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Ta operacja doda tagi i usunie tagi w wybranych dokumentach. + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Potwierdź przypisanie nadawcy + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Ta operacja przypisze nadawcę "" do wybranych dokumentów. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Ta operacja usunie nadawcę z wybranych dokumentów. + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Potwierdź przypisanie typu dokumentu + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Ta operacja przypisze typ dokumentu "" do zaznaczonych dokumentów. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Ta operacja usunie typ dokumentu z wybranych dokumentów. + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Potwierdź Usunięcie + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Ta operacja trwale usunie zaznaczonych dokumentów. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Ta czynność nie może być cofnięta. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Usuń dokument(y) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Wybierz: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Wszystkie + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edycja: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Pobierz oryginały + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Dodaj element + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Sugestie: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Zapisz bieżący widok + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Dodaj tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Pokaż wszystkie + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statystyki + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Łącznie dokumentów: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Dokumenty w skrzynce odbiorczej: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Przetwarzane: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Błąd: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Dodano: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Łączenie... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Wysyłanie... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Przesyłanie zakończone, oczekuje... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Błąd HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Wgraj nowe dokumenty + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Upuść dokumenty tutaj lub + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Wybierz pliki + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Odrzuć zakończone + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, one {} few {Jeszcze dokumenty} many {Jeszcze dokumentów} =1 {Jeszcze jeden dokument} other {Jeszcze dokumentów}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Otwórz dokument + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Pierwsze kroki + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless jest uruchomiony! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Możesz rozpocząć przesyłanie dokumentów, upuszczając je w polu przesyłania plików po prawej stronie lub umieszczając je w skonfigurowanym folderze przetwarzania i zaczną wyświetlać się na liście dokumentów. Po dodaniu metadanych do dokumentów użyj mechanizmów filtrowania Paperless do tworzenia niestandardowych widoków (takich jak "Ostatnio dodane", 'Otagowany TODO') i pojawią się na pulpicie zamiast tej wiadomości. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless oferuje więcej funkcji, które starają się ułatwić Ci życie: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Po zaimportowaniu kilku dokumentów do Paperless i przypisaniu do nich metadanych takich jak Korespondenci, Tagi i Typ dokumentu Paperless może automatycznie przypisać takie same metadane do nowych dokumentów. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Możesz skonfigurować Paperless, aby sprawdzał Twoje konto e-mail i dodawał dokumenty z załączonych plików. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Sprawdź w dokumentacji jak korzystać z tych funkcji. Sekcja o podstawowym użytkowaniu zawiera również ogólne informacje jak używać Paperless. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadane + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Wybierz + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Wybierz obiekt + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Niepoprawna data. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Wyszukiwanie dokumentu z ASN + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Tak + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nie + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (bez tytułu) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Angielski (USA) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Angielski (Wielka Brytania) + + + German + + src/app/services/settings.service.ts + 92 + + Niemiecki + + + Dutch + + src/app/services/settings.service.ts + 93 + + Holenderski + + + French + + src/app/services/settings.service.ts + 94 + + Francuski + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugalski + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugalski (Brazylia) + + + Italian + + src/app/services/settings.service.ts + 97 + + Włoski + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumuński + + + Russian + + src/app/services/settings.service.ts + 99 + + Rosyjski + + + Spanish + + src/app/services/settings.service.ts + 100 + + Hiszpański + + + Polish + + src/app/services/settings.service.ts + 101 + + Polski + + + Swedish + + src/app/services/settings.service.ts + 102 + + Szwedzki + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Dokument już istnieje. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Nie znaleziono pliku. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Skrypt wstępnego przetwarzania nie istnieje. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Wystąpił błąd podczas wykonywania skryptu wstępnego przetwarzania. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Skrypt przetwarzania końcowego nie istnieje. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Wystąpił błąd podczas wykonywania skryptu końcowego przetwarzania. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Otrzymano nowy plik. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Ten typ pliku nie jest obsługiwany. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Przetwarzanie dokumentu... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generowanie miniaturki... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Pobieranie daty z dokumentu... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Zapisywanie dokumentu... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Ukończono. + + + Error + + src/app/services/toast.service.ts + 35 + + Błąd + + + Information + + src/app/services/toast.service.ts + 39 + + Informacje + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Korespondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Typ dokumentu + + + Created + + src/app/services/rest/document.service.ts + 21 + + Utworzony + + + Added + + src/app/services/rest/document.service.ts + 22 + + Dodano + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Zmodyfikowano + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Dopasowanie + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Utwórz nowy element + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edytuj element + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Nie można zapisać elementu: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatyczny + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Czy na pewno chcesz usunąć ten element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Powiązane dokumenty nie będą usunięte. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Usuń + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Błąd podczas usuwania elementu: + + + Any word + + src/app/data/matching-model.ts + 12 + + Dowolne słowo + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Dowolny: Dokument zawiera dowolne z tych słów (oddzielone spacją) + + + All words + + src/app/data/matching-model.ts + 13 + + Wszystkie słowa + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Wszystkie: Dokument zawiera wszystkie poniższe słowa (oddzielone spacją) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Dokładne dopasowanie + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Dokładne: Dokument zawiera ten ciąg znaków + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Wyrażenie regularne + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Wyrażenie regularne: Dokument pasuje do tego wyrażenia regularnego + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Dopasowanie rozmyte + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Rozmyte: Dokument zawiera słowo podobne do tego słowa + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: ucz się dopasowywać automatycznie + + + + diff --git a/src-ui/src/locale/messages.pt_BR.xlf b/src-ui/src/locale/messages.pt_BR.xlf new file mode 100644 index 000000000..e1328cea1 --- /dev/null +++ b/src-ui/src/locale/messages.pt_BR.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Documento adicionado + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Documento foi adicionado ao paperless. + + + Open document + + src/app/app.component.ts + 51 + + Abrir documento + + + Could not add : + + src/app/app.component.ts + 59 + + Não foi possível adicionar : + + + New document detected + + src/app/app.component.ts + 65 + + Novo documento detectado + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Documento está sendo processado pelo paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documentos + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Visualização "" salva com sucesso. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Visualização "" criada com sucesso. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selecionar + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Selecionar nenhum + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Selecionar página + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Selecionar todos + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Ordenar + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Visualizações + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Salvar como... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Salvar "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selecionado de um documento} other {Selecionado of documentos}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Um documento} other { documentos}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrado) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + NSA + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondente + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Título + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Tipo de Documento + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Criado + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Adicionado + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirmar exclusão + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Você realmente deseja excluir o documento ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Os arquivos desse documento serão excluídos permanentemente. Essa operação não pode ser revertida. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Excluir documento + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Erro ao excluir documento: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Excluir + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Baixar + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Mais como este + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Fechar + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detalhes + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Conteúdo + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadados + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Descartar + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Salvar + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Página + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + de + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Baixar original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Número de série de arquivamento + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Data de criação + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Data de modificação + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Data de adição + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nome do arquivo + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Soma de verificação MD5 original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Tamanho do arquivo original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Tipo mime original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Soma de verificação MD5 de arquivamento + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Tamanho arquivado + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadados do documento original + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadados do documento arquivado + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Salvar & próximo + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Olá , bem-vindo ao Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Bem-vindo ao Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Painel de controle + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Você realmente deseja excluir a etiqueta ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etiquetas + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Criar + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrar por: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nome + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Cor + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Detecção + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Número de documentos + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Ações + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documentos + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Editar + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Você realmente deseja excluir o tipo de documento ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Tipos de documento + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Visualização "" excluída. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Configurações salvas com sucesso. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Usar linguagem do sistema + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Usar formato de data da linguagem de exibição + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Erro ao salvar configurações: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Configurações + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Configurações gerais + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notificações + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Visualizações + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Aparência + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Linguagem de exibição + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Você precisar recarregar a página depois de escolher uma nova linguagem. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Exibição de data + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Formato de data + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Curto: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Médio: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Longo: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Itens por página + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Editor de documentos + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Usar visualizador de PDF do navegador + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Isso é geralmente mais rápido para exibir grandes documentos PDF, mas poderá não funcionar em alguns navegadores. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Modo noturno + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Usar configurações do sistema + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Habilitar modo noturno + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Inverter imagens em modo noturno + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Edição em massa + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Mostrar janelas de confirmação + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Ao excluir um documento, sempre será pedido uma confirmação. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Aplicar ao fechar + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Processamento de documentos + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Exibir notificações quando novos documentos forem detectados + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Exibir notificações quando o processamento de um documento concluir com sucesso + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Exibir notificações quando o processamento de um documento falhar + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Não exibir notificações no painel de controle + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Isso esconderá todas as mensagens sobre o status de processamento de documentos no painel de controle. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Aparece em + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Exibir no painel de controle + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Mostrar na navegação lateral + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nenhuma visualização definida. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Não Encontrado + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Você realmente deseja excluir o correspondente ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondentes + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Última correspondência + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmação + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirmar + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancelar + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Criar novo correspondente + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Editar correspondente + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritmo de detecção + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Padrão de detecção + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Não diferenciar maiúsculas de minúsculas + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Criar nova etiqueta + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Editar etiqueta + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Etiqueta caixa de entrada + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Etiquetas de caixa de entrada são atribuídas automaticamente para todos os documentos consumidos. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Criar novo tipo de documento + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Editar tipo de documento + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Procurar documentos + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Encerrar sessão + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Gerenciar + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informação + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentação + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Sugerir uma idéia + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Sessão iniciada como + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Abrir documentos + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Fechar todos + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondente: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Sem correspondente + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Tipo: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Sem tipo de documento + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Etiqueta: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Sem etiquetas + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Título: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Título + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Título & conteúdo + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + NSA + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Advanced search + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + More like + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrar etiquetas + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrar correspondentes + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrar tipos de documento + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Limpar filtros + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Não atribuído + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Aplicar + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Últimos 7 dias + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Último mês + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Últimos 3 meses + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Último ano + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Antes + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Depois + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Limpar + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Ver + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrar por correspondente + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrar por etiqueta + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Nota: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Created: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Added: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modified: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Erro ao executar operação em massa: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" e "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + e "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirmar atribuição de etiqueta + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Essa operação irá adicionar a etiqueta "" em documento(s) selecionado(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Essa operação irá adicionar as etiquetas em documento(s) selecionado(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Essa operação irá remover a etiqueta "" de documento(s) selecionado(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Essa operação irá remover as etiquetas de documento(s) selecionado(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Essa operação irá adicionar as etiquetas e remover as etiquetas de documento(s) selecionado(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirmar atribuição de correspondente + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Essa operação irá atribuir o correspondente "" para documento(s) selecionado(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Essa operação irá remover o correspondente de documento(s) selecionado(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirmar atribuição de tipo de documento + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Essa operação irá atribuir o tipo de documento "" para documento(s) selecionado(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Essa operação irá remover o tipo de documento de documento(s) selecionado(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Confirmar exlcusão + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Essa operação irá excluir permanentemente documento(s) selecionado(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Essa operação não pode ser revertida. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Apagar documento(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Selecionar: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Todos + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Editar: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Descarregar originais + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Sugestões: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Salvar visualização atual + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Mostrar todos + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Estatísticas + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total de documentos: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documentos na caixa de entrada: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processando: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Falha: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Adicionado: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Conectando... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Enviando... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Envio concluído, esperando... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Erro HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Enviar novos documentos + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Arraste documentos aqui ou + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Navegar arquivos + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Esconder completos + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Mais um documento} other {Mais documentos}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Abrir documento + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Primeiros passos + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + O Paperless está a correr! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Pode começar a carregar documentos deixando-os na caixa de carregamento de ficheiros à direita, ou deixando-os na pasta de consumo pré-configurada, que eles depois aparecem na lista de documentos. Depois de ter adicionado alguns meta-dados aos seus documentos, utilize os mecanismos de filtragem do Paperless para criar visualizações personalizadas (tais como 'Adicionados recentemente', 'Etiquetados com PARAFAZER'). Posteriormente estes aparecerão no painel em vez desta mensagem. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + O Paperless oferece algumas funcionalidades para tentar tornar a sua vida mais fácil: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Assim que já tenha alguns documentos previamente adicionados com os respetivos meta-dados, o Paperless consegue adicionar automaticamente meta-dados aos novos documentos. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Pode configurar o Paperless para ler os seus e-mails e adicionar os documentos em anexo à plataforma. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consulte a documentação para saber como utilizar estas funcionalidades. A secção sobre utilização básica têm informação para o ajudar a utilizar a plataforma de uma forma geral. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadados + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Selecione + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Por favor selecione um objeto + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Data inválida. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Searching document with asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Sim + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Não + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (sem título) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Inglês (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Inglês (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Alemão + + + Dutch + + src/app/services/settings.service.ts + 93 + + Holandês + + + French + + src/app/services/settings.service.ts + 94 + + Francês + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portuguese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Português (Brasil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiano + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romeno + + + Russian + + src/app/services/settings.service.ts + 99 + + Russian + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanish + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Documento já existente. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Arquivo não encontrado. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Script pré-consumo não existe. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erro ao executar script pré-consumo. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Script pós-consumo não existe. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erro ao executar script pós-consumo. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Novo arquivo recebido. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Tipo de arquivo não suportado. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processando documento... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Gerando imagem... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Buscando data do documento... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Salvando documento... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Encerrado. + + + Error + + src/app/services/toast.service.ts + 35 + + Erro + + + Information + + src/app/services/toast.service.ts + 39 + + Informação + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondente + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Tipo de Documento + + + Created + + src/app/services/rest/document.service.ts + 21 + + Criado + + + Added + + src/app/services/rest/document.service.ts + 22 + + Adicionado + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modificado + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Criar novo item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Editar item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Não podemos salvar elemento: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automático + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Você realmente quer excluir esse elemento? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Documentos associados não serão excluidos. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Excluir + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Erro ao excluir elemento: + + + Any word + + src/app/data/matching-model.ts + 12 + + Qualquer palavra + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Qualquer: Documento contém qualquer uma dessas palavras (separadas por espaço) + + + All words + + src/app/data/matching-model.ts + 13 + + Todas as palavras + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Todas: Documento contém todas essas palavras (separadas por espaço) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Detecção exata + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exata: Documento contém essa palavra + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Expressão regular + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Expressão regular: Documento condiz com essa expressão regular + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Palavra difusa + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Documento contém uma palavra similar à essa. + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Aprender detecção automaticamente + + + + diff --git a/src-ui/src/locale/messages.pt_PT.xlf b/src-ui/src/locale/messages.pt_PT.xlf new file mode 100644 index 000000000..abcaddd21 --- /dev/null +++ b/src-ui/src/locale/messages.pt_PT.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Documento adicionado + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Documento foi adicionado ao paperless. + + + Open document + + src/app/app.component.ts + 51 + + Abrir documento + + + Could not add : + + src/app/app.component.ts + 59 + + Não foi possível adicionar : + + + New document detected + + src/app/app.component.ts + 65 + + Novo documento detectado + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Documento está sendo processado pelo paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documentos + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Visualização "" salva com sucesso. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Visualização "" criada com sucesso. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selecionar + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Selecionar nenhum + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Selecionar página + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Selecionar todos + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Ordenar + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Visualizações + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Salvar como... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Salvar "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, one {} =1 {Selecionado de um documento} other {Selecionados de documentos}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, one {} =1 {Um documento} other { documentos}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrado) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + NSA + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondente + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Título + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Tipo de Documento + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Criado + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Adicionado + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirmar exclusão + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Você realmente deseja excluir o documento ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Os arquivos desse documento serão excluídos permanentemente. Essa operação não pode ser revertida. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Excluir documento + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Erro ao excluir documento: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Excluir + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Baixar + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Mais como este + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Fechar + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detalhes + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Conteúdo + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadados + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Descartar + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Salvar + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Página + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + de + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Baixar original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Número de série de arquivamento + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Data de criação + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Data de modificação + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Data de adição + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Nome do arquivo + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Soma de verificação MD5 original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Tamanho do arquivo original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Tipo mime original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Soma de verificação MD5 de arquivamento + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Tamanho arquivado + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadados do documento original + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadados do documento arquivado + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Salvar & próximo + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Olá , bem-vindo ao Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Bem-vindo ao Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Painel de controle + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Você realmente deseja excluir a etiqueta ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etiquetas + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Criar + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrar por: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nome + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Cor + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Detecção + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Número de documentos + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Ações + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documentos + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Editar + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Você realmente deseja excluir o tipo de documento ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Tipos de documento + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Visualização "" excluída. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Configurações salvas com sucesso. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Usar linguagem do sistema + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Usar formato de data da linguagem de exibição + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Erro ao salvar configurações: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Configurações + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Configurações gerais + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notificações + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Visualizações + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Aparência + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Linguagem de exibição + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Você precisar recarregar a página depois de escolher uma nova linguagem. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Exibição de data + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Formato de data + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Curto: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Médio: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Longo: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Itens por página + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Editor de documentos + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Usar visualizador de PDF do navegador + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Isso é geralmente mais rápido para exibir grandes documentos PDF, mas poderá não funcionar em alguns navegadores. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Modo noturno + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Usar configurações do sistema + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Habilitar modo noturno + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Inverter miniaturas no modo escuro + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Edição em massa + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Mostrar janelas de confirmação + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Ao excluir um documento, sempre será pedido uma confirmação. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Aplicar ao fechar + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Processamento de documentos + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Exibir notificações quando novos documentos forem detectados + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Exibir notificações quando o processamento de um documento concluir com sucesso + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Exibir notificações quando o processamento de um documento falhar + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Não exibir notificações no painel de controle + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Isso esconderá todas as mensagens sobre o status de processamento de documentos no painel de controle. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Aparece em + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Exibir no painel de controle + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Mostrar na navegação lateral + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nenhuma vista gravada definida. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Não encontrado + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Tem a certeza que deseja eliminar a correspondência ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondentes + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Última correspondência + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmação + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirmar + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancelar + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Criar nova correspondência + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Editar correspondente + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritmo correspondente + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Padrão correspondente + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Não distingue maiúsculas de minúsculas + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Criar uma nova etiqueta + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Editar etiqueta + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Etiqueta de novo + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + As etiquetas de novo são automaticamente adicionadas a todos os documentos consumidos. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Criar novo tipo de documento + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Editar tipo de documento + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Procurar documentos + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Terminar sessão + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Gerir + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administrador + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informação + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentação + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + Github + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Sugerir uma ideia + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Sessão iniciada como + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Abrir documentos + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Fechar todos + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondente: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Sem correspondente + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Tipo: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Sem tipo de documento + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Etiqueta: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Sem etiquetas + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Título: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + NSA: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Título + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Título & conteúdo + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + NSA + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Pesquisa avançada + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Mais semelhante + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrar etiquetas + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrar correspondentes + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrar tipos de documentos + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Limpar filtros + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Não atribuído + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Aplicar + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Últimos 7 dias + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Último mês + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Últimos 3 meses + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Último ano + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Antes + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Depois + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Limpar + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Ver + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrar por correspondente + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrar por etiqueta + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Pontuação: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Criado em: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Adicionado a: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modificado a: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Erro ao executar operação em massa: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" e "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + e "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirmar atribuição de etiquetas + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Esta operação irá adicionar a etiqueta "" ao(s) documento(s) selecionado(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Esta operação irá adicionar as etiquetas ao(s) documento(s) selecionado(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Esta operação irá remover a etiqueta "" do(s) documento(s) selecionado(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Esta operação irá remover as etiquetas do(s) documento(s) selecionado(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Esta operação irá adicionar a(s) etiqueta(s) e remover a(s) no(s) documento(s) selecionado(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirmar atribuição correspondente + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Esta operação irá adicionar a correspondência aos documento(s) selecionado(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Esta operação irá remover a correspondência do(s) documento(s) selecionado(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirmar atribuição de tipo de documentos + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + A operação irá associar o tipo de documento "" ao(s) documento(s) selecionado(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Esta operação irá remover o tipo de documento do(s) documento(s) selecionado(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Confirmar eliminação + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Esta operação irá apagar permanentemente o(s) documento(s) selecionado(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Esta operação não pode ser desfeita. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Apagar documento(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Selecionar: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Todos + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Editar: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Descarregar originais + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Adicionar item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Sugestões: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Guardar a vista atual + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Adicionar etiqueta + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Mostrar tudo + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Estatísticas + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total de documentos: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documentos novos: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processando: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Falha: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Adicionado: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + A ligar... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + A carregar... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Carregamento concluído, à espera... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Erro HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Carregar novos documentos + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Largar documentos aqui ou + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Navegar ficheiros + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Descartar concluído + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, one {} =1 {Mais um documento} other {mais documentos}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Abrir documento + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Primeiros passos + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + O Paperless está a correr! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Pode começar a carregar documentos deixando-os na caixa de carregamento de ficheiros à direita, ou deixando-os na pasta de consumo pré-configurada, que eles depois aparecem na lista de documentos. Depois de ter adicionado alguns meta-dados aos seus documentos, utilize os mecanismos de filtragem do Paperless para criar visualizações personalizadas (tais como 'Adicionados recentemente', 'Etiquetados com PARAFAZER'). Posteriormente estes aparecerão no painel em vez desta mensagem. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + O Paperless oferece algumas funcionalidades para tentar tornar a sua vida mais fácil: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Assim que já tenha alguns documentos previamente adicionados com os respetivos meta-dados, o Paperless consegue adicionar automaticamente meta-dados aos novos documentos. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Pode configurar o Paperless para ler os seus e-mails e adicionar os documentos em anexo à plataforma. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consulte a documentação para saber como utilizar estas funcionalidades. A secção sobre utilização básica têm informação para o ajudar a utilizar a plataforma de uma forma geral. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadados + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Selecione + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Por favor selecione um objeto + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Data inválida. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + À procura do documento com nsa + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Sim + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Não + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (sem título) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Inglês (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Inglês (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Alemão + + + Dutch + + src/app/services/settings.service.ts + 93 + + Holandês + + + French + + src/app/services/settings.service.ts + 94 + + Francês + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Português + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Português (Brasil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiano + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romeno + + + Russian + + src/app/services/settings.service.ts + 99 + + Russo + + + Spanish + + src/app/services/settings.service.ts + 100 + + Espanhol + + + Polish + + src/app/services/settings.service.ts + 101 + + Polaco + + + Swedish + + src/app/services/settings.service.ts + 102 + + Sueco + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Documento já existente. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Arquivo não encontrado. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Script pré-consumo não existe. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erro ao executar script pré-consumo. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Script pós-consumo não existe. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Erro ao executar script pós-consumo. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Novo arquivo recebido. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Tipo de arquivo não suportado. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processando documento... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Gerando imagem... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Buscando data do documento... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Salvando documento... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Encerrado. + + + Error + + src/app/services/toast.service.ts + 35 + + Erro + + + Information + + src/app/services/toast.service.ts + 39 + + Informação + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondente + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Tipo de Documento + + + Created + + src/app/services/rest/document.service.ts + 21 + + Criado + + + Added + + src/app/services/rest/document.service.ts + 22 + + Adicionado + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modificado + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Pesquisar pontuação + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Criar novo item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Editar item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Não podemos salvar elemento: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automático + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Você realmente quer excluir esse elemento? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Documentos associados não serão excluidos. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Excluir + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Erro ao excluir elemento: + + + Any word + + src/app/data/matching-model.ts + 12 + + Qualquer palavra + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Qualquer: Documento contém qualquer uma dessas palavras (separadas por espaço) + + + All words + + src/app/data/matching-model.ts + 13 + + Todas as palavras + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Todas: Documento contém todas essas palavras (separadas por espaço) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Detecção exata + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exata: Documento contém essa palavra + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Expressão regular + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Expressão regular: Documento condiz com essa expressão regular + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Palavra difusa + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Documento contém uma palavra similar à essa. + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Aprender detecção automaticamente + + + + diff --git a/src-ui/src/locale/messages.ro_RO.xlf b/src-ui/src/locale/messages.ro_RO.xlf new file mode 100644 index 000000000..441be5de7 --- /dev/null +++ b/src-ui/src/locale/messages.ro_RO.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document adăugat + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Documentul a fost adăugat. + + + Open document + + src/app/app.component.ts + 51 + + Deschide document + + + Could not add : + + src/app/app.component.ts + 59 + + Nu s-a putut adăuga : + + + New document detected + + src/app/app.component.ts + 65 + + Document nou detectat + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Documentul este în procesare. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Documente + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Vizualizarea "" a fost salvată. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Vizualizarea "" a fost creată. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Selectează + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Deselectează tot + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Selectează pagina + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Selectează tot + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sortează + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Vizualizări + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Salvează ca... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Salvează "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Am selectat din 1 document} other {Am selectat din documente}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Un document} other { documente}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrat) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + Aviz prealabil de expediție + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Corespondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titlu + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Tipul documentului + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Creat + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Adăugat + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirmă ștergerea + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Sunteţi sigur că doriţi să ştergeţi documentul ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Fișierele pentru acest document vor fi șterse permanent. Operațiunea este ireversibila. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Șterge document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Eroare la ștergerea documentului: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Şterge + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Descarcă + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Similare + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Închide + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detalii + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Conținut + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadate + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Renunță + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Salvează + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Pagina + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + din + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Descarcă originalul + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Număr serial în arhivă + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Data creării + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Data ultimei modificări + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Data adăugării + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Numele fișierului media + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + MD5 original + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Dimensiunea fișierului original + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Tip MIME original + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + MD5 arhivă + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Mărimea arhivei + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Metadatele documentului original + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Metadatele documentului arhivat + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Salvează și continuă + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Salut, , bun venit la Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Bun venit la Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Tablou de bord + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Sunteți sigur că doriți să ștergeți documentul ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Etichete + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Creează + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrează după: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Nume + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Culoare + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Potrivire + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Număr de documente + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Acțiuni + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documente + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Editează + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Sunteți sigur că doriți să ștergeți tipul de document ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Tipuri de documente + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Jurnale + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Vizualizarea "" a fost ștearsă. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Setările au fost salvate. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Utilizați limba sistemului + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Folosiți formatul datei corespunzător limbii de afișare + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Eroare la stocarea setărilor pe server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Setări + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Setări generale + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notificări + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Vizualizări + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Afișare + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Limba de afișare + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Trebuie să reîncărcați pagina după aplicarea unei noi limbi. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Afișarea datei + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Formatul datei + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Scurt: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Mediu: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Lung: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Articole pe pagina + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Editor de documente + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Utilizați vizualizatorul PDF furnizat de browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Această opțiune este mai rapidă pentru PDF-uri mari, dar nu este compatibilă cu toate browserele. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Mod întunecat + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Folosește setările sistemului + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Activează modul întunecat + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Inversează miniaturile în modul întunecat + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Editare în bloc + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Afișare dialoguri de confirmare + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Ștergerea documentelor va necesita întotdeauna confirmare. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Aplică la ieșire + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Procesarea documentelor + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Arată notificări atunci când sunt detectate documente noi + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Arată notificări atunci când procesarea documentului se termină cu succes + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Arată notificări atunci când procesarea documentului eșuează + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Ascunde notificările pe tabloul de bord + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Această setare va opri mesajele despre procesarea documentelor pe tabloul de bord. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Apare pe + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Afișează pe tabloul de bord + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Afișează in bara laterala + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Nu sunt definite vizualizări. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Pagina nu a fost gasită + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Sunteți sigur că doriți să ștergeți corespondentul ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Corespondenți + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Ultima corespondență + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmare + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirmă + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Anulează + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Creare corespondent nou + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Modifică un corespondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Algoritm de potrivire + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Model de potrivire + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Ignoră majusculele + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Crează o etichetă nouă + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Modifică o etichetă + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Eticheta inbox + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Etichetele inbox sunt atribuite automat documentelor consumate. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Crează un nou tip de document + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Modifică un tip de document + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Caută documente + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Deconectare + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Administrează + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Administrator + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Informații + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentație + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Sugestii + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Autentificat ca + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Deschide documente + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Închide tot + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Corespondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Fără corespondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Tip: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Fară tip + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Eticheta: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Fară etichete + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titlu: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + Aviz prealabil de expediție: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titlu + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titlu si conținut + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + Aviz prealabil de expediție + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Căutare avansată + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Asemănătoare + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrează etichete + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrează corespondenți + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrează tipuri de documente + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Resetare filtre + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Nealocate + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Aplică + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Ultimele 7 zile + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Ultima lună + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Ultimele 3 luni + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Ultimul an + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + După + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Înainte + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Curăță + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Vizualizare + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrează dupa corespondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrează dupa etichetă + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Scor: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + S-a creat: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + S-a adăugat: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + S-a modificat: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Eroare la executarea operațiunii în bloc: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" și "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + și "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirmați atribuirea etichetelor + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Va fi adăugată eticheta "" la document(e) selectat(e). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Vor fi adăugate etichetele la document(e) selectat(e). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Va fi eliminată eticheta "" din document(e) selectat(e). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Vor fi eliminate etichetele din document(e) selectat(e). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Vor fi adăugate etichetele și vor fi eliminate etichetele pentru document(e) selectat(e). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirmați atribuirea corespondentului + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Va fi adăugat corespondentul "" la document(e) selectat(e). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Va fi eliminat corespondentul din document(e) selectat(e). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirmați atribuirea tipului de document + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Va fi atribuit tipul "" pentru document(e) selectat(e). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Va fi eliminat tipul pentru document(e) selectat(e). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Confirmați stergerea + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + document(e) selectat(e) vor fi șterse permanent. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Această operațiune este ireversibilă. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Șterge document(e) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Selectează: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Tot + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Editează: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Descarcă originalele + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Adaugă element + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Sugestii: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Salvează vizualizarea curenta + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Adaugă etichetă + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Arată tot + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistici + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documente: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documente in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + În procesare: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Eșuate: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Adăugate: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Conectare... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Încarcare... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Încarcare completa, în așteptare... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Eroare HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Încarcă documente noi + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Trage documentele aici sau + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Caută fișiere + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Înlătură operațiunile finalizate + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Încă un document} other {Încă documente}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Deschide document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Primii pași + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless rulează! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Puteți începe să încărcați fișiere trăgându-le în căsuța din dreapta sau în directorul configurat de consum, și vor începe să apară în lista de documente. După ce ați adăugat metadate documentelor, folosiți mecanismele de filtrare pentru a crea vizualizări personalizate (cum ar fi "Adăugate recent", "Etichetate URGENT"), ce vor apărea în locul acestui mesaj. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless mai oferă următoarele funcții pentru a vă face viata mai ușoară: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + După ce aveți câteva documente și le adăugați metadate, Paperless poate atribui acele metadate în mod automat pentru documente noi. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Puteți configura Paperless să vă citească email-urile și să adauge documentele atașate. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consultați documentația pentru a afla cum să folosiți aceste funcții. Secțiunea de baza conține informații despre cum să folosiți Paperless în general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadate + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Selectează + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Selectați un obiect + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Data invalidă. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Se caută documentul cu asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Da + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nu + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (fără titlu) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Engleză (Americană) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Engleză (Britanică) + + + German + + src/app/services/settings.service.ts + 92 + + Germană + + + Dutch + + src/app/services/settings.service.ts + 93 + + Olandeză + + + French + + src/app/services/settings.service.ts + 94 + + Franceză + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugheză + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugheză (Brazilia) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italiană + + + Romanian + + src/app/services/settings.service.ts + 98 + + Română + + + Russian + + src/app/services/settings.service.ts + 99 + + Rusă + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spaniolă + + + Polish + + src/app/services/settings.service.ts + 101 + + Poloneză + + + Swedish + + src/app/services/settings.service.ts + 102 + + Suedeză + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Documentul există deja. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Fișierul nu a fost găsit. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Scriptul pre-consum nu există. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Eroare la execuția scriptului pre-consum. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Scriptul post-consum nu există. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Eroare la execuția scriptului post-consum. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Fișier nou primit. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Tip de fișier nesuportat. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Se procesează documentul... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Se generează miniatura... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Se extrage data din document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Se salvează documentul... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Terminat. + + + Error + + src/app/services/toast.service.ts + 35 + + Eroare + + + Information + + src/app/services/toast.service.ts + 39 + + Informații + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Corespondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Tipul documentului + + + Created + + src/app/services/rest/document.service.ts + 21 + + Creat + + + Added + + src/app/services/rest/document.service.ts + 22 + + Adăugat + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modificat + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Scor de căutare + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Crează articol nou + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Modifică articol + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Nu s-a putut salva elementul: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automat + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Sunteți sigur că doriți să ștergeți acest element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Documentele asociate nu vor fi șterse. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Șterge + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Eroare la ștergerea elementului: + + + Any word + + src/app/data/matching-model.ts + 12 + + Oricare cuvânt + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Oricare: Documentul conține oricare dintre aceste cuvinte (separate prin spațiu) + + + All words + + src/app/data/matching-model.ts + 13 + + Toate cuvintele + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Toate: Documentul conține toate aceste cuvinte (separate prin spațiu) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Potrivire exacta + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Documentul conține acest șir de caractere + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Expresie regulată + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Expresie regulată: Documentul se potrivește cu această expresie regulată + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Cuvânt neclar + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Neclar: Documentul conține un cuvânt similar cu acesta + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Învață potrivirea automat + + + + diff --git a/src-ui/src/locale/messages.ru_RU.xlf b/src-ui/src/locale/messages.ru_RU.xlf new file mode 100644 index 000000000..4e7a4e318 --- /dev/null +++ b/src-ui/src/locale/messages.ru_RU.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Документ добавлен + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Документ был добавлен в paperless. + + + Open document + + src/app/app.component.ts + 51 + + Открыть документ + + + Could not add : + + src/app/app.component.ts + 59 + + Невозможно добавить : + + + New document detected + + src/app/app.component.ts + 65 + + Обнаружен новый документ + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Документ обрабатывается paperless + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Документы + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Представление "" успешно сохранено. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Представление "" успешно создано. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Выбор + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Ничего + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Выбрать страницу + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Выбрать всё + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Сортировка + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Представления + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Сохранить как... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Сохранить "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Выделено документов: из 1} other {Выделено документов: из }} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {Один документ} other { документа(ов)}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (отфильтровано) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + Архивный номер + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Корреспондент + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Название + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Тип документа + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Создано + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Добавлено + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Подтвердите удаление + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Вы действительно хотите удалить документ ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Файлы из этого документа будут удалены незамедлительно. Это операцию нельзя отменить. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Удалить документ + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Ошибка удаления документа: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Удалить + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Скачать + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Больше похожих + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Закрыть + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Детали + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Содержимое + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Метаданные + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Отменить + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Сохранить + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Страница + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + из + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Скачать оригинал + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Архивный номер (АН) + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Дата создания + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Дата изменения + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Дата добавления + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Имя файла + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Оригинальная MD5 сумма + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Размер оригинального файла + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Оригинальный MIME тип + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + MD5 сумма архива + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Размер архива + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Метаданные оригинального документа + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Метаданные архивного документа + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Сохранить & следующий + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Привет, , добро пожаловать в Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Добро пожаловать в Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Главная + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Вы действительно хотите удалить тег ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Теги + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Создать + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Фильтр по: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Имя + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Цвет + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Совпадения + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Количество документов + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Действия + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Документы + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Редактировать + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Вы действительно хотите удалить этот тип документа: ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Типы документов + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Логи + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Сохраненное представление "" удалено. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Настройки успешно сохранены. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Использовать язык системы + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Использовать формат даты, соответствующий языку + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Ошибка при хранении настроек на сервере: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Настройки + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Основные + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Уведомления + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Представления + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Вид + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Язык интерфейса + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Вы должны перезагрузить страница, после применения нового языка. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Дата + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Формат даты + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Краткий: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Средний: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Длинный: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Элементов на странице + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Редактор документов + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Использовать просмотрщик PDF встроенный в браузер + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Это, обычно, более быстрый способ отображения больших PDF документов, но он может не работать в некоторых браузерах. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Тёмный режим + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Использовать системные настройки + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Включить тёмный режим + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Инвертировать эскизы при включенном тёмном режиме + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Массовое редактирование + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Показывать диалоги подтверждения + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Удаление документов всегда будет требовать подтверждения. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Применить при закрытии + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Обработка документа + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Показывать уведомления, когда новый документ удалён + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Показывать уведомления, когда обработка документа успешна + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Показывать уведомления, когда обработка документа не удалась + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Спрятать уведомления на главной + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Это отключит все сообщения о статусе обработки документов на главной. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Появляется на + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Показать на главной + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Показать в боковой панели + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Нет сохраненных представлений. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Не найдено + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Вы действительно хотите удалить этого корреспондента ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Корреспонденты + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Недавние корреспонденты + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Подтверждение + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Подтвердить + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Отменить + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Создать нового корреспондента + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Редактировать корреспондента + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Алгоритм сопоставления + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Шаблон соответствия + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Без учёта регистра + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Создать новый тег + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Редактировать тег + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Входящий теги + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Входящий тег будет автоматически назначен всем добавленным документам. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Создать новый тип документа + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Редактировать тип документа + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Поиск документов + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Выход + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Управлять + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Администрирование + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Информация + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Документация + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Предложить идею + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Вошёл как + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Открыть документы + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Закрыть всё + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Корреспондент: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Без корреспондента + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Тип: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Без типа документа + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Тег: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Без тегов + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Название: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + Архивный номер: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Заголовок + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Название и содержимое + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + АН + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Расширенный поиск + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Больше похожих + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Фильтр тегов + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Фильтр корреспондентов + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Фильтр типа документов + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Сбросить фильтры + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Не назначено + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Подтвердить + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + 7 дней + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Месяц + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Три месяца + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Год + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + После + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + До + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Очистить + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Представление + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Отфильтровать по корреспонденту + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Отфильтровать по тегу + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Оценка: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Создано: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Добавлено: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Изменено: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Ошибка выполнения массовой операции: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" и "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + и "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Подтвердить применяемый тег + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Эта операция добавит тег "" выбранным документам . + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Эта операция добавит теги выбранным документам. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Эта операция удалит теги "" с выбранных документов. + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Эта операция удалит теги с выбранных документов. + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Эта операция добавит теги и удалит теги на выбранных документах. + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Подтвердите назначения корреспондента + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Эта операция назначит корреспондента "" выбранным документам. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Эта операция удалит корреспондента из выбранных документов. + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Подтвердите назначение типа документа + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Эта операция присвоит тип "" выбранным документам. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Эта операция удалит тип из выбранных документов. + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Подтвердите удаление + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Эта операция навсегда удалит выбранных документов. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Эту операцию нельзя отменить. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Удалить документ(ы) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Выбрано: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Все + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Редактировать: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Скачать оригиналы + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Рекомендации: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Сохранить текущее представление + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Показать всё + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Статистика + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Всего документов: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Документов во входящих: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Обрабатываются: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Ошибка: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Добавлено: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Подключение... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Загрузка... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Загрузка завершена, ждите... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + Ошибка HTTP: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Загрузить новые документы + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Перетащите документы или + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Загрузите файлы + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Скрытие завершено + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {Ещё документ: 1} other {Ещё документов: }} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Открыть документ + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Первые шаги + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless запущен! ;) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Вы можете загрузить документы, перетащив их на панель загрузки файлов справа, либо положив их в каталог, указанный в настройках. После обработки документы появятся в списке документов. После того, как вы добавите метаданные в свои документы, используйте безбумажные механизмы фильтрации для создания пользовательских представлений (таких как «Недавно добавленные», «Tagged TODO»), и они появятся здесь вместо этого сообщения. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless предлагает больше возможностей, чтобы сделать вашу жизнь легче: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + После того, как вы добавите несколько документов в paperless и добавите к ним метаданные, paperless сможет автоматически назначать эти метаданные новым документам. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Вы можете настроить paperless на чтение ваших писем и добавление документов из прикрепленных файлов. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Обратитесь к документации по использованию этих функций. Раздел "Использование" содержит общую информацию о работе с paperless. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Метаданные + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Выбрать + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Пожалуйста, выберите объект + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Неверная дата. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Поиск документа с asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Да + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Нет + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (без названия) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + German + + + Dutch + + src/app/services/settings.service.ts + 93 + + Голландский + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Португальский + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 99 + + Русский + + + Spanish + + src/app/services/settings.service.ts + 100 + + Испанский + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Такой документ уже существует. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Файл не найден. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Скрипт предобработки не существует. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Ошибка при выполнении скрипта предобработки. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Скрипт постобработки не существует. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Ошибка при выполнении скрипта постобработки. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Получен новый файл. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Этот тип файла не поддерживается. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Обработка документа... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Создание эскиза... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Получение даты из документа... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Сохранение документа... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Завершено. + + + Error + + src/app/services/toast.service.ts + 35 + + Ошибка + + + Information + + src/app/services/toast.service.ts + 39 + + Информация + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Корреспондент + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Тип документа + + + Created + + src/app/services/rest/document.service.ts + 21 + + Создано + + + Added + + src/app/services/rest/document.service.ts + 22 + + Добавлено + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Изменено + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Создать новый объект + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Редактировать объект + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Не могу сохранить элемент: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Автоматически + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Вы действительно хотите удалить этот элемент? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Связанные документы не будут удалены. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Удалить + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Возникла ошибка при удалении: + + + Any word + + src/app/data/matching-model.ts + 12 + + Любые слова + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Любой: Документ содержит любое из этих слов (разделитель - пробел) + + + All words + + src/app/data/matching-model.ts + 13 + + Все слова + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + Все: Документ содержит все эти слова (разделитель - пробел) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Точное соответствие + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Точное: Документ содержит эту строку + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Регулярное выражение + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Регулярное выражение: Документ соответствует этому регулярному выражению + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + "Нечёткий" режим + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Нечетко: Документ содержит слово, подобное этому слову + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Авто: Автоматически изучать соответствие + + + + diff --git a/src-ui/src/locale/messages.sv_SE.xlf b/src-ui/src/locale/messages.sv_SE.xlf new file mode 100644 index 000000000..9ca41e6db --- /dev/null +++ b/src-ui/src/locale/messages.sv_SE.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Dokument tillagt + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Dokument lades till i paperless. + + + Open document + + src/app/app.component.ts + 51 + + Öppna dokument + + + Could not add : + + src/app/app.component.ts + 59 + + Kunde inte lägga till : + + + New document detected + + src/app/app.component.ts + 65 + + Nytt dokument upptäckt + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Dokument behandlas av paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + Dokument + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + Vy "" sparades. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + Vy "" skapades. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Välj + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Markera ingen + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Välj sida + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Välj alla + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sortera + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Vyer + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Spara som... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Spara "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, one {} =1 {Vald av ett dokument} other {Vald av dokument}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, one {} =1 {Ett dokument} other { dokument}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtrerad) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Korrespondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Titel + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Dokumenttyp + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Skapad + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Tillagd + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Bekräfta borttagning + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Vill du verkligen ta bort dokumentet ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + Filerna för detta dokument kommer att raderas permanent. Den här åtgärden kan inte ångras. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Ta bort dokument + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Fel vid borttagning av dokument: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Radera + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Ladda ner + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + Mer som detta + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Stäng + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Detaljer + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Innehåll + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Avfärda + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Spara + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Sida + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + av + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Ladda ner original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Arkivets serienummer + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Datum skapad + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Datum ändrad + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Datum tillagd + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filnamn + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5-kontrollsumma + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Ursprunglig filstorlek + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Ursprunglig mime-typ + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Arkiv MD5-kontrollsumma + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Arkiv filstorlek + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Ursprungliga dokument metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Arkiverade dokument metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Spara & nästa + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hej , välkommen till Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Välkommen till Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Kontrollpanel + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Vill du verkligen ta bort taggen ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Taggar + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Skapa + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filtrera på: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Namn + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Färg + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matchar + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Antal dokument + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Åtgärder + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Dokument + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Redigera + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Vill du verkligen ta bort dokumenttypen ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Dokumenttyper + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Loggar + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Sparad vy "" borttagen. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Inställningar har sparats. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Använd systemspråk + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Använd datumformat för visningsspråk + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Fel vid lagring av inställningar på server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Inställningar + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + Allmänna inställningar + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifieringar + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Sparade vyer + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Utseende + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Visningsspråk + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + Du måste ladda om sidan efter att du ställt in ett nytt språk. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Visning av datum + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Datumformat + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Kort: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Lång: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Objekt per sida + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Redigerare för dokument + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Använd PDF-läsaren som tillhandahålls av webbläsaren + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + Detta är oftast snabbare för att visa stora PDF-dokument, men det kanske inte fungerar på vissa webbläsare. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Mörkt läge + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Använd systeminställningar + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Aktivera mörkt läge + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invertera miniatyrer i mörkt läge + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Massändring + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Visa bekräftelsedialoger + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Radering av dokument kommer alltid att be om bekräftelse. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Tillämpa vid stängning + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Dokument bearbetas + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Visa notifieringar när nya dokument upptäcks + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Visa notifieringar när dokumentbehandlingen är klar + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Visa aviseringar när dokumentbehandling misslyckas + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Förhindra aviseringar på instrumentpanelen + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + Detta kommer att förhindra alla meddelanden om status för dokumenthantering på instrumentpanelen. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Visas på + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Visa på instrumentpanelen + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Visa i sidofältet + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + Inga sparade vyer har definierats. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Hittades inte + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Vill du verkligen ta bort korrespondenten ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Korrespondenter + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Senaste korrespondens + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Bekräftelse + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Bekräfta + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Avbryt + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Skapa ny korrespondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Senaste korrespondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matchande algoritm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matchande mönster + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Ej skiftlägeskänsligt + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Skapa ny tagg + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Redigera tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inkorgs tagg + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inkorgtaggar tilldelas automatiskt till alla konsumerade dokument. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Skapa ny dokumenttyp + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Redigera dokumenttyp + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Sök dokument + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logga ut + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Hantera + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Dokumentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Föreslå en idé + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Inloggad som + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Öppna dokument + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Stäng alla + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Korrespondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Utan korrespondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Typ: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Utan dokumenttyp + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Utan tagg + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Titel: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Titel + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Titel & innehåll + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Avancerad sökning + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + Mer som + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filtrera taggar + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filtrera korrespondenter + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filtrera dokument typ + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Återställ filter + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Inte tilldelad + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Tillämpa + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Senaste 7 dagarna + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Senaste månaden + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Senaste 3 månaderna + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Senaste året + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + Efter + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Innan + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Rensa + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + Vy + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filtrera på korrespondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filtrera efter tagg + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Poäng: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Skapad: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Tillagd: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Ändrad: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Fel vid körning av massoperation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" och "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + " och "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Bekräfta tilldelning av taggar + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + Denna åtgärd kommer att lägga till taggen "" till valda dokument. + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + Denna åtgärd kommer att lägga till taggen "" till valda dokument. + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + Denna åtgärd kommer att ta bort taggen "" från valda dokument. + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + Denna åtgärd kommer att ta bort taggar från valda dokument. + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + Den här åtgärden kommer att lägga till taggar och ta bort taggar valda dokument. + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Bekräfta korrespondenttilldelning + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + Den här åtgärden kommer att tilldela korrespondenten "" till valda dokument. + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + Den här åtgärden kommer att ta bort korrespondenten från valda dokument. + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Bekräfta tilldelning av dokumenttyp + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + Den här åtgärden kommer att tilldela dokumenttypen "" till valda dokument. + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + Den här åtgärden kommer att ta bort dokumenttypen från valda dokument. + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Bekräfta borttagning + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + Den här åtgärden kommer att permanent ta bort markerade dokument. + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + Den här åtgärden kan inte ångras. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Ta bort dokument + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Välj: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + Alla + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Redigera: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Ladda ner original + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Lägg till objekt + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Förslag: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Spara aktuell vy + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Lägg till tagg + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Visa alla + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistik + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Totalt antal dokument: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Dokument i inkorg: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Bearbetar: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Misslyckades: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Tillagda: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Ansluter... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Laddar upp... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Uppladdning klar, väntar... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP-fel: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Ladda upp dokument + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Släpp dokument här eller + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Bläddra bland filer + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Avfärda slutförd + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, one {} =1 {Ett dokument till} other { fler dokument}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Öppna dokument + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + Första stegen + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Papperslös är igång! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + Du kan börja ladda upp dokument genom att släppa dem i rutan för filuppladdning till höger eller genom att släppa dem i den konfigurerade konsumtionsmappen och de börjar visas i dokumentlistan. När du har lagt till några metadata i dina dokument, använd filtreringsmekanismerna för paperless för att skapa anpassade vyer (såsom "Nyligen tillagd", 'Tagged TODO') och de kommer att visas på instrumentpanelen i stället för detta meddelande. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless erbjuder några ytterligare funktioner för att göra ditt liv enklare: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + När du har fått ett par dokument i paperless och lagt till metadata till dem, kan paperless tilldela detta metadata till nya dokument automatiskt. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + Du kan konfigurera paperless för att läsa dina e-postmeddelanden och lägga till dokument från bifogade filer. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Konsultera dokumentationen om hur man använder dessa funktioner. Avsnittet om grundläggande användning har också viss information om hur man använder paperless i allmänhet. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Välj + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Välj ett objekt + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Ogiltigt datum. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Söker dokument med asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Ja + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + Nej + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (ingen titel) + + + English (US) + + src/app/services/settings.service.ts + 90 + + Engelska (USA) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + Engelska (GB) + + + German + + src/app/services/settings.service.ts + 92 + + Tyska + + + Dutch + + src/app/services/settings.service.ts + 93 + + Holländska + + + French + + src/app/services/settings.service.ts + 94 + + Franska + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portugisiska + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portugisiska (Brasilien) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italienska + + + Romanian + + src/app/services/settings.service.ts + 98 + + Rumänska + + + Russian + + src/app/services/settings.service.ts + 99 + + Ryska + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanska + + + Polish + + src/app/services/settings.service.ts + 101 + + Polska + + + Swedish + + src/app/services/settings.service.ts + 102 + + Svenska + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Dokumentet finns redan. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + Filen hittades inte. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume skript finns inte. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fel vid körning av pre-consume skript. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume skript finns inte. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Fel vid körning av post-consume skript. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Tog emot ny fil. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + Filtypen stöds inte. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Bearbetar dokument... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Miniatyrer genereras... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Hämtar datum från dokument... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Sparar dokument... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Slutförd. + + + Error + + src/app/services/toast.service.ts + 35 + + Fel + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Korrespondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Dokumenttyp + + + Created + + src/app/services/rest/document.service.ts + 21 + + Skapad + + + Added + + src/app/services/rest/document.service.ts + 22 + + Tillagd + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Ändrad + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Sök resultat + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Skapa nytt objekt + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Redigera objekt + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Kunde inte spara element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatisk + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Vill du verkligen ta bort detta element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Kopplade dokument kommer inte att raderas. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Radera + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Fel vid borttagning av element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Valfritt ord + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Dokumentet innehåller något av dessa ord (mellanslag separerade) + + + All words + + src/app/data/matching-model.ts + 13 + + Alla ord + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Dokumentet innehåller alla dessa ord (mellanslag separerade) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exakt matchning + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Dokumentet innehåller denna sträng + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Reguljära uttryck + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Reguljärt uttryck: Dokumentet matchar detta reguljära uttryck + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy word + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Dokumentet innehåller ett ord som liknar detta ord + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Lär matchning automatiskt + + + + diff --git a/src-ui/src/locale/messages.th_TH.xlf b/src-ui/src/locale/messages.th_TH.xlf new file mode 100644 index 000000000..6fac379bd --- /dev/null +++ b/src-ui/src/locale/messages.th_TH.xlf @@ -0,0 +1,2290 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 49 + + Documents + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 115 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 136 + + View "" created successfully. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Select + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Select none + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Select page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Select all + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Save as... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 105 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 111 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 117 + + Title + + + Document type + + src/app/components/document-list/document-list.component.html + 123 + + Document type + + + Created + + src/app/components/document-list/document-list.component.html + 129 + + Created + + + Added + + src/app/components/document-list/document-list.component.html + 135 + + Added + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 203 + + Confirm delete + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 204 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 205 + + The files for this document will be deleted permanently. This operation cannot be undone. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 207 + + Delete document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 214 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Delete + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + More like this + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Close + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Content + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Discard + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Save + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + of + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archive serial number + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date created + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Date modified + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date added + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filename + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Original file size + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Original mime type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archive MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archive file size + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Original document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archived document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Save & next + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , welcome to Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welcome to Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Do you really want to delete the tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Create + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter by: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Color + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matching + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Document count + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Actions + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + Documents + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edit + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Do you really want to delete the document type ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Document types + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Settings saved successfully. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error while storing settings on server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Settings + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + General settings + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Saved views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Items per page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Bulk editing + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Show confirmation dialogs + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Deleting documents will always ask for confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Apply on close + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appears on + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Show on dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Show in sidebar + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No saved views defined. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Not Found + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Do you really want to delete the correspondent ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondents + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Last correspondence + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirm + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancel + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Create new correspondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edit correspondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matching algorithm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Case insensitive + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Create new tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edit tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tags are automatically assigned to all consumed documents. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Create new document type + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edit document type + + + Search results + + src/app/components/search/search.component.html + 1 + + Search results + + + Invalid search query: + + src/app/components/search/search.component.html + 4 + + Invalid search query: + + + Showing documents similar to + + src/app/components/search/search.component.html + 7 + + Showing documents similar to + + + Search query: + + src/app/components/search/search.component.html + 11 + + Search query: + + + Did you mean ""? + + src/app/components/search/search.component.html + 13 + + Did you mean ""? + + + {VAR_PLURAL, plural, =0 {No results} =1 {One result} other { results}} + + src/app/components/search/search.component.html + 18 + + {VAR_PLURAL, plural, =0 {No results} =1 {One result} other { results}} + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logout + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Manage + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documents + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Close all + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 73 + + Title + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 74 + + Title & content + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 32 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 34 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 41 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 45 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 49 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 53 + + Title: + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 20 + + Filter tags + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 28 + + Filter correspondents + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 35 + + Filter document types + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 58 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Not assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Apply + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Last 7 days + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Last month + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Last 3 months + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Last year + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + After + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Before + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Clear + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 50 + + View + + + Created: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 67 + + Created: + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filter by correspondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filter by tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 62 + + Score: + + + View in browser + + src/app/components/document-list/document-card-small/document-card-small.component.html + 40 + + View in browser + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" and "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + and "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirm tags assignment + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirm correspondent assignment + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirm document type assignment + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Delete confirm + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + This operation cannot be undone. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Delete document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Select: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edit: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 26 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Save current view + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Show all + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistics + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Upload new documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Drop documents here or + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Browse files + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + First steps + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is running! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Select + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Please select an object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Yes + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (no title) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + German + + + Dutch + + src/app/services/settings.service.ts + 93 + + Dutch + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 95 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 96 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 97 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 98 + + Russian + + + ISO 8601 + + src/app/services/settings.service.ts + 103 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + ASN + + src/app/services/rest/document.service.ts + 17 + + ASN + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Document type + + + Created + + src/app/services/rest/document.service.ts + 21 + + Created + + + Added + + src/app/services/rest/document.service.ts + 22 + + Added + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modified + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Create new item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edit item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Could not save element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + Automatic + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Do you really want to delete this element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Associated documents will not be deleted. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Delete + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + Any word + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + All words + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + Exact match + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + Fuzzy word + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/locale/messages.xh_ZA.xlf b/src-ui/src/locale/messages.xh_ZA.xlf new file mode 100644 index 000000000..f3d23366e --- /dev/null +++ b/src-ui/src/locale/messages.xh_ZA.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + crwdns2838:0crwdne2838:0 + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + crwdns2840:0crwdne2840:0 + + + Open document + + src/app/app.component.ts + 51 + + crwdns2842:0crwdne2842:0 + + + Could not add : + + src/app/app.component.ts + 59 + + crwdns2844:0crwdne2844:0 + + + New document detected + + src/app/app.component.ts + 65 + + crwdns2846:0crwdne2846:0 + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + crwdns2848:0crwdne2848:0 + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + crwdns2850:0crwdne2850:0 + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + crwdns2852:0crwdne2852:0 + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + crwdns2854:0crwdne2854:0 + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + crwdns2856:0crwdne2856:0 + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + crwdns2858:0crwdne2858:0 + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + crwdns2860:0crwdne2860:0 + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + crwdns2862:0crwdne2862:0 + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + crwdns2864:0crwdne2864:0 + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + crwdns2866:0crwdne2866:0 + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + crwdns2868:0crwdne2868:0 + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + crwdns2870:0{{list.activeSavedViewTitle}}crwdne2870:0 + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + crwdns2872:0VAR_PLURAL={VAR_PLURAL}crwdne2872:0 + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + crwdns2874:0VAR_PLURAL={VAR_PLURAL}crwdne2874:0 + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + crwdns2876:0crwdne2876:0 + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + crwdns2878:0crwdne2878:0 + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + crwdns2880:0crwdne2880:0 + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + crwdns2882:0crwdne2882:0 + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + crwdns2884:0crwdne2884:0 + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + crwdns2886:0crwdne2886:0 + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + crwdns2888:0crwdne2888:0 + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + crwdns2890:0crwdne2890:0 + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + crwdns2892:0crwdne2892:0 + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + crwdns2894:0crwdne2894:0 + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + crwdns2896:0crwdne2896:0 + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + crwdns2898:0crwdne2898:0 + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + crwdns2900:0crwdne2900:0 + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + crwdns2902:0crwdne2902:0 + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + crwdns2904:0crwdne2904:0 + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + crwdns2906:0crwdne2906:0 + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + crwdns2908:0crwdne2908:0 + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + crwdns2910:0crwdne2910:0 + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + crwdns2912:0crwdne2912:0 + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + crwdns2914:0crwdne2914:0 + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + crwdns2916:0crwdne2916:0 + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + crwdns2918:0crwdne2918:0 + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + crwdns2920:0{{previewNumPages}}crwdne2920:0 + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + crwdns2922:0crwdne2922:0 + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + crwdns2924:0crwdne2924:0 + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + crwdns2926:0crwdne2926:0 + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + crwdns2928:0crwdne2928:0 + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + crwdns2930:0crwdne2930:0 + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + crwdns2932:0crwdne2932:0 + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + crwdns2934:0crwdne2934:0 + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + crwdns2936:0crwdne2936:0 + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + crwdns2938:0crwdne2938:0 + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + crwdns2940:0crwdne2940:0 + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + crwdns2942:0crwdne2942:0 + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + crwdns2944:0crwdne2944:0 + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + crwdns2946:0crwdne2946:0 + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + crwdns2948:0crwdne2948:0 + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + crwdns2950:0crwdne2950:0 + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + crwdns2952:0crwdne2952:0 + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + crwdns2954:0crwdne2954:0 + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + crwdns2956:0crwdne2956:0 + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + crwdns2958:0crwdne2958:0 + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + crwdns2960:0crwdne2960:0 + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + crwdns2962:0crwdne2962:0 + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + crwdns2964:0crwdne2964:0 + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + crwdns2966:0crwdne2966:0 + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + crwdns2968:0crwdne2968:0 + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + crwdns2970:0crwdne2970:0 + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + crwdns2972:0crwdne2972:0 + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + crwdns2974:0crwdne2974:0 + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + crwdns2976:0crwdne2976:0 + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + crwdns2978:0crwdne2978:0 + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + crwdns2980:0crwdne2980:0 + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + crwdns2982:0crwdne2982:0 + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + crwdns2984:0crwdne2984:0 + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + crwdns2986:0crwdne2986:0 + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + crwdns2988:0crwdne2988:0 + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + crwdns2990:0crwdne2990:0 + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + crwdns2992:0crwdne2992:0 + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + crwdns2994:0crwdne2994:0 + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + crwdns2996:0crwdne2996:0 + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + crwdns2998:0crwdne2998:0 + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + crwdns3000:0crwdne3000:0 + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + crwdns3002:0crwdne3002:0 + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + crwdns3004:0crwdne3004:0 + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + crwdns3006:0crwdne3006:0 + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + crwdns3008:0crwdne3008:0 + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + crwdns3010:0crwdne3010:0 + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + crwdns3012:0{today | customDate:'shortDate':null:computedDateLocale}crwdne3012:0 + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + crwdns3014:0{today | customDate:'mediumDate':null:computedDateLocale}crwdne3014:0 + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + crwdns3016:0{today | customDate:'longDate':null:computedDateLocale}crwdne3016:0 + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + crwdns3018:0crwdne3018:0 + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + crwdns3020:0crwdne3020:0 + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + crwdns3022:0crwdne3022:0 + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + crwdns3024:0crwdne3024:0 + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + crwdns3026:0crwdne3026:0 + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + crwdns3028:0crwdne3028:0 + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + crwdns3030:0crwdne3030:0 + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + crwdns3032:0crwdne3032:0 + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + crwdns3034:0crwdne3034:0 + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + crwdns3036:0crwdne3036:0 + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + crwdns3038:0crwdne3038:0 + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + crwdns3040:0crwdne3040:0 + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + crwdns3042:0crwdne3042:0 + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + crwdns3044:0crwdne3044:0 + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + crwdns3046:0crwdne3046:0 + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + crwdns3048:0crwdne3048:0 + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + crwdns3050:0crwdne3050:0 + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + crwdns3052:0crwdne3052:0 + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + crwdns3054:0crwdne3054:0 + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + crwdns3056:0crwdne3056:0 + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + crwdns3058:0crwdne3058:0 + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + crwdns3060:0crwdne3060:0 + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + crwdns3062:0crwdne3062:0 + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + crwdns3064:0crwdne3064:0 + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + crwdns3066:0crwdne3066:0 + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + crwdns3068:0crwdne3068:0 + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + crwdns3070:0crwdne3070:0 + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + crwdns3072:0crwdne3072:0 + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + crwdns3074:0crwdne3074:0 + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + crwdns3076:0crwdne3076:0 + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + crwdns3078:0crwdne3078:0 + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + crwdns3080:0crwdne3080:0 + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + crwdns3082:0crwdne3082:0 + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + crwdns3084:0crwdne3084:0 + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + crwdns3086:0crwdne3086:0 + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + crwdns3088:0crwdne3088:0 + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + crwdns3090:0crwdne3090:0 + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + crwdns3092:0crwdne3092:0 + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + crwdns3094:0crwdne3094:0 + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + crwdns3096:0crwdne3096:0 + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + crwdns3110:0crwdne3110:0 + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + crwdns3112:0crwdne3112:0 + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + crwdns3114:0crwdne3114:0 + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + crwdns3116:0crwdne3116:0 + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + crwdns3118:0crwdne3118:0 + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + crwdns3120:0crwdne3120:0 + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + crwdns3122:0crwdne3122:0 + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + crwdns3124:0crwdne3124:0 + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + crwdns3126:0crwdne3126:0 + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + crwdns3128:0{{displayName}}crwdne3128:0 + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + crwdns3130:0crwdne3130:0 + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + crwdns3132:0crwdne3132:0 + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + crwdns3138:0crwdne3138:0 + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + crwdns3140:0crwdne3140:0 + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + crwdns3142:0crwdne3142:0 + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + crwdns3144:0crwdne3144:0 + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + crwdns3146:0crwdne3146:0 + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + crwdns3148:0crwdne3148:0 + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + crwdns3150:0crwdne3150:0 + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + crwdns3416:0crwdne3416:0 + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + crwdns3134:0crwdne3134:0 + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + crwdns3136:0crwdne3136:0 + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + crwdns3354:0crwdne3354:0 + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + crwdns3434:0crwdne3434:0 + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + crwdns3436:0crwdne3436:0 + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + crwdns3152:0crwdne3152:0 + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + crwdns3154:0crwdne3154:0 + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + crwdns3156:0crwdne3156:0 + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + crwdns3158:0crwdne3158:0 + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + crwdns3160:0crwdne3160:0 + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + crwdns3162:0crwdne3162:0 + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + crwdns3164:0crwdne3164:0 + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + crwdns3166:0crwdne3166:0 + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + crwdns3168:0crwdne3168:0 + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + crwdns3170:0crwdne3170:0 + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + crwdns3172:0crwdne3172:0 + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + crwdns3174:0crwdne3174:0 + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + crwdns3176:0crwdne3176:0 + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + crwdns3178:0crwdne3178:0 + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + crwdns3182:0crwdne3182:0 + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + crwdns3184:0crwdne3184:0 + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + crwdns3186:0crwdne3186:0 + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + crwdns3426:0{{ document.created | customDate}}crwdne3426:0 + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + crwdns3428:0{{ document.added | customDate}}crwdne3428:0 + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + crwdns3430:0{{ document.modified | customDate}}crwdne3430:0 + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + crwdns3190:0crwdne3190:0 + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + crwdns3192:0[0]crwdne3192:0 + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + crwdns3194:0[0]crwdnd3194:0[1]crwdne3194:0 + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + crwdns3196:0crwdne3196:0 + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + crwdns3198:0crwdne3198:0 + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + crwdns3200:0crwdne3200:0 + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + crwdns3202:0crwdne3202:0 + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + crwdns3204:0crwdne3204:0 + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + crwdns3206:0crwdne3206:0 + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + crwdns3208:0crwdne3208:0 + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + crwdns3210:0crwdne3210:0 + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + crwdns3212:0crwdne3212:0 + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + crwdns3214:0crwdne3214:0 + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + crwdns3216:0crwdne3216:0 + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + crwdns3218:0crwdne3218:0 + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + crwdns3220:0crwdne3220:0 + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + crwdns3222:0crwdne3222:0 + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + crwdns3224:0crwdne3224:0 + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + crwdns3226:0crwdne3226:0 + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + crwdns3228:0crwdne3228:0 + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + crwdns3230:0crwdne3230:0 + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + crwdns3232:0crwdne3232:0 + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + crwdns3234:0crwdne3234:0 + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + crwdns3236:0crwdne3236:0 + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + crwdns3238:0crwdne3238:0 + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + crwdns3450:0crwdne3450:0 + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + crwdns3240:0crwdne3240:0 + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + crwdns3242:0crwdne3242:0 + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + crwdns3452:0crwdne3452:0 + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + crwdns3244:0crwdne3244:0 + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + crwdns3246:0crwdne3246:0 + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + crwdns3248:0{{statistics?.documents_total}}crwdne3248:0 + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + crwdns3250:0{{statistics?.documents_inbox}}crwdne3250:0 + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + crwdns3252:0crwdne3252:0 + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + crwdns3254:0crwdne3254:0 + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + crwdns3256:0crwdne3256:0 + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + crwdns3258:0crwdne3258:0 + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + crwdns3260:0crwdne3260:0 + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + crwdns3262:0crwdne3262:0 + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + crwdns3264:0crwdne3264:0 + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + crwdns3266:0crwdne3266:0 + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + crwdns3268:0crwdne3268:0 + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + crwdns3270:0crwdne3270:0 + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + crwdns3272:0crwdne3272:0 + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + crwdns3274:0VAR_PLURAL={VAR_PLURAL}crwdne3274:0 + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + crwdns3276:0crwdne3276:0 + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + crwdns3278:0crwdne3278:0 + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + crwdns3280:0crwdne3280:0 + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + crwdns3282:0crwdne3282:0 + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + crwdns3284:0crwdne3284:0 + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + crwdns3286:0crwdne3286:0 + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + crwdns3288:0crwdne3288:0 + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + crwdns3290:0crwdne3290:0 + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + crwdns3292:0crwdne3292:0 + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + crwdns3294:0crwdne3294:0 + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + crwdns3296:0crwdne3296:0 + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + crwdns3298:0crwdne3298:0 + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + crwdns3432:0{{asn}}crwdne3432:0 + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + crwdns3300:0crwdne3300:0 + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + crwdns3302:0crwdne3302:0 + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + crwdns3304:0crwdne3304:0 + + + English (US) + + src/app/services/settings.service.ts + 90 + + crwdns3306:0crwdne3306:0 + + + English (GB) + + src/app/services/settings.service.ts + 91 + + crwdns3308:0crwdne3308:0 + + + German + + src/app/services/settings.service.ts + 92 + + crwdns3310:0crwdne3310:0 + + + Dutch + + src/app/services/settings.service.ts + 93 + + crwdns3312:0crwdne3312:0 + + + French + + src/app/services/settings.service.ts + 94 + + crwdns3314:0crwdne3314:0 + + + Portuguese + + src/app/services/settings.service.ts + 95 + + crwdns3422:0crwdne3422:0 + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + crwdns3316:0crwdne3316:0 + + + Italian + + src/app/services/settings.service.ts + 97 + + crwdns3318:0crwdne3318:0 + + + Romanian + + src/app/services/settings.service.ts + 98 + + crwdns3320:0crwdne3320:0 + + + Russian + + src/app/services/settings.service.ts + 99 + + crwdns3404:0crwdne3404:0 + + + Spanish + + src/app/services/settings.service.ts + 100 + + crwdns3418:0crwdne3418:0 + + + Polish + + src/app/services/settings.service.ts + 101 + + crwdns3442:0crwdne3442:0 + + + Swedish + + src/app/services/settings.service.ts + 102 + + crwdns3446:0crwdne3446:0 + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + crwdns3322:0crwdne3322:0 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + crwdns3324:0crwdne3324:0 + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + crwdns3326:0crwdne3326:0 + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + crwdns3328:0crwdne3328:0 + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + crwdns3330:0crwdne3330:0 + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + crwdns3332:0crwdne3332:0 + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + crwdns3334:0crwdne3334:0 + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + crwdns3336:0crwdne3336:0 + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + crwdns3338:0crwdne3338:0 + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + crwdns3340:0crwdne3340:0 + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + crwdns3342:0crwdne3342:0 + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + crwdns3344:0crwdne3344:0 + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + crwdns3346:0crwdne3346:0 + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + crwdns3348:0crwdne3348:0 + + + Error + + src/app/services/toast.service.ts + 35 + + crwdns3350:0crwdne3350:0 + + + Information + + src/app/services/toast.service.ts + 39 + + crwdns3352:0crwdne3352:0 + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + crwdns3356:0crwdne3356:0 + + + Document type + + src/app/services/rest/document.service.ts + 20 + + crwdns3358:0crwdne3358:0 + + + Created + + src/app/services/rest/document.service.ts + 21 + + crwdns3360:0crwdne3360:0 + + + Added + + src/app/services/rest/document.service.ts + 22 + + crwdns3362:0crwdne3362:0 + + + Modified + + src/app/services/rest/document.service.ts + 23 + + crwdns3364:0crwdne3364:0 + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + crwdns3454:0crwdne3454:0 + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + crwdns3366:0crwdne3366:0 + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + crwdns3368:0crwdne3368:0 + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + crwdns3370:0crwdne3370:0 + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + crwdns3372:0crwdne3372:0 + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + crwdns3374:0crwdne3374:0 + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + crwdns3376:0crwdne3376:0 + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + crwdns3378:0crwdne3378:0 + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + crwdns3380:0crwdne3380:0 + + + Any word + + src/app/data/matching-model.ts + 12 + + crwdns3382:0crwdne3382:0 + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + crwdns3384:0crwdne3384:0 + + + All words + + src/app/data/matching-model.ts + 13 + + crwdns3386:0crwdne3386:0 + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + crwdns3388:0crwdne3388:0 + + + Exact match + + src/app/data/matching-model.ts + 14 + + crwdns3390:0crwdne3390:0 + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + crwdns3392:0crwdne3392:0 + + + Regular expression + + src/app/data/matching-model.ts + 15 + + crwdns3394:0crwdne3394:0 + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + crwdns3396:0crwdne3396:0 + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + crwdns3398:0crwdne3398:0 + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + crwdns3400:0crwdne3400:0 + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + crwdns3402:0crwdne3402:0 + + + + diff --git a/src-ui/src/locale/messages.zh_CN.xlf b/src-ui/src/locale/messages.zh_CN.xlf new file mode 100644 index 000000000..951c1e112 --- /dev/null +++ b/src-ui/src/locale/messages.zh_CN.xlf @@ -0,0 +1,2340 @@ + + + + + + Document added + + src/app/app.component.ts + 51 + + Document added + + + Document was added to paperless. + + src/app/app.component.ts + 51 + + Document was added to paperless. + + + Open document + + src/app/app.component.ts + 51 + + Open document + + + Could not add : + + src/app/app.component.ts + 59 + + Could not add : + + + New document detected + + src/app/app.component.ts + 65 + + New document detected + + + Document is being processed by paperless. + + src/app/app.component.ts + 65 + + Document is being processed by paperless. + + + Documents + + src/app/components/document-list/document-list.component.ts + 51 + + 文件 + + + View "" saved successfully. + + src/app/components/document-list/document-list.component.ts + 116 + + View "" saved successfully. + + + View "" created successfully. + + src/app/components/document-list/document-list.component.ts + 138 + + View "" created successfully. + + + Select + + src/app/components/document-list/document-list.component.html + 7 + + Select + + + Select none + + src/app/components/document-list/document-list.component.html + 10 + + Select none + + + Select page + + src/app/components/document-list/document-list.component.html + 11 + + Select page + + + Select all + + src/app/components/document-list/document-list.component.html + 12 + + Select all + + + Sort + + src/app/components/document-list/document-list.component.html + 39 + + Sort + + + Views + + src/app/components/document-list/document-list.component.html + 64 + + Views + + + Save as... + + src/app/components/document-list/document-list.component.html + 72 + + Save as... + + + Save "" + + src/app/components/document-list/document-list.component.html + 71 + + Save "" + + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + src/app/components/document-list/document-list.component.html + 85 + + {VAR_PLURAL, plural, =1 {Selected of one document} other {Selected of documents}} + + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + src/app/components/document-list/document-list.component.html + 86 + + {VAR_PLURAL, plural, =1 {One document} other { documents}} + + + (filtered) + + src/app/components/document-list/document-list.component.html + 86 + + (filtered) + + + ASN + + src/app/components/document-list/document-list.component.html + 111 + + ASN + + + Correspondent + + src/app/components/document-list/document-list.component.html + 117 + + Correspondent + + + Title + + src/app/components/document-list/document-list.component.html + 123 + + Title + + + Document type + + src/app/components/document-list/document-list.component.html + 129 + + Document type + + + Created + + src/app/components/document-list/document-list.component.html + 135 + + Created + + + Added + + src/app/components/document-list/document-list.component.html + 141 + + Added + + + Confirm delete + + src/app/components/document-detail/document-detail.component.ts + 206 + + Confirm delete + + + Do you really want to delete document ""? + + src/app/components/document-detail/document-detail.component.ts + 207 + + Do you really want to delete document ""? + + + The files for this document will be deleted permanently. This operation cannot be undone. + + src/app/components/document-detail/document-detail.component.ts + 208 + + The files for this document will be deleted permanently. This operation cannot be undone. + + + Delete document + + src/app/components/document-detail/document-detail.component.ts + 210 + + Delete document + + + Error deleting document: + + src/app/components/document-detail/document-detail.component.ts + 217 + + Error deleting document: + + + Delete + + src/app/components/document-detail/document-detail.component.html + 15 + + Delete + + + Download + + src/app/components/document-detail/document-detail.component.html + 23 + + Download + + + More like this + + src/app/components/document-detail/document-detail.component.html + 38 + + More like this + + + Close + + src/app/components/document-detail/document-detail.component.html + 44 + + Close + + + Details + + src/app/components/document-detail/document-detail.component.html + 56 + + Details + + + Content + + src/app/components/document-detail/document-detail.component.html + 72 + + Content + + + Metadata + + src/app/components/document-detail/document-detail.component.html + 81 + + Metadata + + + Discard + + src/app/components/document-detail/document-detail.component.html + 130 + + Discard + + + Save + + src/app/components/document-detail/document-detail.component.html + 132 + + Save + + + Page + + src/app/components/document-detail/document-detail.component.html + 4 + + Page + + + of + + src/app/components/document-detail/document-detail.component.html + 8 + + of + + + Download original + + src/app/components/document-detail/document-detail.component.html + 29 + + Download original + + + Archive serial number + + src/app/components/document-detail/document-detail.component.html + 60 + + Archive serial number + + + Date created + + src/app/components/document-detail/document-detail.component.html + 61 + + Date created + + + Date modified + + src/app/components/document-detail/document-detail.component.html + 87 + + Date modified + + + Date added + + src/app/components/document-detail/document-detail.component.html + 91 + + Date added + + + Media filename + + src/app/components/document-detail/document-detail.component.html + 95 + + Media filename + + + Original MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 99 + + Original MD5 checksum + + + Original file size + + src/app/components/document-detail/document-detail.component.html + 103 + + Original file size + + + Original mime type + + src/app/components/document-detail/document-detail.component.html + 107 + + Original mime type + + + Archive MD5 checksum + + src/app/components/document-detail/document-detail.component.html + 111 + + Archive MD5 checksum + + + Archive file size + + src/app/components/document-detail/document-detail.component.html + 115 + + Archive file size + + + Original document metadata + + src/app/components/document-detail/document-detail.component.html + 121 + + Original document metadata + + + Archived document metadata + + src/app/components/document-detail/document-detail.component.html + 122 + + Archived document metadata + + + Save & next + + src/app/components/document-detail/document-detail.component.html + 131 + + Save & next + + + Hello , welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 33 + + Hello , welcome to Paperless-ng! + + + Welcome to Paperless-ng! + + src/app/components/dashboard/dashboard.component.ts + 35 + + Welcome to Paperless-ng! + + + Dashboard + + src/app/components/dashboard/dashboard.component.html + 1 + + Dashboard + + + Do you really want to delete the tag ""? + + src/app/components/manage/tag-list/tag-list.component.ts + 26 + + Do you really want to delete the tag ""? + + + Tags + + src/app/components/manage/tag-list/tag-list.component.html + 1 + + Tags + + + Create + + src/app/components/manage/tag-list/tag-list.component.html + 2 + + Create + + + Filter by: + + src/app/components/manage/tag-list/tag-list.component.html + 8 + + Filter by: + + + Name + + src/app/components/manage/tag-list/tag-list.component.html + 9 + + Name + + + Color + + src/app/components/manage/tag-list/tag-list.component.html + 20 + + Color + + + Matching + + src/app/components/manage/tag-list/tag-list.component.html + 21 + + Matching + + + Document count + + src/app/components/manage/tag-list/tag-list.component.html + 22 + + Document count + + + Actions + + src/app/components/manage/tag-list/tag-list.component.html + 23 + + Actions + + + Documents + + src/app/components/manage/tag-list/tag-list.component.html + 38 + + 文件 + + + Edit + + src/app/components/manage/tag-list/tag-list.component.html + 43 + + Edit + + + Do you really want to delete the document type ""? + + src/app/components/manage/document-type-list/document-type-list.component.ts + 26 + + Do you really want to delete the document type ""? + + + Document types + + src/app/components/manage/document-type-list/document-type-list.component.html + 1 + + Document types + + + Logs + + src/app/components/manage/logs/logs.component.html + 1 + + Logs + + + Saved view "" deleted. + + src/app/components/manage/settings/settings.component.ts + 68 + + Saved view "" deleted. + + + Settings saved successfully. + + src/app/components/manage/settings/settings.component.ts + 89 + + Settings saved successfully. + + + Use system language + + src/app/components/manage/settings/settings.component.ts + 94 + + Use system language + + + Use date format of display language + + src/app/components/manage/settings/settings.component.ts + 100 + + Use date format of display language + + + Error while storing settings on server: + + src/app/components/manage/settings/settings.component.ts + 117 + + Error while storing settings on server: + + + Settings + + src/app/components/manage/settings/settings.component.html + 1 + + Settings + + + General settings + + src/app/components/manage/settings/settings.component.html + 10 + + General settings + + + Notifications + + src/app/components/manage/settings/settings.component.html + 116 + + Notifications + + + Saved views + + src/app/components/manage/settings/settings.component.html + 134 + + Saved views + + + Appearance + + src/app/components/manage/settings/settings.component.html + 13 + + Appearance + + + Display language + + src/app/components/manage/settings/settings.component.html + 17 + + Display language + + + You need to reload the page after applying a new language. + + src/app/components/manage/settings/settings.component.html + 25 + + You need to reload the page after applying a new language. + + + Date display + + src/app/components/manage/settings/settings.component.html + 32 + + Date display + + + Date format + + src/app/components/manage/settings/settings.component.html + 45 + + Date format + + + Short: + + src/app/components/manage/settings/settings.component.html + 51 + + Short: + + + Medium: + + src/app/components/manage/settings/settings.component.html + 55 + + Medium: + + + Long: + + src/app/components/manage/settings/settings.component.html + 59 + + Long: + + + Items per page + + src/app/components/manage/settings/settings.component.html + 67 + + Items per page + + + Document editor + + src/app/components/manage/settings/settings.component.html + 83 + + Document editor + + + Use PDF viewer provided by the browser + + src/app/components/manage/settings/settings.component.html + 87 + + Use PDF viewer provided by the browser + + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + src/app/components/manage/settings/settings.component.html + 87 + + This is usually faster for displaying large PDF documents, but it might not work on some browsers. + + + Dark mode + + src/app/components/manage/settings/settings.component.html + 94 + + Dark mode + + + Use system settings + + src/app/components/manage/settings/settings.component.html + 97 + + Use system settings + + + Enable dark mode + + src/app/components/manage/settings/settings.component.html + 98 + + Enable dark mode + + + Invert thumbnails in dark mode + + src/app/components/manage/settings/settings.component.html + 99 + + Invert thumbnails in dark mode + + + Bulk editing + + src/app/components/manage/settings/settings.component.html + 103 + + Bulk editing + + + Show confirmation dialogs + + src/app/components/manage/settings/settings.component.html + 107 + + Show confirmation dialogs + + + Deleting documents will always ask for confirmation. + + src/app/components/manage/settings/settings.component.html + 107 + + Deleting documents will always ask for confirmation. + + + Apply on close + + src/app/components/manage/settings/settings.component.html + 108 + + Apply on close + + + Document processing + + src/app/components/manage/settings/settings.component.html + 119 + + Document processing + + + Show notifications when new documents are detected + + src/app/components/manage/settings/settings.component.html + 123 + + Show notifications when new documents are detected + + + Show notifications when document processing completes successfully + + src/app/components/manage/settings/settings.component.html + 124 + + Show notifications when document processing completes successfully + + + Show notifications when document processing fails + + src/app/components/manage/settings/settings.component.html + 125 + + Show notifications when document processing fails + + + Suppress notifications on dashboard + + src/app/components/manage/settings/settings.component.html + 126 + + Suppress notifications on dashboard + + + This will suppress all messages about document processing status on the dashboard. + + src/app/components/manage/settings/settings.component.html + 126 + + This will suppress all messages about document processing status on the dashboard. + + + Appears on + + src/app/components/manage/settings/settings.component.html + 146 + + Appears on + + + Show on dashboard + + src/app/components/manage/settings/settings.component.html + 149 + + Show on dashboard + + + Show in sidebar + + src/app/components/manage/settings/settings.component.html + 153 + + Show in sidebar + + + No saved views defined. + + src/app/components/manage/settings/settings.component.html + 163 + + No saved views defined. + + + 404 Not Found + + src/app/components/not-found/not-found.component.html + 7 + + 404 Not Found + + + Do you really want to delete the correspondent ""? + + src/app/components/manage/correspondent-list/correspondent-list.component.ts + 26 + + Do you really want to delete the correspondent ""? + + + Correspondents + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 1 + + Correspondents + + + Last correspondence + + src/app/components/manage/correspondent-list/correspondent-list.component.html + 22 + + Last correspondence + + + Confirmation + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 17 + + Confirmation + + + Confirm + + src/app/components/common/confirm-dialog/confirm-dialog.component.ts + 29 + + Confirm + + + Cancel + + src/app/components/common/confirm-dialog/confirm-dialog.component.html + 12 + + Cancel + + + Create new correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 21 + + Create new correspondent + + + Edit correspondent + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.ts + 25 + + Edit correspondent + + + Matching algorithm + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 10 + + Matching algorithm + + + Matching pattern + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 11 + + Matching pattern + + + Case insensitive + + src/app/components/manage/correspondent-list/correspondent-edit-dialog/correspondent-edit-dialog.component.html + 12 + + Case insensitive + + + Create new tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 22 + + Create new tag + + + Edit tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.ts + 26 + + Edit tag + + + Inbox tag + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tag + + + Inbox tags are automatically assigned to all consumed documents. + + src/app/components/manage/tag-list/tag-edit-dialog/tag-edit-dialog.component.html + 13 + + Inbox tags are automatically assigned to all consumed documents. + + + Create new document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 21 + + Create new document type + + + Edit document type + + src/app/components/manage/document-type-list/document-type-edit-dialog/document-type-edit-dialog.component.ts + 25 + + Edit document type + + + Paperless-ng + + src/app/components/app-frame/app-frame.component.html + 11 + + app title + Paperless-ng + + + Search documents + + src/app/components/app-frame/app-frame.component.html + 15 + + Search documents + + + Logout + + src/app/components/app-frame/app-frame.component.html + 45 + + Logout + + + Manage + + src/app/components/app-frame/app-frame.component.html + 112 + + Manage + + + Admin + + src/app/components/app-frame/app-frame.component.html + 154 + + Admin + + + Info + + src/app/components/app-frame/app-frame.component.html + 160 + + Info + + + Documentation + + src/app/components/app-frame/app-frame.component.html + 167 + + Documentation + + + GitHub + + src/app/components/app-frame/app-frame.component.html + 175 + + GitHub + + + Suggest an idea + + src/app/components/app-frame/app-frame.component.html + 181 + + Suggest an idea + + + Logged in as + + src/app/components/app-frame/app-frame.component.html + 34 + + Logged in as + + + Open documents + + src/app/components/app-frame/app-frame.component.html + 87 + + Open documents + + + Close all + + src/app/components/app-frame/app-frame.component.html + 106 + + Close all + + + Correspondent: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 37 + + Correspondent: + + + Without correspondent + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 39 + + Without correspondent + + + Type: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 44 + + Type: + + + Without document type + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 46 + + Without document type + + + Tag: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 50 + + Tag: + + + Without any tag + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 54 + + Without any tag + + + Title: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 58 + + Title: + + + ASN: + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 61 + + ASN: + + + Title + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 88 + + Title + + + Title & content + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 89 + + Title & content + + + ASN + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 90 + + ASN + + + Advanced search + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 91 + + Advanced search + + + More like + + src/app/components/document-list/filter-editor/filter-editor.component.ts + 94 + + More like + + + Filter tags + + src/app/components/document-list/filter-editor/filter-editor.component.html + 19 + + Filter tags + + + Filter correspondents + + src/app/components/document-list/filter-editor/filter-editor.component.html + 27 + + Filter correspondents + + + Filter document types + + src/app/components/document-list/filter-editor/filter-editor.component.html + 34 + + Filter document types + + + Reset filters + + src/app/components/document-list/filter-editor/filter-editor.component.html + 57 + + Reset filters + + + Not assigned + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.ts + 166 + + Filter drop down element to filter for documents with no correspondent/type/tag assigned + Not assigned + + + Apply + + src/app/components/common/filterable-dropdown/filterable-dropdown.component.html + 26 + + Apply + + + Last 7 days + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 34 + + Last 7 days + + + Last month + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 35 + + Last month + + + Last 3 months + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 36 + + Last 3 months + + + Last year + + src/app/components/common/date-dropdown/date-dropdown.component.ts + 37 + + Last year + + + After + + src/app/components/common/date-dropdown/date-dropdown.component.html + 13 + + After + + + Before + + src/app/components/common/date-dropdown/date-dropdown.component.html + 38 + + Before + + + Clear + + src/app/components/common/date-dropdown/date-dropdown.component.html + 18 + + Clear + + + View + + src/app/components/document-list/document-card-large/document-card-large.component.html + 51 + + View + + + Filter by correspondent + + src/app/components/document-list/document-card-large/document-card-large.component.html + 20 + + Filter by correspondent + + + Filter by tag + + src/app/components/document-list/document-card-large/document-card-large.component.html + 24 + + Filter by tag + + + Score: + + src/app/components/document-list/document-card-large/document-card-large.component.html + 87 + + Score: + + + Created: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 43 + + Created: + + + Added: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 44 + + Added: + + + Modified: + + src/app/components/document-list/document-card-small/document-card-small.component.html + 45 + + Modified: + + + Error executing bulk operation: + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 74 + + Error executing bulk operation: + + + "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 113 + + "" + + + "" and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 115 + + This is for messages like 'modify "tag1" and "tag2"' + "" and "" + + + , + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 117 + + this is used to separate enumerations and should probably be a comma and a whitespace in most languages + , + + + and "" + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 118 + + this is for messages like 'modify "tag1", "tag2" and "tag3"' + and "" + + + Confirm tags assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 127 + + Confirm tags assignment + + + This operation will add the tag "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 130 + + This operation will add the tag "" to selected document(s). + + + This operation will add the tags to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 132 + + This operation will add the tags to selected document(s). + + + This operation will remove the tag "" from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 135 + + This operation will remove the tag "" from selected document(s). + + + This operation will remove the tags from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 137 + + This operation will remove the tags from selected document(s). + + + This operation will add the tags and remove the tags on selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 139 + + This operation will add the tags and remove the tags on selected document(s). + + + Confirm correspondent assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 159 + + Confirm correspondent assignment + + + This operation will assign the correspondent "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 161 + + This operation will assign the correspondent "" to selected document(s). + + + This operation will remove the correspondent from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 163 + + This operation will remove the correspondent from selected document(s). + + + Confirm document type assignment + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 182 + + Confirm document type assignment + + + This operation will assign the document type "" to selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 184 + + This operation will assign the document type "" to selected document(s). + + + This operation will remove the document type from selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 186 + + This operation will remove the document type from selected document(s). + + + Delete confirm + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 201 + + Delete confirm + + + This operation will permanently delete selected document(s). + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 202 + + This operation will permanently delete selected document(s). + + + This operation cannot be undone. + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 203 + + This operation cannot be undone. + + + Delete document(s) + + src/app/components/document-list/bulk-editor/bulk-editor.component.ts + 205 + + Delete document(s) + + + Select: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 10 + + Select: + + + All + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 20 + + All + + + Edit: + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 27 + + Edit: + + + Download originals + + src/app/components/document-list/bulk-editor/bulk-editor.component.html + 68 + + Download originals + + + Add item + + src/app/components/common/input/select/select.component.html + 11 + + Used for both types and correspondents + Add item + + + Suggestions: + + src/app/components/common/input/select/select.component.html + 31 + + Suggestions: + + + Save current view + + src/app/components/document-list/save-view-config-dialog/save-view-config-dialog.component.html + 3 + + Save current view + + + Add tag + + src/app/components/common/input/tags/tags.component.html + 11 + + Add tag + + + Show all + + src/app/components/dashboard/widgets/saved-view-widget/saved-view-widget.component.html + 3 + + Show all + + + Statistics + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 1 + + Statistics + + + Total documents: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 4 + + Total documents: + + + Documents in inbox: + + src/app/components/dashboard/widgets/statistics-widget/statistics-widget.component.html + 3 + + Documents in inbox: + + + Processing: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 32 + + Processing: + + + Failed: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 35 + + Failed: + + + Added: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 38 + + Added: + + + Connecting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 118 + + Connecting... + + + Uploading... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 123 + + Uploading... + + + Upload complete, waiting... + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 126 + + Upload complete, waiting... + + + HTTP error: + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.ts + 136 + + HTTP error: + + + Upload new documents + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 1 + + Upload new documents + + + Drop documents here or + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Drop documents here or + + + Browse files + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 13 + + Browse files + + + Dismiss completed + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 4 + + This button dismisses all status messages about processed documents on the dashboard (failed and successful) + Dismiss completed + + + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 25 + + This is shown as a summary line when there are more than 5 document in the processing pipeline. + {VAR_PLURAL, plural, =1 {One more document} other { more documents}} + + + Open document + + src/app/components/dashboard/widgets/upload-file-widget/upload-file-widget.component.html + 45 + + Open document + + + First steps + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 1 + + First steps + + + Paperless is running! :) + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 5 + + Paperless is running! :) + + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 6,7 + + You can start uploading documents by dropping them in the file upload box to the right or by dropping them in the configured consumption folder and they'll start showing up in the documents list. After you've added some metadata to your documents, use the filtering mechanisms of paperless to create custom views (such as 'Recently added', 'Tagged TODO') and they will appear on the dashboard instead of this message. + + + Paperless offers some more features that try to make your life easier: + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 8 + + Paperless offers some more features that try to make your life easier: + + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 10 + + Once you've got a couple documents in paperless and added metadata to them, paperless can assign that metadata to new documents automatically. + + + You can configure paperless to read your mails and add documents from attached files. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 11 + + You can configure paperless to read your mails and add documents from attached files. + + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + src/app/components/dashboard/widgets/welcome-widget/welcome-widget.component.html + 13 + + Consult the documentation on how to use these features. The section on basic usage also has some information on how to use paperless in general. + + + Metadata + + src/app/components/document-detail/metadata-collapse/metadata-collapse.component.ts + 18 + + Metadata + + + Select + + src/app/components/common/select-dialog/select-dialog.component.ts + 18 + + Select + + + Please select an object + + src/app/components/common/select-dialog/select-dialog.component.ts + 21 + + Please select an object + + + Invalid date. + + src/app/components/common/input/date/date.component.html + 14 + + Invalid date. + + + Searching document with asn + + src/app/components/document-asn/document-asn.component.html + 1 + + Searching document with asn + + + Yes + + src/app/pipes/yes-no.pipe.ts + 9 + + Yes + + + No + + src/app/pipes/yes-no.pipe.ts + 9 + + No + + + (no title) + + src/app/pipes/document-title.pipe.ts + 12 + + (no title) + + + English (US) + + src/app/services/settings.service.ts + 90 + + English (US) + + + English (GB) + + src/app/services/settings.service.ts + 91 + + English (GB) + + + German + + src/app/services/settings.service.ts + 92 + + German + + + Dutch + + src/app/services/settings.service.ts + 93 + + Dutch + + + French + + src/app/services/settings.service.ts + 94 + + French + + + Portuguese + + src/app/services/settings.service.ts + 95 + + Portuguese + + + Portuguese (Brazil) + + src/app/services/settings.service.ts + 96 + + Portuguese (Brazil) + + + Italian + + src/app/services/settings.service.ts + 97 + + Italian + + + Romanian + + src/app/services/settings.service.ts + 98 + + Romanian + + + Russian + + src/app/services/settings.service.ts + 99 + + Russian + + + Spanish + + src/app/services/settings.service.ts + 100 + + Spanish + + + Polish + + src/app/services/settings.service.ts + 101 + + Polish + + + Swedish + + src/app/services/settings.service.ts + 102 + + Swedish + + + ISO 8601 + + src/app/services/settings.service.ts + 107 + + ISO 8601 + + + Document already exists. + + src/app/services/consumer-status.service.ts + 15 + + Document already exists. + + + File not found. + + src/app/services/consumer-status.service.ts + 16 + + File not found. + + + Pre-consume script does not exist. + + src/app/services/consumer-status.service.ts + 17 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Pre-consume script does not exist. + + + Error while executing pre-consume script. + + src/app/services/consumer-status.service.ts + 18 + + Pre-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing pre-consume script. + + + Post-consume script does not exist. + + src/app/services/consumer-status.service.ts + 19 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Post-consume script does not exist. + + + Error while executing post-consume script. + + src/app/services/consumer-status.service.ts + 20 + + Post-Consume is a term that appears like that in the documentation as well and does not need a specific translation + Error while executing post-consume script. + + + Received new file. + + src/app/services/consumer-status.service.ts + 21 + + Received new file. + + + File type not supported. + + src/app/services/consumer-status.service.ts + 22 + + File type not supported. + + + Processing document... + + src/app/services/consumer-status.service.ts + 23 + + Processing document... + + + Generating thumbnail... + + src/app/services/consumer-status.service.ts + 24 + + Generating thumbnail... + + + Retrieving date from document... + + src/app/services/consumer-status.service.ts + 25 + + Retrieving date from document... + + + Saving document... + + src/app/services/consumer-status.service.ts + 26 + + Saving document... + + + Finished. + + src/app/services/consumer-status.service.ts + 27 + + Finished. + + + Error + + src/app/services/toast.service.ts + 35 + + Error + + + Information + + src/app/services/toast.service.ts + 39 + + Information + + + Correspondent + + src/app/services/rest/document.service.ts + 18 + + Correspondent + + + Document type + + src/app/services/rest/document.service.ts + 20 + + Document type + + + Created + + src/app/services/rest/document.service.ts + 21 + + Created + + + Added + + src/app/services/rest/document.service.ts + 22 + + Added + + + Modified + + src/app/services/rest/document.service.ts + 23 + + Modified + + + Search score + + src/app/services/rest/document.service.ts + 28 + + Score is a value returned by the full text search engine and specifies how well a result matches the given query + Search score + + + Create new item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 50 + + Create new item + + + Edit item + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 54 + + Edit item + + + Could not save element: + + src/app/components/common/edit-dialog/edit-dialog.component.ts + 58 + + Could not save element: + + + Automatic + + src/app/components/manage/generic-list/generic-list.component.ts + 39 + + 自动 + + + Do you really want to delete this element? + + src/app/components/manage/generic-list/generic-list.component.ts + 97 + + Do you really want to delete this element? + + + Associated documents will not be deleted. + + src/app/components/manage/generic-list/generic-list.component.ts + 104 + + Associated documents will not be deleted. + + + Delete + + src/app/components/manage/generic-list/generic-list.component.ts + 106 + + Delete + + + Error while deleting element: + + src/app/components/manage/generic-list/generic-list.component.ts + 114 + + Error while deleting element: + + + Any word + + src/app/data/matching-model.ts + 12 + + 任何词 + + + Any: Document contains any of these words (space separated) + + src/app/data/matching-model.ts + 12 + + Any: Document contains any of these words (space separated) + + + All words + + src/app/data/matching-model.ts + 13 + + 所有词 + + + All: Document contains all of these words (space separated) + + src/app/data/matching-model.ts + 13 + + All: Document contains all of these words (space separated) + + + Exact match + + src/app/data/matching-model.ts + 14 + + 完全符合 + + + Exact: Document contains this string + + src/app/data/matching-model.ts + 14 + + Exact: Document contains this string + + + Regular expression + + src/app/data/matching-model.ts + 15 + + 正则表达式 + + + Regular expression: Document matches this regular expression + + src/app/data/matching-model.ts + 15 + + Regular expression: Document matches this regular expression + + + Fuzzy word + + src/app/data/matching-model.ts + 16 + + 模糊词汇 + + + Fuzzy: Document contains a word similar to this word + + src/app/data/matching-model.ts + 16 + + Fuzzy: Document contains a word similar to this word + + + Auto: Learn matching automatically + + src/app/data/matching-model.ts + 17 + + Auto: Learn matching automatically + + + + diff --git a/src-ui/src/manifest.webmanifest b/src-ui/src/manifest.webmanifest new file mode 100644 index 000000000..60151bb5c --- /dev/null +++ b/src-ui/src/manifest.webmanifest @@ -0,0 +1,14 @@ +{ + "background_color": "white", + "description": "A supercharged version of paperless: scan, index and archive all your physical documents", + "display": "fullscreen", + "icons": [ + { + "src": "favicon.ico", + "sizes": "128x128" + } + ], + "name": "Paperless NG", + "short_name": "Paperless NG", + "start_url": "/" +} diff --git a/src-ui/src/print.scss b/src-ui/src/print.scss new file mode 100644 index 000000000..1b29c5100 --- /dev/null +++ b/src-ui/src/print.scss @@ -0,0 +1,30 @@ +@media print { + #sidebarMenu, .btn-toolbar { + display: none !important + } + + .sticky-top { + display: none; + } + + main, main.ml-sm-auto, main.mx-sm-auto { + margin-left: 0 !important; + margin-right: 0 !important; + } + + main.col-lg-10 { + max-width: 100%; + flex-basis: 100%; + display: block; + } + + .d-none.d-lg-table-cell { // always display ASN on print + display: table-cell !important; + } + + app-document-list table { + thead th:first-child, tbody td:first-child { // hide checkboxes + display: none; + } + } +} diff --git a/src-ui/src/styles.scss b/src-ui/src/styles.scss index b0b66b7f9..88f5b2665 100644 --- a/src-ui/src/styles.scss +++ b/src-ui/src/styles.scss @@ -1,7 +1,8 @@ @import "theme"; - +@import "theme_dark"; +@import "print"; @import "node_modules/bootstrap/scss/bootstrap"; - +@import "~@ng-select/ng-select/themes/default.theme.css"; .toolbaricon { width: 1.2em; @@ -20,7 +21,7 @@ } body { - font-size: .875rem; + font-size: 0.875rem; } .form-control-dark { @@ -65,4 +66,53 @@ body { display: block; background-size: 1rem; float: right; -} \ No newline at end of file +} + +.paperless-input-select { + .ng-select { + position: relative; + flex: 1 1 auto; + margin-bottom: 0; + min-height: calc(1.5em + 0.75rem + 5px); + line-height: 1.5; + + .ng-select-container { + height: 100%; + border-top-right-radius: 0; + border-bottom-right-radius: 0; + + .ng-value-container .ng-input { + top: 10px; + } + } + + .ng-dropdown-panel .ng-dropdown-panel-items .ng-option.ng-option-selected, + .ng-dropdown-panel .ng-dropdown-panel-items .ng-option.ng-option-selected.ng-option-marked { + background: none; + } + } +} + +.paperless-input-tags { + .ng-select.ng-select-multiple .ng-select-container .ng-value-container .ng-value { + background-color: transparent; + } + + .ng-select.ng-select-multiple .ng-select-container .ng-value-container { + padding-top: 1px; + } +} + +@supports (-webkit-touch-callout: none) { + input[type="number"], + input[type="search"], + input[type="text"], + select:focus, + textarea { + font-size: 16px; + } +} + +.ngx-file-drop__drop-zone--over { + background-color: $primaryFaded !important; +} diff --git a/src-ui/src/theme.scss b/src-ui/src/theme.scss index 88f3ae30f..113968a9f 100644 --- a/src-ui/src/theme.scss +++ b/src-ui/src/theme.scss @@ -1,6 +1,11 @@ $paperless-green: #17541f; $primary: #17541f; +$primaryFaded: #d1ddd2; $theme-colors: ( "primary": $primary -); \ No newline at end of file +); + +.bg-body { + background-color: #fff; +} diff --git a/src-ui/src/theme_dark.scss b/src-ui/src/theme_dark.scss new file mode 100644 index 000000000..25abac773 --- /dev/null +++ b/src-ui/src/theme_dark.scss @@ -0,0 +1,436 @@ +$primary-dark-mode: #45973a; +$danger-dark-mode: #b71631; +$bg-dark-mode: #161618; +$bg-dark-mode-accent: #21262d; +$bg-light-dark-mode: #1c1c1f; +$text-color-dark-mode: #abb2bf; +$text-color-dark-mode-accent: lighten($text-color-dark-mode, 10%); +$border-color-dark-mode: #47494f; + +* { + transition: background-color 0.3s ease, border-color 0.3s ease; +} + +@mixin dark-mode { + background-color: $bg-dark-mode !important; + color: $text-color-dark-mode; + + .navbar-brand { + color: $text-color-dark-mode; + } + + svg.logo { + .leaf { + color: $primary-dark-mode !important; + } + .text { + fill: $text-color-dark-mode !important; + } + } + + .bg-light { + background-color: $bg-light-dark-mode !important; + + a, + div { + color: $text-color-dark-mode; + } + } + + .bg-body { + background-color: $bg-dark-mode !important; + } + + .text-light { + color: $text-color-dark-mode !important; + } + + .border { + border-color: $border-color-dark-mode !important; + } + + .border-right { + border-right: 1px solid $border-color-dark-mode !important; + } + + .border-left { + border-left: 1px solid $border-color-dark-mode !important; + } + + .border-bottom { + border-bottom: 1px solid $border-color-dark-mode !important; + } + + .nav-link { + color: $text-color-dark-mode !important; + + &.active { + background-color: $bg-dark-mode; + color: $text-color-dark-mode; + border-color: $border-color-dark-mode $border-color-dark-mode $bg-dark-mode; + + .close { + background-color: inherit !important; + } + } + + &:hover { + color: $text-color-dark-mode-accent !important; + border-color: $border-color-dark-mode $border-color-dark-mode $bg-dark-mode; + } + } + + .page-item.active .page-link { + background-color: darken($primary-dark-mode, 10%); + } + + .nav-tabs { + border-color: $border-color-dark-mode; + + .nav-link { + color: $primary-dark-mode !important; + + &.active { + color: $text-color-dark-mode !important; + } + } + } + + .dropdown-menu { + background-color: $bg-light-dark-mode; + + .dropdown-divider { + border-color: $border-color-dark-mode; + } + + .dropdown-item { + color: $text-color-dark-mode; + + &:hover { + background-color: $bg-light-dark-mode; + color: $text-color-dark-mode; + } + } + + .dropdown-item.disabled { + color: darken($text-color-dark-mode, 20%); + } + } + + .card { + background-color: $bg-light-dark-mode; + + .card-text { + color: $text-color-dark-mode; + } + } + + .text-dark { + color: $text-color-dark-mode !important; + } + + .modal-content, .modal-header, .modal-body, .modal-footer { + background-color: $bg-light-dark-mode; + border-color: $border-color-dark-mode; + } + + app-tag .badge { + filter: brightness(.8); + } + + .badge-light { + background-color: darken($bg-dark-mode, 20%); + color: $text-color-dark-mode-accent; + } + + .doc-img-container { + border: none !important; + border-top-left-radius: .25rem; + border-top-right-radius: .25rem; + overflow: hidden; + } + + .doc-img { + mix-blend-mode: normal; + border-radius: 0; + border-color: $bg-dark-mode; + filter: invert(10%); + &.border-right { + border-right: none !important; + } + } + + .doc-img.inverted { + filter: invert(95%) hue-rotate(180deg); + } + + .card-selected .doc-img { + mix-blend-mode: luminosity; + } + + .toast { + background-color: opacify($bg-light-dark-mode, .85); + } + + .toast-header { + background-color: opacify($bg-dark-mode, .85); + } + + a, + .card-title a { + color: $primary-dark-mode; + + &:hover { + color: lighten($primary, 10%); + } + } + + table { + background-color: $bg-dark-mode; + color: $text-color-dark-mode; + border-color: $border-color-dark-mode; + + .des, + .asc { + background-color: transparent !important; + color: $text-color-dark-mode; + border-color: $border-color-dark-mode; + + &::after { + filter: invert(0.8); /* arrow is a black inline png bkgd image (!) so use filter */ + } + } + + tr:hover { + background-color: $bg-light-dark-mode; + color: $text-color-dark-mode-accent; + } + } + + .table td, + .table th { + border-color: $border-color-dark-mode; + } + + .table-row-selected { + background-color: $bg-light-dark-mode; + } + + .close { + color: $text-color-dark-mode; + text-shadow: 0 1px 0 #666; + } + + .btn-outline-primary { + border-color: $primary-dark-mode; + color: $primary-dark-mode; + + &:not(:disabled):not(.disabled).active, + &:not(:disabled):not(.disabled):hover { + background-color: darken($primary-dark-mode, 10%); + border-color: darken($primary-dark-mode, 10%); + color: $text-color-dark-mode-accent; + } + } + + .btn-outline-secondary { + border-color: darken($text-color-dark-mode, 30%); + color: $text-color-dark-mode; + + &:not(:disabled):not(.disabled):hover { + background-color: $bg-dark-mode; + } + } + + .btn-outline-danger { + border-color: $danger-dark-mode; + color: $danger-dark-mode; + + &:not(:disabled):not(.disabled):hover { + background-color: darken($danger-dark-mode, 10%); + border-color: darken($danger-dark-mode, 10%); + color: $text-color-dark-mode-accent; + } + } + + .btn-outline-dark { + border-color: $border-color-dark-mode; + color: $text-color-dark-mode; + + &:not(:disabled):not(.disabled):hover { + color: $text-color-dark-mode-accent; + } + } + + .btn-light:not(:disabled):not(.disabled) { + background-color: $bg-dark-mode; + color: $text-color-dark-mode-accent; + + &:hover { + background-color: $text-color-dark-mode; + color: $bg-dark-mode; + } + } + + .btn-link:not(:disabled):not(.disabled) { + color: $primary-dark-mode; + } + + .btn-link:hover, + .btn-outline-primary:not(:disabled):not(.disabled).active, + .btn-outline-primary:not(:disabled):not(.disabled):active, + .show > .btn-outline-primary.dropdown-toggle { + color: $text-color-dark-mode-accent; + } + + button.bg-light:hover { + background-color: $bg-dark-mode !important; + } + + .card-footer button:hover { + color: $primary-dark-mode !important; + } + + .form-control:not(.is-invalid):not(.btn), + input:not(.is-invalid), + textarea:not(.is-invalid) { + border-color: $border-color-dark-mode; /* we dont want to override controls that get highlighting for errors */ + } + + .form-control:not(.btn), + input, + select, + textarea { + background-color: $bg-dark-mode; + color: $text-color-dark-mode; + + &::placeholder { + color: $text-color-dark-mode; + } + + &:focus { + background-color: $bg-light-dark-mode !important; + color: darken($text-color-dark-mode, 10%) !important; + } + } + + .ng-select-container, + .ng-select.ng-select-opened > .ng-select-container, + .ng-dropdown-panel, + .ng-dropdown-panel .ng-dropdown-panel-items .ng-option { + background-color: $bg-dark-mode; + color: $text-color-dark-mode; + border-color: $border-color-dark-mode; + + input:focus { + background-color: transparent !important; + } + } + + .ng-dropdown-panel .ng-dropdown-panel-items .ng-option:hover, + .ng-dropdown-panel .ng-dropdown-panel-items .ng-option.ng-option-marked { + background-color: $bg-light-dark-mode; + } + + .custom-control-label:before { + background-color: $bg-dark-mode; + color: $text-color-dark-mode; + } + + .custom-control-input:checked ~ .custom-control-label::before { + color: $text-color-dark-mode-accent; + } + + .input-group-text { + color: $text-color-dark-mode; + background-color: $bg-light-dark-mode; + border-color: $border-color-dark-mode; + } + + .list-group-item { + color: $text-color-dark-mode; + background-color: $bg-light-dark-mode; + border-color: $border-color-dark-mode; + } + + .page-item.disabled .page-link { + background-color: $bg-dark-mode; + border-color: $border-color-dark-mode; + } + + .list-group-item, + .page-link { + background-color: $bg-light-dark-mode; + border-color: $border-color-dark-mode; + } + + .page-item.active .page-link { + border-color: $border-color-dark-mode; + color: $text-color-dark-mode-accent; + } + + .progress { + background-color: $border-color-dark-mode; + } + + .alert-danger { + color: $text-color-dark-mode-accent; + background-color: darken($danger-dark-mode, 20%); + border-color: darken($danger-dark-mode, 20%); + } + + .bg-dark { + background-color: $bg-light-dark-mode !important; + } + + .ngx-file-drop__drop-zone--over { + background-color: darken($primary-dark-mode, 35%) !important; + } + + .alert-secondary { + background-color: $bg-light-dark-mode; + border-color: darken($bg-light-dark-mode, 10%); + color: $text-color-dark-mode-accent; + } + + .progress-bar.bg-primary { + background-color: darken($primary-dark-mode, 5%) !important; + } + + .popover { + .popover-header, + .popover-body { + background-color: $bg-dark-mode-accent; + border-color: $border-color-dark-mode; + } + } + + $placements: 'top', 'right', 'bottom', 'left'; + + @each $placement in $placements { + .bs-popover-#{$placement} > .arrow::after, + .bs-popover-auto[x-placement^=#{$placement}] > .arrow::after { + border-#{$placement}-color: $bg-dark-mode-accent; + } + } + + .bs-popover-bottom .popover-header::before, + .bs-popover-auto[x-placement^=bottom] .popover-header::before { + border-bottom-color: $bg-dark-mode-accent; + } + + .ngb-dp-header, + .ngb-dp-weekdays, + .ngb-dp-month { + background-color: $bg-light-dark-mode; + } +} + +body.color-scheme-dark { + @include dark-mode; +} +body.color-scheme-system { + @media (prefers-color-scheme: dark) { + @include dark-mode; + } +} diff --git a/src/documents/admin.py b/src/documents/admin.py old mode 100755 new mode 100644 index 055a6fd93..c8c16d791 --- a/src/documents/admin.py +++ b/src/documents/admin.py @@ -1,10 +1,7 @@ from django.contrib import admin -from django.utils.html import format_html, format_html_join -from django.utils.safestring import mark_safe -from whoosh.writing import AsyncWriter -from . import index -from .models import Correspondent, Document, DocumentType, Log, Tag +from .models import Correspondent, Document, DocumentType, Tag, \ + SavedView, SavedViewFilterRule class CorrespondentAdmin(admin.ModelAdmin): @@ -22,12 +19,12 @@ class TagAdmin(admin.ModelAdmin): list_display = ( "name", - "colour", + "color", "match", "matching_algorithm" ) - list_filter = ("colour", "matching_algorithm") - list_editable = ("colour", "match", "matching_algorithm") + list_filter = ("color", "matching_algorithm") + list_editable = ("color", "match", "matching_algorithm") class DocumentTypeAdmin(admin.ModelAdmin): @@ -49,26 +46,31 @@ class DocumentAdmin(admin.ModelAdmin): "modified", "mime_type", "storage_type", - "filename") + "filename", + "checksum", + "archive_filename", + "archive_checksum" + ) list_display_links = ("title",) list_display = ( - "correspondent", + "id", "title", - "tags_", - "created", + "mime_type", + "filename", + "archive_filename" ) list_filter = ( - "document_type", - "tags", - "correspondent" + ("mime_type"), + ("archive_serial_number", admin.EmptyFieldListFilter), + ("archive_filename", admin.EmptyFieldListFilter), ) filter_horizontal = ("tags",) - ordering = ["-created", "correspondent"] + ordering = ["-id"] date_hierarchy = "created" @@ -80,59 +82,40 @@ class DocumentAdmin(admin.ModelAdmin): created_.short_description = "Created" def delete_queryset(self, request, queryset): - ix = index.open_index() - with AsyncWriter(ix) as writer: + from documents import index + + with index.open_index_writer() as writer: for o in queryset: index.remove_document(writer, o) + super(DocumentAdmin, self).delete_queryset(request, queryset) def delete_model(self, request, obj): + from documents import index index.remove_document_from_index(obj) super(DocumentAdmin, self).delete_model(request, obj) def save_model(self, request, obj, form, change): + from documents import index index.add_or_update_document(obj) super(DocumentAdmin, self).save_model(request, obj, form, change) - @mark_safe - def tags_(self, obj): - r = "" - for tag in obj.tags.all(): - r += self._html_tag( - "span", - tag.name + ", " - ) - return r - @staticmethod - def _html_tag(kind, inside=None, **kwargs): - attributes = format_html_join(' ', '{}="{}"', kwargs.items()) - - if inside is not None: - return format_html("<{kind} {attributes}>{inside}", - kind=kind, attributes=attributes, inside=inside) - - return format_html("<{} {}/>", kind, attributes) +class RuleInline(admin.TabularInline): + model = SavedViewFilterRule -class LogAdmin(admin.ModelAdmin): +class SavedViewAdmin(admin.ModelAdmin): - def has_add_permission(self, request): - return False + list_display = ("name", "user") - def has_change_permission(self, request, obj=None): - return False - - list_display = ("created", "message", "level",) - list_filter = ("level", "created",) - - ordering = ('-created',) - - list_display_links = ("created", "message") + inlines = [ + RuleInline + ] admin.site.register(Correspondent, CorrespondentAdmin) admin.site.register(Tag, TagAdmin) admin.site.register(DocumentType, DocumentTypeAdmin) admin.site.register(Document, DocumentAdmin) -admin.site.register(Log, LogAdmin) +admin.site.register(SavedView, SavedViewAdmin) diff --git a/src/documents/apps.py b/src/documents/apps.py index 2cd7d6c0e..e21e14097 100644 --- a/src/documents/apps.py +++ b/src/documents/apps.py @@ -1,34 +1,30 @@ from django.apps import AppConfig +from django.utils.translation import gettext_lazy as _ + class DocumentsConfig(AppConfig): name = "documents" - def ready(self): + verbose_name = _("Documents") - from .signals import document_consumption_started + def ready(self): from .signals import document_consumption_finished from .signals.handlers import ( add_inbox_tags, - run_pre_consume_script, - run_post_consume_script, set_log_entry, set_correspondent, set_document_type, set_tags, add_to_index - ) - document_consumption_started.connect(run_pre_consume_script) - document_consumption_finished.connect(add_inbox_tags) document_consumption_finished.connect(set_correspondent) document_consumption_finished.connect(set_document_type) document_consumption_finished.connect(set_tags) document_consumption_finished.connect(set_log_entry) document_consumption_finished.connect(add_to_index) - document_consumption_finished.connect(run_post_consume_script) AppConfig.ready(self) diff --git a/src/documents/bulk_download.py b/src/documents/bulk_download.py new file mode 100644 index 000000000..8c675b4b5 --- /dev/null +++ b/src/documents/bulk_download.py @@ -0,0 +1,60 @@ +from zipfile import ZipFile + +from documents.models import Document + + +class BulkArchiveStrategy: + + def __init__(self, zipf: ZipFile): + self.zipf = zipf + + def make_unique_filename(self, + doc: Document, + archive: bool = False, + folder: str = ""): + counter = 0 + while True: + filename = folder + doc.get_public_filename(archive, counter) + if filename in self.zipf.namelist(): + counter += 1 + else: + return filename + + def add_document(self, doc: Document): + raise NotImplementedError() # pragma: no cover + + +class OriginalsOnlyStrategy(BulkArchiveStrategy): + + def add_document(self, doc: Document): + self.zipf.write(doc.source_path, self.make_unique_filename(doc)) + + +class ArchiveOnlyStrategy(BulkArchiveStrategy): + + def __init__(self, zipf): + super(ArchiveOnlyStrategy, self).__init__(zipf) + + def add_document(self, doc: Document): + if doc.has_archive_version: + self.zipf.write(doc.archive_path, + self.make_unique_filename(doc, archive=True)) + else: + self.zipf.write(doc.source_path, + self.make_unique_filename(doc)) + + +class OriginalAndArchiveStrategy(BulkArchiveStrategy): + + def add_document(self, doc: Document): + if doc.has_archive_version: + self.zipf.write( + doc.archive_path, self.make_unique_filename( + doc, archive=True, folder="archive/" + ) + ) + + self.zipf.write( + doc.source_path, + self.make_unique_filename(doc, folder="originals/") + ) diff --git a/src/documents/bulk_edit.py b/src/documents/bulk_edit.py new file mode 100644 index 000000000..7503eafc5 --- /dev/null +++ b/src/documents/bulk_edit.py @@ -0,0 +1,106 @@ +import itertools + +from django.db.models import Q +from django_q.tasks import async_task + +from documents.models import Document, Correspondent, DocumentType + + +def set_correspondent(doc_ids, correspondent): + if correspondent: + correspondent = Correspondent.objects.get(id=correspondent) + + qs = Document.objects.filter( + Q(id__in=doc_ids) & ~Q(correspondent=correspondent)) + affected_docs = [doc.id for doc in qs] + qs.update(correspondent=correspondent) + + async_task( + "documents.tasks.bulk_update_documents", document_ids=affected_docs) + + return "OK" + + +def set_document_type(doc_ids, document_type): + if document_type: + document_type = DocumentType.objects.get(id=document_type) + + qs = Document.objects.filter( + Q(id__in=doc_ids) & ~Q(document_type=document_type)) + affected_docs = [doc.id for doc in qs] + qs.update(document_type=document_type) + + async_task( + "documents.tasks.bulk_update_documents", document_ids=affected_docs) + + return "OK" + + +def add_tag(doc_ids, tag): + + qs = Document.objects.filter(Q(id__in=doc_ids) & ~Q(tags__id=tag)) + affected_docs = [doc.id for doc in qs] + + DocumentTagRelationship = Document.tags.through + + DocumentTagRelationship.objects.bulk_create([ + DocumentTagRelationship( + document_id=doc, tag_id=tag) for doc in affected_docs + ]) + + async_task( + "documents.tasks.bulk_update_documents", document_ids=affected_docs) + + return "OK" + + +def remove_tag(doc_ids, tag): + + qs = Document.objects.filter(Q(id__in=doc_ids) & Q(tags__id=tag)) + affected_docs = [doc.id for doc in qs] + + DocumentTagRelationship = Document.tags.through + + DocumentTagRelationship.objects.filter( + Q(document_id__in=affected_docs) & + Q(tag_id=tag) + ).delete() + + async_task( + "documents.tasks.bulk_update_documents", document_ids=affected_docs) + + return "OK" + + +def modify_tags(doc_ids, add_tags, remove_tags): + qs = Document.objects.filter(id__in=doc_ids) + affected_docs = [doc.id for doc in qs] + + DocumentTagRelationship = Document.tags.through + + DocumentTagRelationship.objects.filter( + document_id__in=affected_docs, + tag_id__in=remove_tags, + ).delete() + + DocumentTagRelationship.objects.bulk_create([DocumentTagRelationship( + document_id=doc, tag_id=tag) for (doc, tag) in itertools.product( + affected_docs, add_tags) + ], ignore_conflicts=True) + + async_task( + "documents.tasks.bulk_update_documents", document_ids=affected_docs) + + return "OK" + + +def delete(doc_ids): + Document.objects.filter(id__in=doc_ids).delete() + + from documents import index + + with index.open_index_writer() as writer: + for id in doc_ids: + index.remove_document_by_id(writer, id) + + return "OK" diff --git a/src/documents/checks.py b/src/documents/checks.py index 3e3ddb1fb..ba55b1397 100644 --- a/src/documents/checks.py +++ b/src/documents/checks.py @@ -2,6 +2,7 @@ import textwrap from django.conf import settings from django.core.checks import Error, register +from django.core.exceptions import FieldError from django.db.utils import OperationalError, ProgrammingError from documents.signals import document_consumer_declaration @@ -16,7 +17,7 @@ def changed_password_check(app_configs, **kwargs): try: encrypted_doc = Document.objects.filter( storage_type=Document.STORAGE_TYPE_GPG).first() - except (OperationalError, ProgrammingError): + except (OperationalError, ProgrammingError, FieldError): return [] # No documents table yet if encrypted_doc: @@ -50,6 +51,6 @@ def parser_check(app_configs, **kwargs): if len(parsers) == 0: return [Error("No parsers found. This is a bug. The consumer won't be " - "able to onsume any documents without parsers.")] + "able to consume any documents without parsers.")] else: return [] diff --git a/src/documents/classifier.py b/src/documents/classifier.py old mode 100755 new mode 100644 index 60c9abeec..36e906350 --- a/src/documents/classifier.py +++ b/src/documents/classifier.py @@ -3,12 +3,9 @@ import logging import os import pickle import re +import shutil from django.conf import settings -from sklearn.feature_extraction.text import CountVectorizer -from sklearn.neural_network import MLPClassifier -from sklearn.preprocessing import MultiLabelBinarizer, LabelBinarizer -from sklearn.utils.multiclass import type_of_target from documents.models import Document, MatchingModel @@ -17,7 +14,11 @@ class IncompatibleClassifierVersionError(Exception): pass -logger = logging.getLogger(__name__) +class ClassifierModelCorruptError(Exception): + pass + + +logger = logging.getLogger("paperless.classifier") def preprocess_content(content): @@ -26,15 +27,46 @@ def preprocess_content(content): return content +def load_classifier(): + if not os.path.isfile(settings.MODEL_FILE): + logger.debug( + f"Document classification model does not exist (yet), not " + f"performing automatic matching." + ) + return None + + classifier = DocumentClassifier() + try: + classifier.load() + + except (ClassifierModelCorruptError, + IncompatibleClassifierVersionError): + # there's something wrong with the model file. + logger.exception( + f"Unrecoverable error while loading document " + f"classification model, deleting model file." + ) + os.unlink(settings.MODEL_FILE) + classifier = None + except OSError: + logger.exception( + f"IO error while loading document classification model" + ) + classifier = None + except Exception: + logger.exception( + f"Unknown error while loading document classification model" + ) + classifier = None + + return classifier + + class DocumentClassifier(object): FORMAT_VERSION = 6 def __init__(self): - # mtime of the model file on disk. used to prevent reloading when - # nothing has changed. - self.classifier_version = 0 - # hash of the training data. used to prevent re-training when the # training data has not changed. self.data_hash = None @@ -45,20 +77,15 @@ class DocumentClassifier(object): self.correspondent_classifier = None self.document_type_classifier = None - def reload(self): - if os.path.getmtime(settings.MODEL_FILE) > self.classifier_version: - with open(settings.MODEL_FILE, "rb") as f: - schema_version = pickle.load(f) + def load(self): + with open(settings.MODEL_FILE, "rb") as f: + schema_version = pickle.load(f) - if schema_version != self.FORMAT_VERSION: - raise IncompatibleClassifierVersionError( - "Cannor load classifier, incompatible versions.") - else: - if self.classifier_version > 0: - # Don't be confused by this check. It's simply here - # so that we wont log anything on initial reload. - logger.info("Classifier updated on disk, " - "reloading classifier models") + if schema_version != self.FORMAT_VERSION: + raise IncompatibleClassifierVersionError( + "Cannor load classifier, incompatible versions.") + else: + try: self.data_hash = pickle.load(f) self.data_vectorizer = pickle.load(f) self.tags_binarizer = pickle.load(f) @@ -66,10 +93,14 @@ class DocumentClassifier(object): self.tags_classifier = pickle.load(f) self.correspondent_classifier = pickle.load(f) self.document_type_classifier = pickle.load(f) - self.classifier_version = os.path.getmtime(settings.MODEL_FILE) + except Exception: + raise ClassifierModelCorruptError() - def save_classifier(self): - with open(settings.MODEL_FILE, "wb") as f: + def save(self): + target_file = settings.MODEL_FILE + target_file_temp = settings.MODEL_FILE + ".part" + + with open(target_file_temp, "wb") as f: pickle.dump(self.FORMAT_VERSION, f) pickle.dump(self.data_hash, f) pickle.dump(self.data_vectorizer, f) @@ -80,14 +111,19 @@ class DocumentClassifier(object): pickle.dump(self.correspondent_classifier, f) pickle.dump(self.document_type_classifier, f) + if os.path.isfile(target_file): + os.unlink(target_file) + shutil.move(target_file_temp, target_file) + def train(self): + data = list() labels_tags = list() labels_correspondent = list() labels_document_type = list() # Step 1: Extract and preprocess training data from the database. - logging.getLogger(__name__).debug("Gathering data from database...") + logger.debug("Gathering data from database...") m = hashlib.sha1() for doc in Document.objects.order_by('pk').exclude(tags__is_inbox_tag=True): # NOQA: E501 preprocessed_content = preprocess_content(doc.content) @@ -108,10 +144,11 @@ class DocumentClassifier(object): m.update(y.to_bytes(4, 'little', signed=True)) labels_correspondent.append(y) - tags = [tag.pk for tag in doc.tags.filter( + tags = sorted([tag.pk for tag in doc.tags.filter( matching_algorithm=MatchingModel.MATCH_AUTO - )] - m.update(bytearray(tags)) + )]) + for tag in tags: + m.update(tag.to_bytes(4, 'little', signed=True)) labels_tags.append(tags) if not data: @@ -134,7 +171,7 @@ class DocumentClassifier(object): num_correspondents = len(set(labels_correspondent) | {-1}) - 1 num_document_types = len(set(labels_document_type) | {-1}) - 1 - logging.getLogger(__name__).debug( + logger.debug( "{} documents, {} tag(s), {} correspondent(s), " "{} document type(s).".format( len(data), @@ -144,8 +181,12 @@ class DocumentClassifier(object): ) ) + from sklearn.feature_extraction.text import CountVectorizer + from sklearn.neural_network import MLPClassifier + from sklearn.preprocessing import MultiLabelBinarizer, LabelBinarizer + # Step 2: vectorize data - logging.getLogger(__name__).debug("Vectorizing data...") + logger.debug("Vectorizing data...") self.data_vectorizer = CountVectorizer( analyzer="word", ngram_range=(1, 2), @@ -155,7 +196,7 @@ class DocumentClassifier(object): # Step 3: train the classifiers if num_tags > 0: - logging.getLogger(__name__).debug("Training tags classifier...") + logger.debug("Training tags classifier...") if num_tags == 1: # Special case where only one tag has auto: @@ -174,12 +215,12 @@ class DocumentClassifier(object): self.tags_classifier.fit(data_vectorized, labels_tags_vectorized) else: self.tags_classifier = None - logging.getLogger(__name__).debug( + logger.debug( "There are no tags. Not training tags classifier." ) if num_correspondents > 0: - logging.getLogger(__name__).debug( + logger.debug( "Training correspondent classifier..." ) self.correspondent_classifier = MLPClassifier(tol=0.01) @@ -189,13 +230,13 @@ class DocumentClassifier(object): ) else: self.correspondent_classifier = None - logging.getLogger(__name__).debug( + logger.debug( "There are no correspondents. Not training correspondent " "classifier." ) if num_document_types > 0: - logging.getLogger(__name__).debug( + logger.debug( "Training document type classifier..." ) self.document_type_classifier = MLPClassifier(tol=0.01) @@ -205,7 +246,7 @@ class DocumentClassifier(object): ) else: self.document_type_classifier = None - logging.getLogger(__name__).debug( + logger.debug( "There are no document types. Not training document type " "classifier." ) @@ -237,6 +278,8 @@ class DocumentClassifier(object): return None def predict_tags(self, content): + from sklearn.utils.multiclass import type_of_target + if self.tags_classifier: X = self.data_vectorizer.transform([preprocess_content(content)]) y = self.tags_classifier.predict(X) diff --git a/src/documents/consumer.py b/src/documents/consumer.py old mode 100755 new mode 100644 index e4da51f1d..39275cee3 --- a/src/documents/consumer.py +++ b/src/documents/consumer.py @@ -1,16 +1,20 @@ import datetime import hashlib -import logging import os +import uuid +from subprocess import Popen import magic +from asgiref.sync import async_to_sync +from channels.layers import get_channel_layer from django.conf import settings from django.db import transaction from django.db.models import Q from django.utils import timezone from filelock import FileLock +from rest_framework.reverse import reverse -from .classifier import DocumentClassifier, IncompatibleClassifierVersionError +from .classifier import load_classifier from .file_handling import create_source_path_directory, \ generate_unique_filename from .loggers import LoggingMixin @@ -26,8 +30,45 @@ class ConsumerError(Exception): pass +MESSAGE_DOCUMENT_ALREADY_EXISTS = "document_already_exists" +MESSAGE_FILE_NOT_FOUND = "file_not_found" +MESSAGE_PRE_CONSUME_SCRIPT_NOT_FOUND = "pre_consume_script_not_found" +MESSAGE_PRE_CONSUME_SCRIPT_ERROR = "pre_consume_script_error" +MESSAGE_POST_CONSUME_SCRIPT_NOT_FOUND = "post_consume_script_not_found" +MESSAGE_POST_CONSUME_SCRIPT_ERROR = "post_consume_script_error" +MESSAGE_NEW_FILE = "new_file" +MESSAGE_UNSUPPORTED_TYPE = "unsupported_type" +MESSAGE_PARSING_DOCUMENT = "parsing_document" +MESSAGE_GENERATING_THUMBNAIL = "generating_thumbnail" +MESSAGE_PARSE_DATE = "parse_date" +MESSAGE_SAVE_DOCUMENT = "save_document" +MESSAGE_FINISHED = "finished" + + class Consumer(LoggingMixin): + logging_name = "paperless.consumer" + + def _send_progress(self, current_progress, max_progress, status, + message=None, document_id=None): + payload = { + 'filename': os.path.basename(self.filename) if self.filename else None, # NOQA: E501 + 'task_id': self.task_id, + 'current_progress': current_progress, + 'max_progress': max_progress, + 'status': status, + 'message': message, + 'document_id': document_id + } + async_to_sync(self.channel_layer.group_send)("status_updates", + {'type': 'status_update', + 'data': payload}) + + def _fail(self, message, log_message=None, exc_info=None): + self._send_progress(100, 100, 'FAILED', message) + self.log("error", log_message or message, exc_info=exc_info) + raise ConsumerError(f"{self.filename}: {log_message or message}") + def __init__(self): super().__init__() self.path = None @@ -36,15 +77,16 @@ class Consumer(LoggingMixin): self.override_correspondent_id = None self.override_tag_ids = None self.override_document_type_id = None + self.task_id = None + + self.channel_layer = get_channel_layer() def pre_check_file_exists(self): if not os.path.isfile(self.path): - self.log( - "error", - "Cannot consume {}: It is not a file.".format(self.path) + self._fail( + MESSAGE_FILE_NOT_FOUND, + f"Cannot consume {self.path}: File not found." ) - raise ConsumerError("Cannot consume {}: It is not a file".format( - self.path)) def pre_check_duplicate(self): with open(self.path, "rb") as f: @@ -52,12 +94,9 @@ class Consumer(LoggingMixin): if Document.objects.filter(Q(checksum=checksum) | Q(archive_checksum=checksum)).exists(): # NOQA: E501 if settings.CONSUMER_DELETE_DUPLICATES: os.unlink(self.path) - self.log( - "error", - "Not consuming {}: It is a duplicate.".format(self.filename) - ) - raise ConsumerError( - "Not consuming {}: It is a duplicate.".format(self.filename) + self._fail( + MESSAGE_DOCUMENT_ALREADY_EXISTS, + f"Not consuming {self.filename}: It is a duplicate." ) def pre_check_directories(self): @@ -66,13 +105,72 @@ class Consumer(LoggingMixin): os.makedirs(settings.ORIGINALS_DIR, exist_ok=True) os.makedirs(settings.ARCHIVE_DIR, exist_ok=True) + def run_pre_consume_script(self): + if not settings.PRE_CONSUME_SCRIPT: + return + + if not os.path.isfile(settings.PRE_CONSUME_SCRIPT): + self._fail( + MESSAGE_PRE_CONSUME_SCRIPT_NOT_FOUND, + f"Configured pre-consume script " + f"{settings.PRE_CONSUME_SCRIPT} does not exist.") + + self.log("info", + f"Executing pre-consume script {settings.PRE_CONSUME_SCRIPT}") + + try: + Popen((settings.PRE_CONSUME_SCRIPT, self.path)).wait() + except Exception as e: + self._fail( + MESSAGE_PRE_CONSUME_SCRIPT_ERROR, + f"Error while executing pre-consume script: {e}", + exc_info=True + ) + + def run_post_consume_script(self, document): + if not settings.POST_CONSUME_SCRIPT: + return + + if not os.path.isfile(settings.POST_CONSUME_SCRIPT): + self._fail( + MESSAGE_POST_CONSUME_SCRIPT_NOT_FOUND, + f"Configured post-consume script " + f"{settings.POST_CONSUME_SCRIPT} does not exist." + ) + + self.log( + "info", + f"Executing post-consume script {settings.POST_CONSUME_SCRIPT}" + ) + + try: + Popen(( + settings.POST_CONSUME_SCRIPT, + str(document.pk), + document.get_public_filename(), + os.path.normpath(document.source_path), + os.path.normpath(document.thumbnail_path), + reverse("document-download", kwargs={"pk": document.pk}), + reverse("document-thumb", kwargs={"pk": document.pk}), + str(document.correspondent), + str(",".join(document.tags.all().values_list( + "name", flat=True))) + )).wait() + except Exception as e: + self._fail( + MESSAGE_POST_CONSUME_SCRIPT_ERROR, + f"Error while executing post-consume script: {e}", + exc_info=True + ) + def try_consume_file(self, path, override_filename=None, override_title=None, override_correspondent_id=None, override_document_type_id=None, - override_tag_ids=None): + override_tag_ids=None, + task_id=None): """ Return the document object if it was successfully created. """ @@ -83,6 +181,9 @@ class Consumer(LoggingMixin): self.override_correspondent_id = override_correspondent_id self.override_document_type_id = override_document_type_id self.override_tag_ids = override_tag_ids + self.task_id = task_id or str(uuid.uuid4()) + + self._send_progress(0, 100, 'STARTING', MESSAGE_NEW_FILE) # this is for grouping logging entries for this particular file # together. @@ -95,19 +196,20 @@ class Consumer(LoggingMixin): self.pre_check_directories() self.pre_check_duplicate() - self.log("info", "Consuming {}".format(self.filename)) + self.log("info", f"Consuming {self.filename}") # Determine the parser class. mime_type = magic.from_file(self.path, mime=True) + self.log("debug", f"Detected mime type: {mime_type}") + parser_class = get_parser_class_for_mime_type(mime_type) if not parser_class: - raise ConsumerError(f"No parsers abvailable for {self.filename}") - else: - self.log("debug", - f"Parser: {parser_class.__name__} " - f"based on mime type {mime_type}") + self._fail( + MESSAGE_UNSUPPORTED_TYPE, + f"Unsupported mime type {mime_type}" + ) # Notify all listeners that we're going to do some work. @@ -117,35 +219,55 @@ class Consumer(LoggingMixin): logging_group=self.logging_group ) + self.run_pre_consume_script() + + def progress_callback(current_progress, max_progress): + # recalculate progress to be within 20 and 80 + p = int((current_progress / max_progress) * 50 + 20) + self._send_progress(p, 100, "WORKING") + # This doesn't parse the document yet, but gives us a parser. - document_parser = parser_class(self.logging_group) + document_parser = parser_class(self.logging_group, progress_callback) + + self.log("debug", f"Parser: {type(document_parser).__name__}") # However, this already created working directories which we have to # clean up. # Parse the document. This may take some time. + text = None + date = None + thumbnail = None + archive_path = None + try: + self._send_progress(20, 100, 'WORKING', MESSAGE_PARSING_DOCUMENT) self.log("debug", "Parsing {}...".format(self.filename)) - document_parser.parse(self.path, mime_type) + document_parser.parse(self.path, mime_type, self.filename) self.log("debug", f"Generating thumbnail for {self.filename}...") + self._send_progress(70, 100, 'WORKING', + MESSAGE_GENERATING_THUMBNAIL) thumbnail = document_parser.get_optimised_thumbnail( - self.path, mime_type) + self.path, mime_type, self.filename) text = document_parser.get_text() date = document_parser.get_date() if not date: + self._send_progress(90, 100, 'WORKING', + MESSAGE_PARSE_DATE) date = parse_date(self.filename, text) archive_path = document_parser.get_archive_path() except ParseError as e: document_parser.cleanup() - self.log( - "error", - f"Error while consuming document {self.filename}: {e}") - raise ConsumerError(e) + self._fail( + str(e), + f"Error while consuming document {self.filename}: {e}", + exc_info=True + ) # Prepare the document classifier. @@ -153,15 +275,9 @@ class Consumer(LoggingMixin): # reloading the classifier multiple times, since there are multiple # post-consume hooks that all require the classifier. - try: - classifier = DocumentClassifier() - classifier.reload() - except (FileNotFoundError, IncompatibleClassifierVersionError) as e: - self.log( - "warning", - f"Cannot classify documents: {e}.") - classifier = None + classifier = load_classifier() + self._send_progress(95, 100, 'WORKING', MESSAGE_SAVE_DOCUMENT) # now that everything is done, we can start to store the document # in the system. This will be a transaction and reasonably fast. try: @@ -187,8 +303,7 @@ class Consumer(LoggingMixin): # After everything is in the database, copy the files into # place. If this fails, we'll also rollback the transaction. with FileLock(settings.MEDIA_LOCK): - document.filename = generate_unique_filename( - document, settings.ORIGINALS_DIR) + document.filename = generate_unique_filename(document) create_source_path_directory(document.source_path) self._write(document.storage_type, @@ -198,6 +313,10 @@ class Consumer(LoggingMixin): thumbnail, document.thumbnail_path) if archive_path and os.path.isfile(archive_path): + document.archive_filename = generate_unique_filename( + document, + archive_filename=True + ) create_source_path_directory(document.archive_path) self._write(document.storage_type, archive_path, document.archive_path) @@ -213,21 +332,35 @@ class Consumer(LoggingMixin): # Delete the file only if it was successfully consumed self.log("debug", "Deleting file {}".format(self.path)) os.unlink(self.path) + + # https://github.com/jonaswinkler/paperless-ng/discussions/1037 + shadow_file = os.path.join( + os.path.dirname(self.path), + "._" + os.path.basename(self.path)) + + if os.path.isfile(shadow_file): + self.log("debug", "Deleting file {}".format(shadow_file)) + os.unlink(shadow_file) + except Exception as e: - self.log( - "error", + self._fail( + str(e), f"The following error occured while consuming " - f"{self.filename}: {e}" + f"{self.filename}: {e}", + exc_info=True ) - raise ConsumerError(e) finally: document_parser.cleanup() + self.run_post_consume_script(document) + self.log( "info", "Document {} consumption finished".format(document) ) + self._send_progress(100, 100, 'SUCCESS', MESSAGE_FINISHED, document.id) + return document def _store(self, text, date, mime_type): @@ -247,7 +380,6 @@ class Consumer(LoggingMixin): with open(self.path, "rb") as f: document = Document.objects.create( - correspondent=file_info.correspondent, title=(self.override_title or file_info.title)[:127], content=text, mime_type=mime_type, @@ -257,12 +389,6 @@ class Consumer(LoggingMixin): storage_type=storage_type ) - relevant_tags = set(file_info.tags) - if relevant_tags: - tag_names = ", ".join([t.name for t in relevant_tags]) - self.log("debug", "Tagging with {}".format(tag_names)) - document.tags.add(*relevant_tags) - self.apply_overrides(document) document.save() diff --git a/src/documents/file_handling.py b/src/documents/file_handling.py index c5efc33e4..d7a42db24 100644 --- a/src/documents/file_handling.py +++ b/src/documents/file_handling.py @@ -8,6 +8,15 @@ from django.conf import settings from django.template.defaultfilters import slugify +logger = logging.getLogger("paperless.filehandling") + + +class defaultdictNoStr(defaultdict): + + def __str__(self): + raise ValueError("Don't use {tags} directly.") + + def create_source_path_directory(source_path): os.makedirs(os.path.dirname(source_path), exist_ok=True) @@ -70,12 +79,40 @@ def many_to_dictionary(field): return mydictionary -def generate_unique_filename(doc, root): +def generate_unique_filename(doc, + archive_filename=False): + """ + Generates a unique filename for doc in settings.ORIGINALS_DIR. + + The returned filename is guaranteed to be either the current filename + of the document if unchanged, or a new filename that does not correspondent + to any existing files. The function will append _01, _02, etc to the + filename before the extension to avoid conflicts. + + If archive_filename is True, return a unique archive filename instead. + + """ + if archive_filename: + old_filename = doc.archive_filename + root = settings.ARCHIVE_DIR + else: + old_filename = doc.filename + root = settings.ORIGINALS_DIR + + # If generating archive filenames, try to make a name that is similar to + # the original filename first. + + if archive_filename and doc.filename: + new_filename = os.path.splitext(doc.filename)[0] + ".pdf" + if new_filename == old_filename or not os.path.exists(os.path.join(root, new_filename)): # NOQA: E501 + return new_filename + counter = 0 while True: - new_filename = generate_filename(doc, counter) - if new_filename == doc.filename: + new_filename = generate_filename( + doc, counter, archive_filename=archive_filename) + if new_filename == old_filename: # still the same as before. return new_filename @@ -85,13 +122,20 @@ def generate_unique_filename(doc, root): return new_filename -def generate_filename(doc, counter=0): +def generate_filename(doc, counter=0, append_gpg=True, archive_filename=False): path = "" try: if settings.PAPERLESS_FILENAME_FORMAT is not None: - tags = defaultdict(lambda: slugify(None), - many_to_dictionary(doc.tags)) + tags = defaultdictNoStr(lambda: slugify(None), + many_to_dictionary(doc.tags)) + + tag_list = pathvalidate.sanitize_filename( + ",".join(sorted( + [tag.name for tag in doc.tags.all()] + )), + replacement_text="-" + ) if doc.correspondent: correspondent = pathvalidate.sanitize_filename( @@ -107,6 +151,11 @@ def generate_filename(doc, counter=0): else: document_type = "none" + if doc.archive_serial_number: + asn = str(doc.archive_serial_number) + else: + asn = "none" + path = settings.PAPERLESS_FILENAME_FORMAT.format( title=pathvalidate.sanitize_filename( doc.title, replacement_text="-"), @@ -114,32 +163,35 @@ def generate_filename(doc, counter=0): document_type=document_type, created=datetime.date.isoformat(doc.created), created_year=doc.created.year if doc.created else "none", - created_month=doc.created.month if doc.created else "none", - created_day=doc.created.day if doc.created else "none", + created_month=f"{doc.created.month:02}" if doc.created else "none", # NOQA: E501 + created_day=f"{doc.created.day:02}" if doc.created else "none", added=datetime.date.isoformat(doc.added), added_year=doc.added.year if doc.added else "none", - added_month=doc.added.month if doc.added else "none", - added_day=doc.added.day if doc.added else "none", + added_month=f"{doc.added.month:02}" if doc.added else "none", + added_day=f"{doc.added.day:02}" if doc.added else "none", + asn=asn, tags=tags, - ) + tag_list=tag_list + ).strip() + + path = path.strip(os.sep) + except (ValueError, KeyError, IndexError): - logging.getLogger(__name__).warning( + logger.warning( f"Invalid PAPERLESS_FILENAME_FORMAT: " f"{settings.PAPERLESS_FILENAME_FORMAT}, falling back to default") counter_str = f"_{counter:02}" if counter else "" + + filetype_str = ".pdf" if archive_filename else doc.file_type + if len(path) > 0: - filename = f"{path}{counter_str}{doc.file_type}" + filename = f"{path}{counter_str}{filetype_str}" else: - filename = f"{doc.pk:07}{counter_str}{doc.file_type}" + filename = f"{doc.pk:07}{counter_str}{filetype_str}" # Append .gpg for encrypted files - if doc.storage_type == doc.STORAGE_TYPE_GPG: + if append_gpg and doc.storage_type == doc.STORAGE_TYPE_GPG: filename += ".gpg" return filename - - -def archive_name_from_filename(filename): - - return os.path.splitext(filename)[0] + ".pdf" diff --git a/src/documents/filters.py b/src/documents/filters.py old mode 100755 new mode 100644 index b3c92eba3..a1ad94b1e --- a/src/documents/filters.py +++ b/src/documents/filters.py @@ -1,10 +1,11 @@ +from django.db.models import Q from django_filters.rest_framework import BooleanFilter, FilterSet, Filter from .models import Correspondent, Document, Tag, DocumentType, Log CHAR_KWARGS = ["istartswith", "iendswith", "icontains", "iexact"] ID_KWARGS = ["in", "exact"] -INT_KWARGS = ["exact", "gt", "gte", "lt", "lte"] +INT_KWARGS = ["exact", "gt", "gte", "lt", "lte", "isnull"] DATE_KWARGS = ["year", "month", "day", "date__gt", "gt", "date__lt", "lt"] @@ -70,6 +71,16 @@ class InboxFilter(Filter): return qs +class TitleContentFilter(Filter): + + def filter(self, qs, value): + if value: + return qs.filter(Q(title__icontains=value) | + Q(content__icontains=value)) + else: + return qs + + class DocumentFilterSet(FilterSet): is_tagged = BooleanFilter( @@ -85,6 +96,8 @@ class DocumentFilterSet(FilterSet): is_in_inbox = InboxFilter() + title_content = TitleContentFilter() + class Meta: model = Document fields = { @@ -98,12 +111,14 @@ class DocumentFilterSet(FilterSet): "added": DATE_KWARGS, "modified": DATE_KWARGS, + "correspondent": ["isnull"], "correspondent__id": ID_KWARGS, "correspondent__name": CHAR_KWARGS, "tags__id": ID_KWARGS, "tags__name": CHAR_KWARGS, + "document_type": ["isnull"], "document_type__id": ID_KWARGS, "document_type__name": CHAR_KWARGS, diff --git a/src/documents/index.py b/src/documents/index.py index 53bf34542..cb302da45 100644 --- a/src/documents/index.py +++ b/src/documents/index.py @@ -2,70 +2,70 @@ import logging import os from contextlib import contextmanager +import math +from dateutil.parser import isoparse from django.conf import settings -from whoosh import highlight -from whoosh.fields import Schema, TEXT, NUMERIC, KEYWORD, DATETIME -from whoosh.highlight import Formatter, get_text +from whoosh import highlight, classify, query +from whoosh.fields import Schema, TEXT, NUMERIC, KEYWORD, DATETIME, BOOLEAN +from whoosh.highlight import HtmlFormatter from whoosh.index import create_in, exists_in, open_dir from whoosh.qparser import MultifieldParser from whoosh.qparser.dateparse import DateParserPlugin +from whoosh.searching import ResultsPage, Searcher from whoosh.writing import AsyncWriter +from documents.models import Document -logger = logging.getLogger(__name__) - - -class JsonFormatter(Formatter): - def __init__(self): - self.seen = {} - - def format_token(self, text, token, replace=False): - seen = self.seen - ttext = self._text(get_text(text, token, replace)) - if ttext in seen: - termnum = seen[ttext] - else: - termnum = len(seen) - seen[ttext] = termnum - - return {'text': ttext, 'term': termnum} - - def format_fragment(self, fragment, replace=False): - output = [] - index = fragment.startchar - text = fragment.text - - for t in fragment.matches: - if t.startchar is None: - continue - if t.startchar < index: - continue - if t.startchar > index: - output.append({'text': text[index:t.startchar]}) - output.append(self.format_token(text, t, replace)) - index = t.endchar - if index < fragment.endchar: - output.append({'text': text[index:fragment.endchar]}) - return output - - def format(self, fragments, replace=False): - output = [] - for fragment in fragments: - output.append(self.format_fragment(fragment, replace=replace)) - return output +logger = logging.getLogger("paperless.index") def get_schema(): return Schema( - id=NUMERIC(stored=True, unique=True, numtype=int), - title=TEXT(stored=True), + id=NUMERIC( + stored=True, + unique=True + ), + title=TEXT( + sortable=True + ), content=TEXT(), - correspondent=TEXT(stored=True), - tag=KEYWORD(stored=True, commas=True, scorable=True, lowercase=True), - type=TEXT(stored=True), - created=DATETIME(stored=True, sortable=True), - modified=DATETIME(stored=True, sortable=True), - added=DATETIME(stored=True, sortable=True), + asn=NUMERIC( + sortable=True + ), + + correspondent=TEXT( + sortable=True + ), + correspondent_id=NUMERIC(), + has_correspondent=BOOLEAN(), + + tag=KEYWORD( + commas=True, + scorable=True, + lowercase=True + ), + tag_id=KEYWORD( + commas=True, + scorable=True + ), + has_tag=BOOLEAN(), + + type=TEXT( + sortable=True + ), + type_id=NUMERIC(), + has_type=BOOLEAN(), + + created=DATETIME( + sortable=True + ), + modified=DATETIME( + sortable=True + ), + added=DATETIME( + sortable=True + ), + ) @@ -73,76 +73,225 @@ def open_index(recreate=False): try: if exists_in(settings.INDEX_DIR) and not recreate: return open_dir(settings.INDEX_DIR, schema=get_schema()) - except Exception as e: - logger.error(f"Error while opening the index: {e}, recreating.") + except Exception: + logger.exception(f"Error while opening the index, recreating.") if not os.path.isdir(settings.INDEX_DIR): os.makedirs(settings.INDEX_DIR, exist_ok=True) return create_in(settings.INDEX_DIR, get_schema()) +@contextmanager +def open_index_writer(optimize=False): + writer = AsyncWriter(open_index()) + + try: + yield writer + except Exception as e: + logger.exception(str(e)) + writer.cancel() + finally: + writer.commit(optimize=optimize) + + +@contextmanager +def open_index_searcher(): + searcher = open_index().searcher() + + try: + yield searcher + finally: + searcher.close() + + def update_document(writer, doc): - # TODO: this line caused many issues all around, since: - # We need to make sure that this method does not get called with - # deserialized documents (i.e, document objects that don't come from - # Django's ORM interfaces directly. - logger.debug("Indexing {}...".format(doc)) tags = ",".join([t.name for t in doc.tags.all()]) + tags_ids = ",".join([str(t.id) for t in doc.tags.all()]) writer.update_document( id=doc.pk, title=doc.title, content=doc.content, correspondent=doc.correspondent.name if doc.correspondent else None, + correspondent_id=doc.correspondent.id if doc.correspondent else None, + has_correspondent=doc.correspondent is not None, tag=tags if tags else None, + tag_id=tags_ids if tags_ids else None, + has_tag=len(tags) > 0, type=doc.document_type.name if doc.document_type else None, + type_id=doc.document_type.id if doc.document_type else None, + has_type=doc.document_type is not None, created=doc.created, added=doc.added, + asn=doc.archive_serial_number, modified=doc.modified, ) def remove_document(writer, doc): - # TODO: see above. - logger.debug("Removing {} from index...".format(doc)) - writer.delete_by_term('id', doc.pk) + remove_document_by_id(writer, doc.pk) + + +def remove_document_by_id(writer, doc_id): + writer.delete_by_term('id', doc_id) def add_or_update_document(document): - ix = open_index() - with AsyncWriter(ix) as writer: + with open_index_writer() as writer: update_document(writer, document) def remove_document_from_index(document): - ix = open_index() - with AsyncWriter(ix) as writer: + with open_index_writer() as writer: remove_document(writer, document) -@contextmanager -def query_page(ix, querystring, page): - searcher = ix.searcher() - try: +class DelayedQuery: + + def _get_query(self): + raise NotImplementedError() + + def _get_query_filter(self): + criterias = [] + for k, v in self.query_params.items(): + if k == 'correspondent__id': + criterias.append(query.Term('correspondent_id', v)) + elif k == 'tags__id__all': + for tag_id in v.split(","): + criterias.append(query.Term('tag_id', tag_id)) + elif k == 'document_type__id': + criterias.append(query.Term('type_id', v)) + elif k == 'correspondent__isnull': + criterias.append(query.Term("has_correspondent", v == "false")) + elif k == 'is_tagged': + criterias.append(query.Term("has_tag", v == "true")) + elif k == 'document_type__isnull': + criterias.append(query.Term("has_type", v == "false")) + elif k == 'created__date__lt': + criterias.append( + query.DateRange("created", start=None, end=isoparse(v))) + elif k == 'created__date__gt': + criterias.append( + query.DateRange("created", start=isoparse(v), end=None)) + elif k == 'added__date__gt': + criterias.append( + query.DateRange("added", start=isoparse(v), end=None)) + elif k == 'added__date__lt': + criterias.append( + query.DateRange("added", start=None, end=isoparse(v))) + if len(criterias) > 0: + return query.And(criterias) + else: + return None + + def _get_query_sortedby(self): + if 'ordering' not in self.query_params: + return None, False + + field: str = self.query_params['ordering'] + + sort_fields_map = { + "created": "created", + "modified": "modified", + "added": "added", + "title": "title", + "correspondent__name": "correspondent", + "document_type__name": "type", + "archive_serial_number": "asn" + } + + if field.startswith('-'): + field = field[1:] + reverse = True + else: + reverse = False + + if field not in sort_fields_map: + return None, False + else: + return sort_fields_map[field], reverse + + def __init__(self, searcher: Searcher, query_params, page_size): + self.searcher = searcher + self.query_params = query_params + self.page_size = page_size + self.saved_results = dict() + self.first_score = None + + def __len__(self): + page = self[0:1] + return len(page) + + def __getitem__(self, item): + if item.start in self.saved_results: + return self.saved_results[item.start] + + q, mask = self._get_query() + sortedby, reverse = self._get_query_sortedby() + + page: ResultsPage = self.searcher.search_page( + q, + mask=mask, + filter=self._get_query_filter(), + pagenum=math.floor(item.start / self.page_size) + 1, + pagelen=self.page_size, + sortedby=sortedby, + reverse=reverse + ) + page.results.fragmenter = highlight.ContextFragmenter( + surround=50) + page.results.formatter = HtmlFormatter(tagname="span", between=" ... ") + + if (not self.first_score and + len(page.results) > 0 and + sortedby is None): + self.first_score = page.results[0].score + + page.results.top_n = list(map( + lambda hit: ( + (hit[0] / self.first_score) if self.first_score else None, + hit[1] + ), + page.results.top_n + )) + + self.saved_results[item.start] = page + + return page + + +class DelayedFullTextQuery(DelayedQuery): + + def _get_query(self): + q_str = self.query_params['query'] qp = MultifieldParser( ["content", "title", "correspondent", "tag", "type"], - ix.schema) + self.searcher.ixreader.schema) qp.add_plugin(DateParserPlugin()) + q = qp.parse(q_str) - q = qp.parse(querystring) - result_page = searcher.search_page(q, page) - result_page.results.fragmenter = highlight.ContextFragmenter( - surround=50) - result_page.results.formatter = JsonFormatter() - - corrected = searcher.correct_query(q, querystring) + corrected = self.searcher.correct_query(q, q_str) if corrected.query != q: corrected_query = corrected.string - else: - corrected_query = None - yield result_page, corrected_query - finally: - searcher.close() + return q, None + + +class DelayedMoreLikeThisQuery(DelayedQuery): + + def _get_query(self): + more_like_doc_id = int(self.query_params['more_like_id']) + content = Document.objects.get(id=more_like_doc_id).content + + docnum = self.searcher.document_number(id=more_like_doc_id) + kts = self.searcher.key_terms_from_text( + 'content', content, numterms=20, + model=classify.Bo1Model, normalize=False) + q = query.Or( + [query.Term('content', word, boost=weight) + for word, weight in kts]) + mask = {docnum} + + return q, mask def autocomplete(ix, term, limit=10): diff --git a/src/documents/loggers.py b/src/documents/loggers.py index 863bc0c34..90f6770e0 100644 --- a/src/documents/loggers.py +++ b/src/documents/loggers.py @@ -4,33 +4,24 @@ import uuid from django.conf import settings -class PaperlessHandler(logging.Handler): - def emit(self, record): - if settings.DISABLE_DBHANDLER: - return - - # We have to do the import here or Django will barf when it tries to - # load this because the apps aren't loaded at that point - from .models import Log - - kwargs = {"message": record.msg, "level": record.levelno} - - if hasattr(record, "group"): - kwargs["group"] = record.group - - Log.objects.create(**kwargs) - - class LoggingMixin: logging_group = None + logging_name = None + def renew_logging_group(self): self.logging_group = uuid.uuid4() def log(self, level, message, **kwargs): - target = ".".join([self.__class__.__module__, self.__class__.__name__]) - logger = logging.getLogger(target) + if self.logging_name: + logger = logging.getLogger(self.logging_name) + else: + name = ".".join([ + self.__class__.__module__, + self.__class__.__name__ + ]) + logger = logging.getLogger(name) getattr(logger, level)(message, extra={ "group": self.logging_group diff --git a/src/documents/management/commands/decrypt_documents.py b/src/documents/management/commands/decrypt_documents.py index 918f1a175..8f5c2e123 100644 --- a/src/documents/management/commands/decrypt_documents.py +++ b/src/documents/management/commands/decrypt_documents.py @@ -2,7 +2,6 @@ import os from django.conf import settings from django.core.management.base import BaseCommand, CommandError -from termcolor import colored as coloured from documents.models import Document from paperless.db import GnuPG @@ -26,16 +25,14 @@ class Command(BaseCommand): def handle(self, *args, **options): try: - print(coloured( + print( "\n\nWARNING: This script is going to work directly on your " "document originals, so\nWARNING: you probably shouldn't run " "this unless you've got a recent backup\nWARNING: handy. It " "*should* work without a hitch, but be safe and backup your\n" "WARNING: stuff first.\n\nHit Ctrl+C to exit now, or Enter to " - "continue.\n\n", - "yellow", - attrs=("bold",) - )) + "continue.\n\n" + ) __ = input() except KeyboardInterrupt: return @@ -57,8 +54,8 @@ class Command(BaseCommand): for document in encrypted_files: - print(coloured("Decrypting {}".format( - document).encode('utf-8'), "green")) + print("Decrypting {}".format( + document).encode('utf-8')) old_paths = [document.source_path, document.thumbnail_path] diff --git a/src/documents/management/commands/document_archiver.py b/src/documents/management/commands/document_archiver.py index 7b9a123d9..96ddebe77 100644 --- a/src/documents/management/commands/document_archiver.py +++ b/src/documents/management/commands/document_archiver.py @@ -11,16 +11,17 @@ from django import db from django.conf import settings from django.core.management.base import BaseCommand from django.db import transaction +from filelock import FileLock from whoosh.writing import AsyncWriter from documents.models import Document from ... import index -from ...file_handling import create_source_path_directory -from ...mixins import Renderable +from ...file_handling import create_source_path_directory, \ + generate_unique_filename from ...parsers import get_parser_class_for_mime_type -logger = logging.getLogger(__name__) +logger = logging.getLogger("paperless.management.archiver") def handle_document(document_id): @@ -30,36 +31,57 @@ def handle_document(document_id): parser_class = get_parser_class_for_mime_type(mime_type) + if not parser_class: + logger.error(f"No parser found for mime type {mime_type}, cannot " + f"archive document {document} (ID: {document_id})") + return + parser = parser_class(logging_group=uuid.uuid4()) try: - parser.parse(document.source_path, mime_type) + parser.parse( + document.source_path, + mime_type, + document.get_public_filename()) + + thumbnail = parser.get_optimised_thumbnail( + document.source_path, + mime_type, + document.get_public_filename() + ) if parser.get_archive_path(): with transaction.atomic(): with open(parser.get_archive_path(), 'rb') as f: checksum = hashlib.md5(f.read()).hexdigest() - # i'm going to save first so that in case the file move + # I'm going to save first so that in case the file move # fails, the database is rolled back. - # we also don't use save() since that triggers the filehandling + # We also don't use save() since that triggers the filehandling # logic, and we don't want that yet (file not yet in place) + document.archive_filename = generate_unique_filename( + document, archive_filename=True) Document.objects.filter(pk=document.pk).update( archive_checksum=checksum, - content=parser.get_text() + content=parser.get_text(), + archive_filename=document.archive_filename ) - create_source_path_directory(document.archive_path) - shutil.move(parser.get_archive_path(), document.archive_path) + with FileLock(settings.MEDIA_LOCK): + create_source_path_directory(document.archive_path) + shutil.move(parser.get_archive_path(), + document.archive_path) + shutil.move(thumbnail, document.thumbnail_path) - with AsyncWriter(index.open_index()) as writer: - index.update_document(writer, document) + with index.open_index_writer() as writer: + index.update_document(writer, document) except Exception as e: - logger.error(f"Error while parsing document {document}: {str(e)}") + logger.exception(f"Error while parsing document {document} " + f"(ID: {document_id})") finally: parser.cleanup() -class Command(Renderable, BaseCommand): +class Command(BaseCommand): help = """ Using the current classification model, assigns correspondents, tags @@ -68,10 +90,6 @@ class Command(Renderable, BaseCommand): modified) after their initial import. """.replace(" ", "") - def __init__(self, *args, **kwargs): - self.verbosity = 0 - BaseCommand.__init__(self, *args, **kwargs) - def add_arguments(self, parser): parser.add_argument( "-f", "--overwrite", @@ -88,6 +106,12 @@ class Command(Renderable, BaseCommand): help="Specify the ID of a document, and this command will only " "run on this specific document." ) + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) def handle(self, *args, **options): @@ -103,7 +127,7 @@ class Command(Renderable, BaseCommand): document_ids = list(map( lambda doc: doc.id, filter( - lambda d: overwrite or not d.archive_checksum, + lambda d: overwrite or not d.has_archive_version, documents ) )) @@ -122,7 +146,8 @@ class Command(Renderable, BaseCommand): handle_document, document_ids ), - total=len(document_ids) + total=len(document_ids), + disable=options['no_progress_bar'] )) except KeyboardInterrupt: print("Aborting...") diff --git a/src/documents/management/commands/document_consumer.py b/src/documents/management/commands/document_consumer.py index 8ac60aa6d..eb8c57c84 100644 --- a/src/documents/management/commands/document_consumer.py +++ b/src/documents/management/commands/document_consumer.py @@ -1,11 +1,11 @@ import logging import os -from pathlib import Path +from pathlib import Path, PurePath +from threading import Thread from time import sleep from django.conf import settings from django.core.management.base import BaseCommand, CommandError -from django.utils.text import slugify from django_q.tasks import async_task from watchdog.events import FileSystemEventHandler from watchdog.observers.polling import PollingObserver @@ -18,11 +18,11 @@ try: except ImportError: INotify = flags = None -logger = logging.getLogger(__name__) +logger = logging.getLogger("paperless.management.consumer") def _tags_from_path(filepath): - """Walk up the directory tree from filepath to CONSUMPTION_DIr + """Walk up the directory tree from filepath to CONSUMPTION_DIR and get or create Tag IDs for every directory. """ tag_ids = set() @@ -36,8 +36,15 @@ def _tags_from_path(filepath): return tag_ids +def _is_ignored(filepath: str) -> bool: + filepath_relative = PurePath(filepath).relative_to( + settings.CONSUMPTION_DIR) + return any( + filepath_relative.match(p) for p in settings.CONSUMER_IGNORE_PATTERNS) + + def _consume(filepath): - if os.path.isdir(filepath): + if os.path.isdir(filepath) or _is_ignored(filepath): return if not os.path.isfile(filepath): @@ -46,7 +53,7 @@ def _consume(filepath): return if not is_file_ext_supported(os.path.splitext(filepath)[1]): - logger.debug( + logger.warning( f"Not consuming file {filepath}: Unknown file extension.") return @@ -55,10 +62,10 @@ def _consume(filepath): if settings.CONSUMER_SUBDIRS_AS_TAGS: tag_ids = _tags_from_path(filepath) except Exception as e: - logger.error( - "Error creating tags from path: {}".format(e)) + logger.exception("Error creating tags from path") try: + logger.info(f"Adding {filepath} to the task queue.") async_task("documents.tasks.consume_file", filepath, override_tag_ids=tag_ids if tag_ids else None, @@ -67,14 +74,17 @@ def _consume(filepath): # Catch all so that the consumer won't crash. # This is also what the test case is listening for to check for # errors. - logger.error( - "Error while consuming document: {}".format(e)) + logger.exception("Error while consuming document") -def _consume_wait_unmodified(file, num_tries=20, wait_time=1): +def _consume_wait_unmodified(file): + if _is_ignored(file): + return + + logger.debug(f"Waiting for file {file} to remain unmodified") mtime = -1 current_try = 0 - while current_try < num_tries: + while current_try < settings.CONSUMER_POLLING_RETRY_COUNT: try: new_mtime = os.stat(file).st_mtime except FileNotFoundError: @@ -85,7 +95,7 @@ def _consume_wait_unmodified(file, num_tries=20, wait_time=1): _consume(file) return mtime = new_mtime - sleep(wait_time) + sleep(settings.CONSUMER_POLLING_DELAY) current_try += 1 logger.error(f"Timeout while waiting on file {file} to remain unmodified.") @@ -94,10 +104,14 @@ def _consume_wait_unmodified(file, num_tries=20, wait_time=1): class Handler(FileSystemEventHandler): def on_created(self, event): - _consume_wait_unmodified(event.src_path) + Thread( + target=_consume_wait_unmodified, args=(event.src_path,) + ).start() def on_moved(self, event): - _consume_wait_unmodified(event.dest_path) + Thread( + target=_consume_wait_unmodified, args=(event.dest_path,) + ).start() class Command(BaseCommand): @@ -109,12 +123,7 @@ class Command(BaseCommand): # This is here primarily for the tests and is irrelevant in production. stop_flag = False - def __init__(self, *args, **kwargs): - - self.logger = logging.getLogger(__name__) - - BaseCommand.__init__(self, *args, **kwargs) - self.observer = None + observer = None def add_arguments(self, parser): parser.add_argument( @@ -162,7 +171,7 @@ class Command(BaseCommand): logger.debug("Consumer exiting.") def handle_polling(self, directory, recursive): - logging.getLogger(__name__).info( + logger.info( f"Polling directory for changes: {directory}") self.observer = PollingObserver(timeout=settings.CONSUMER_POLLING) self.observer.schedule(Handler(), directory, recursive=recursive) @@ -177,7 +186,7 @@ class Command(BaseCommand): self.observer.join() def handle_inotify(self, directory, recursive): - logging.getLogger(__name__).info( + logger.info( f"Using inotify to watch directory for changes: {directory}") inotify = INotify() diff --git a/src/documents/management/commands/document_create_classifier.py b/src/documents/management/commands/document_create_classifier.py old mode 100755 new mode 100644 index fbfb7f7e6..a4ede88b5 --- a/src/documents/management/commands/document_create_classifier.py +++ b/src/documents/management/commands/document_create_classifier.py @@ -1,10 +1,9 @@ from django.core.management.base import BaseCommand -from ...mixins import Renderable from ...tasks import train_classifier -class Command(Renderable, BaseCommand): +class Command(BaseCommand): help = """ Trains the classifier on your data and saves the resulting models to a diff --git a/src/documents/management/commands/document_exporter.py b/src/documents/management/commands/document_exporter.py index a7a17f124..217877397 100644 --- a/src/documents/management/commands/document_exporter.py +++ b/src/documents/management/commands/document_exporter.py @@ -1,19 +1,27 @@ +import hashlib import json import os import shutil import time +import tqdm +from django.conf import settings +from django.contrib.auth.models import User from django.core import serializers from django.core.management.base import BaseCommand, CommandError +from django.db import transaction +from filelock import FileLock -from documents.models import Document, Correspondent, Tag, DocumentType +from documents.models import Document, Correspondent, Tag, DocumentType, \ + SavedView, SavedViewFilterRule from documents.settings import EXPORTER_FILE_NAME, EXPORTER_THUMBNAIL_NAME, \ EXPORTER_ARCHIVE_NAME from paperless.db import GnuPG -from ...mixins import Renderable +from paperless_mail.models import MailAccount, MailRule +from ...file_handling import generate_filename, delete_empty_directories -class Command(Renderable, BaseCommand): +class Command(BaseCommand): help = """ Decrypt and rename all files in our collection into a given target @@ -24,13 +32,53 @@ class Command(Renderable, BaseCommand): def add_arguments(self, parser): parser.add_argument("target") + parser.add_argument( + "-c", "--compare-checksums", + default=False, + action="store_true", + help="Compare file checksums when determining whether to export " + "a file or not. If not specified, file size and time " + "modified is used instead." + ) + + parser.add_argument( + "-f", "--use-filename-format", + default=False, + action="store_true", + help="Use PAPERLESS_FILENAME_FORMAT for storing files in the " + "export directory, if configured." + ) + + parser.add_argument( + "-d", "--delete", + default=False, + action="store_true", + help="After exporting, delete files in the export directory that " + "do not belong to the current export, such as files from " + "deleted documents." + ) + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) + def __init__(self, *args, **kwargs): BaseCommand.__init__(self, *args, **kwargs) self.target = None + self.files_in_export_dir = [] + self.exported_files = [] + self.compare_checksums = False + self.use_filename_format = False + self.delete = False def handle(self, *args, **options): self.target = options["target"] + self.compare_checksums = options['compare_checksums'] + self.use_filename_format = options['use_filename_format'] + self.delete = options['delete'] if not os.path.exists(self.target): raise CommandError("That path doesn't exist") @@ -38,83 +86,166 @@ class Command(Renderable, BaseCommand): if not os.access(self.target, os.W_OK): raise CommandError("That path doesn't appear to be writable") - if os.listdir(self.target): - raise CommandError("That directory is not empty.") + with FileLock(settings.MEDIA_LOCK): + self.dump(options['no_progress_bar']) - self.dump() + def dump(self, progress_bar_disable=False): + # 1. Take a snapshot of what files exist in the current export folder + for root, dirs, files in os.walk(self.target): + self.files_in_export_dir.extend( + map(lambda f: os.path.abspath(os.path.join(root, f)), files) + ) - def dump(self): + # 2. Create manifest, containing all correspondents, types, tags and + # documents + with transaction.atomic(): + manifest = json.loads( + serializers.serialize("json", Correspondent.objects.all())) - documents = Document.objects.all() - document_map = {d.pk: d for d in documents} - manifest = json.loads(serializers.serialize("json", documents)) + manifest += json.loads(serializers.serialize( + "json", Tag.objects.all())) - for index, document_dict in enumerate(manifest): + manifest += json.loads(serializers.serialize( + "json", DocumentType.objects.all())) - # Force output to unencrypted as that will be the current state. - # The importer will make the decision to encrypt or not. - manifest[index]["fields"]["storage_type"] = Document.STORAGE_TYPE_UNENCRYPTED # NOQA: E501 + documents = Document.objects.order_by("id") + document_map = {d.pk: d for d in documents} + document_manifest = json.loads( + serializers.serialize("json", documents)) + manifest += document_manifest + + manifest += json.loads(serializers.serialize( + "json", MailAccount.objects.all())) + + manifest += json.loads(serializers.serialize( + "json", MailRule.objects.all())) + + manifest += json.loads(serializers.serialize( + "json", SavedView.objects.all())) + + manifest += json.loads(serializers.serialize( + "json", SavedViewFilterRule.objects.all())) + + manifest += json.loads(serializers.serialize( + "json", User.objects.all())) + + # 3. Export files from each document + for index, document_dict in tqdm.tqdm( + enumerate(document_manifest), + total=len(document_manifest), + disable=progress_bar_disable + ): + # 3.1. store files unencrypted + document_dict["fields"]["storage_type"] = Document.STORAGE_TYPE_UNENCRYPTED # NOQA: E501 document = document_map[document_dict["pk"]] - print(f"Exporting: {document}") - + # 3.2. generate a unique filename filename_counter = 0 while True: - original_name = document.get_public_filename( - counter=filename_counter) - original_target = os.path.join(self.target, original_name) + if self.use_filename_format: + base_name = generate_filename( + document, counter=filename_counter, + append_gpg=False) + else: + base_name = document.get_public_filename( + counter=filename_counter) - if not os.path.exists(original_target): + if base_name not in self.exported_files: + self.exported_files.append(base_name) break else: filename_counter += 1 - thumbnail_name = original_name + "-thumbnail.png" - thumbnail_target = os.path.join(self.target, thumbnail_name) - + # 3.3. write filenames into manifest + original_name = base_name + original_target = os.path.join(self.target, original_name) document_dict[EXPORTER_FILE_NAME] = original_name + + thumbnail_name = base_name + "-thumbnail.png" + thumbnail_target = os.path.join(self.target, thumbnail_name) document_dict[EXPORTER_THUMBNAIL_NAME] = thumbnail_name - if os.path.exists(document.archive_path): - archive_name = document.get_public_filename( - archive=True, counter=filename_counter, suffix="_archive") + if document.has_archive_version: + archive_name = base_name + "-archive.pdf" archive_target = os.path.join(self.target, archive_name) document_dict[EXPORTER_ARCHIVE_NAME] = archive_name else: archive_target = None + # 3.4. write files to target folder t = int(time.mktime(document.created.timetuple())) if document.storage_type == Document.STORAGE_TYPE_GPG: + os.makedirs(os.path.dirname(original_target), exist_ok=True) with open(original_target, "wb") as f: f.write(GnuPG.decrypted(document.source_file)) os.utime(original_target, times=(t, t)) + os.makedirs(os.path.dirname(thumbnail_target), exist_ok=True) with open(thumbnail_target, "wb") as f: f.write(GnuPG.decrypted(document.thumbnail_file)) os.utime(thumbnail_target, times=(t, t)) if archive_target: + os.makedirs(os.path.dirname(archive_target), exist_ok=True) with open(archive_target, "wb") as f: f.write(GnuPG.decrypted(document.archive_path)) os.utime(archive_target, times=(t, t)) else: + self.check_and_copy(document.source_path, + document.checksum, + original_target) - shutil.copy(document.source_path, original_target) - shutil.copy(document.thumbnail_path, thumbnail_target) + self.check_and_copy(document.thumbnail_path, + None, + thumbnail_target) if archive_target: - shutil.copy(document.archive_path, archive_target) + self.check_and_copy(document.archive_path, + document.archive_checksum, + archive_target) - manifest += json.loads( - serializers.serialize("json", Correspondent.objects.all())) + # 4. write manifest to target forlder + manifest_path = os.path.abspath( + os.path.join(self.target, "manifest.json")) - manifest += json.loads(serializers.serialize( - "json", Tag.objects.all())) - - manifest += json.loads(serializers.serialize( - "json", DocumentType.objects.all())) - - with open(os.path.join(self.target, "manifest.json"), "w") as f: + with open(manifest_path, "w") as f: json.dump(manifest, f, indent=2) + + if self.delete: + # 5. Remove files which we did not explicitly export in this run + + if manifest_path in self.files_in_export_dir: + self.files_in_export_dir.remove(manifest_path) + + for f in self.files_in_export_dir: + os.remove(f) + + delete_empty_directories(os.path.abspath(os.path.dirname(f)), + os.path.abspath(self.target)) + + def check_and_copy(self, source, source_checksum, target): + if os.path.abspath(target) in self.files_in_export_dir: + self.files_in_export_dir.remove(os.path.abspath(target)) + + perform_copy = False + + if os.path.exists(target): + source_stat = os.stat(source) + target_stat = os.stat(target) + if self.compare_checksums and source_checksum: + with open(target, "rb") as f: + target_checksum = hashlib.md5(f.read()).hexdigest() + perform_copy = target_checksum != source_checksum + elif source_stat.st_mtime != target_stat.st_mtime: + perform_copy = True + elif source_stat.st_size != target_stat.st_size: + perform_copy = True + else: + # Copy if it does not exist + perform_copy = True + + if perform_copy: + os.makedirs(os.path.dirname(target), exist_ok=True) + shutil.copy2(source, target) diff --git a/src/documents/management/commands/document_importer.py b/src/documents/management/commands/document_importer.py index 70d05d98b..4606b91fe 100644 --- a/src/documents/management/commands/document_importer.py +++ b/src/documents/management/commands/document_importer.py @@ -1,21 +1,33 @@ import json +import logging import os import shutil +from contextlib import contextmanager +import tqdm from django.conf import settings from django.core.management import call_command from django.core.management.base import BaseCommand, CommandError +from django.db.models.signals import post_save, m2m_changed from filelock import FileLock from documents.models import Document from documents.settings import EXPORTER_FILE_NAME, EXPORTER_THUMBNAIL_NAME, \ EXPORTER_ARCHIVE_NAME -from ...file_handling import create_source_path_directory, \ - generate_unique_filename -from ...mixins import Renderable +from ...file_handling import create_source_path_directory +from ...signals.handlers import update_filename_and_move_files -class Command(Renderable, BaseCommand): +@contextmanager +def disable_signal(sig, receiver, sender): + try: + sig.disconnect(receiver=receiver, sender=sender) + yield + finally: + sig.connect(receiver=receiver, sender=sender) + + +class Command(BaseCommand): help = """ Using a manifest.json file, load the data from there, and import the @@ -24,6 +36,12 @@ class Command(Renderable, BaseCommand): def add_arguments(self, parser): parser.add_argument("source") + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) def __init__(self, *args, **kwargs): BaseCommand.__init__(self, *args, **kwargs) @@ -32,6 +50,8 @@ class Command(Renderable, BaseCommand): def handle(self, *args, **options): + logging.getLogger().handlers[0].level = logging.ERROR + self.source = options["source"] if not os.path.exists(self.source): @@ -47,11 +67,19 @@ class Command(Renderable, BaseCommand): self.manifest = json.load(f) self._check_manifest() + with disable_signal(post_save, + receiver=update_filename_and_move_files, + sender=Document): + with disable_signal(m2m_changed, + receiver=update_filename_and_move_files, + sender=Document.tags.through): + # Fill up the database with whatever is in the manifest + call_command("loaddata", manifest_path) - # Fill up the database with whatever is in the manifest - call_command("loaddata", manifest_path) + self._import_files_from_manifest(options['no_progress_bar']) - self._import_files_from_manifest() + print("Updating search index...") + call_command('document_index', 'reindex') @staticmethod def _check_manifest_exists(path): @@ -89,16 +117,22 @@ class Command(Renderable, BaseCommand): f"does not appear to be in the source directory." ) - def _import_files_from_manifest(self): + def _import_files_from_manifest(self, progress_bar_disable): os.makedirs(settings.ORIGINALS_DIR, exist_ok=True) os.makedirs(settings.THUMBNAIL_DIR, exist_ok=True) os.makedirs(settings.ARCHIVE_DIR, exist_ok=True) - for record in self.manifest: + print("Copy files into paperless...") - if not record["model"] == "documents.document": - continue + manifest_documents = list(filter( + lambda r: r["model"] == "documents.document", + self.manifest)) + + for record in tqdm.tqdm( + manifest_documents, + disable=progress_bar_disable + ): document = Document.objects.get(pk=record["pk"]) @@ -117,19 +151,18 @@ class Command(Renderable, BaseCommand): document.storage_type = Document.STORAGE_TYPE_UNENCRYPTED with FileLock(settings.MEDIA_LOCK): - document.filename = generate_unique_filename( - document, settings.ORIGINALS_DIR) - if os.path.isfile(document.source_path): raise FileExistsError(document.source_path) create_source_path_directory(document.source_path) - print(f"Moving {document_path} to {document.source_path}") - shutil.copy(document_path, document.source_path) - shutil.copy(thumbnail_path, document.thumbnail_path) + shutil.copy2(document_path, document.source_path) + shutil.copy2(thumbnail_path, document.thumbnail_path) if archive_path: create_source_path_directory(document.archive_path) - shutil.copy(archive_path, document.archive_path) + # TODO: this assumes that the export is valid and + # archive_filename is present on all documents with + # archived files + shutil.copy2(archive_path, document.archive_path) document.save() diff --git a/src/documents/management/commands/document_index.py b/src/documents/management/commands/document_index.py index 7dfdbaa42..d76f48293 100644 --- a/src/documents/management/commands/document_index.py +++ b/src/documents/management/commands/document_index.py @@ -1,25 +1,25 @@ from django.core.management import BaseCommand +from django.db import transaction -from documents.mixins import Renderable from documents.tasks import index_reindex, index_optimize -class Command(Renderable, BaseCommand): +class Command(BaseCommand): help = "Manages the document index." - def __init__(self, *args, **kwargs): - self.verbosity = 0 - BaseCommand.__init__(self, *args, **kwargs) - def add_arguments(self, parser): parser.add_argument("command", choices=['reindex', 'optimize']) + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) def handle(self, *args, **options): - - self.verbosity = options["verbosity"] - - if options['command'] == 'reindex': - index_reindex() - elif options['command'] == 'optimize': - index_optimize() + with transaction.atomic(): + if options['command'] == 'reindex': + index_reindex(progress_bar_disable=options['no_progress_bar']) + elif options['command'] == 'optimize': + index_optimize() diff --git a/src/documents/management/commands/document_logs.py b/src/documents/management/commands/document_logs.py deleted file mode 100644 index 06efc3850..000000000 --- a/src/documents/management/commands/document_logs.py +++ /dev/null @@ -1,12 +0,0 @@ -from django.core.management.base import BaseCommand - -from documents.models import Log - - -class Command(BaseCommand): - - help = "A quick & dirty way to see what's in the logs" - - def handle(self, *args, **options): - for log in Log.objects.order_by("pk"): - print(log) diff --git a/src/documents/management/commands/document_renamer.py b/src/documents/management/commands/document_renamer.py index 745d2d03d..682705c2d 100644 --- a/src/documents/management/commands/document_renamer.py +++ b/src/documents/management/commands/document_renamer.py @@ -5,24 +5,28 @@ from django.core.management.base import BaseCommand from django.db.models.signals import post_save from documents.models import Document -from ...mixins import Renderable -class Command(Renderable, BaseCommand): +class Command(BaseCommand): help = """ This will rename all documents to match the latest filename format. """.replace(" ", "") - def __init__(self, *args, **kwargs): - self.verbosity = 0 - BaseCommand.__init__(self, *args, **kwargs) + def add_arguments(self, parser): + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) def handle(self, *args, **options): - self.verbosity = options["verbosity"] - logging.getLogger().handlers[0].level = logging.ERROR - for document in tqdm.tqdm(Document.objects.all()): + for document in tqdm.tqdm( + Document.objects.all(), + disable=options['no_progress_bar'] + ): post_save.send(Document, instance=document) diff --git a/src/documents/management/commands/document_retagger.py b/src/documents/management/commands/document_retagger.py old mode 100755 new mode 100644 index cf014dc6f..6636af20a --- a/src/documents/management/commands/document_retagger.py +++ b/src/documents/management/commands/document_retagger.py @@ -1,15 +1,17 @@ import logging +import tqdm from django.core.management.base import BaseCommand -from documents.classifier import DocumentClassifier, \ - IncompatibleClassifierVersionError +from documents.classifier import load_classifier from documents.models import Document -from ...mixins import Renderable from ...signals.handlers import set_correspondent, set_document_type, set_tags -class Command(Renderable, BaseCommand): +logger = logging.getLogger("paperless.management.retagger") + + +class Command(BaseCommand): help = """ Using the current classification model, assigns correspondents, tags @@ -18,10 +20,6 @@ class Command(Renderable, BaseCommand): modified) after their initial import. """.replace(" ", "") - def __init__(self, *args, **kwargs): - self.verbosity = 0 - BaseCommand.__init__(self, *args, **kwargs) - def add_arguments(self, parser): parser.add_argument( "-c", "--correspondent", @@ -59,10 +57,26 @@ class Command(Renderable, BaseCommand): "set correspondent, document and remove correspondents, types" "and tags that do not match anymore due to changed rules." ) + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) + parser.add_argument( + "--suggest", + default=False, + action="store_true", + help="Return the suggestion, don't change anything." + ) + parser.add_argument( + "--base-url", + help="The base URL to use to build the link to the documents." + ) def handle(self, *args, **options): - - self.verbosity = options["verbosity"] + # Detect if we support color + color = self.style.ERROR("test") != "test" if options["inbox_only"]: queryset = Document.objects.filter(tags__is_inbox_tag=True) @@ -70,17 +84,12 @@ class Command(Renderable, BaseCommand): queryset = Document.objects.all() documents = queryset.distinct() - classifier = DocumentClassifier() - try: - classifier.reload() - except (FileNotFoundError, IncompatibleClassifierVersionError) as e: - logging.getLogger(__name__).warning( - f"Cannot classify documents: {e}.") - classifier = None + classifier = load_classifier() - for document in documents: - logging.getLogger(__name__).info( - f"Processing document {document.title}") + for document in tqdm.tqdm( + documents, + disable=options['no_progress_bar'] + ): if options['correspondent']: set_correspondent( @@ -88,18 +97,27 @@ class Command(Renderable, BaseCommand): document=document, classifier=classifier, replace=options['overwrite'], - use_first=options['use_first']) + use_first=options['use_first'], + suggest=options['suggest'], + base_url=options['base_url'], + color=color) if options['document_type']: set_document_type(sender=None, document=document, classifier=classifier, replace=options['overwrite'], - use_first=options['use_first']) + use_first=options['use_first'], + suggest=options['suggest'], + base_url=options['base_url'], + color=color) if options['tags']: set_tags( sender=None, document=document, classifier=classifier, - replace=options['overwrite']) + replace=options['overwrite'], + suggest=options['suggest'], + base_url=options['base_url'], + color=color) diff --git a/src/documents/management/commands/document_sanity_checker.py b/src/documents/management/commands/document_sanity_checker.py new file mode 100644 index 000000000..fbbb8d600 --- /dev/null +++ b/src/documents/management/commands/document_sanity_checker.py @@ -0,0 +1,23 @@ +from django.core.management.base import BaseCommand +from documents.sanity_checker import check_sanity + + +class Command(BaseCommand): + + help = """ + This command checks your document archive for issues. + """.replace(" ", "") + + def add_arguments(self, parser): + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) + + def handle(self, *args, **options): + + messages = check_sanity(progress=not options['no_progress_bar']) + + messages.log_messages() diff --git a/src/documents/management/commands/document_thumbnails.py b/src/documents/management/commands/document_thumbnails.py new file mode 100644 index 000000000..05f059039 --- /dev/null +++ b/src/documents/management/commands/document_thumbnails.py @@ -0,0 +1,77 @@ +import logging +import multiprocessing +import shutil + +import tqdm +from django import db +from django.core.management.base import BaseCommand + +from documents.models import Document +from ...parsers import get_parser_class_for_mime_type + + +def _process_document(doc_in): + document = Document.objects.get(id=doc_in) + parser_class = get_parser_class_for_mime_type(document.mime_type) + + if parser_class: + parser = parser_class(logging_group=None) + else: + print(f"{document} No parser for mime type {document.mime_type}") + return + + try: + thumb = parser.get_optimised_thumbnail( + document.source_path, + document.mime_type, + document.get_public_filename() + ) + + shutil.move(thumb, document.thumbnail_path) + finally: + parser.cleanup() + + +class Command(BaseCommand): + + help = """ + This will regenerate the thumbnails for all documents. + """.replace(" ", "") + + def add_arguments(self, parser): + parser.add_argument( + "-d", "--document", + default=None, + type=int, + required=False, + help="Specify the ID of a document, and this command will only " + "run on this specific document." + ) + parser.add_argument( + "--no-progress-bar", + default=False, + action="store_true", + help="If set, the progress bar will not be shown" + ) + + def handle(self, *args, **options): + logging.getLogger().handlers[0].level = logging.ERROR + + if options['document']: + documents = Document.objects.filter(pk=options['document']) + else: + documents = Document.objects.all() + + ids = [doc.id for doc in documents] + + # Note to future self: this prevents django from reusing database + # conncetions between processes, which is bad and does not work + # with postgres. + db.connections.close_all() + + with multiprocessing.Pool() as pool: + list(tqdm.tqdm( + pool.imap_unordered(_process_document, ids), + total=len(ids), + disable=options['no_progress_bar'] + )) diff --git a/src/documents/management/commands/manage_superuser.py b/src/documents/management/commands/manage_superuser.py new file mode 100644 index 000000000..ef3635e52 --- /dev/null +++ b/src/documents/management/commands/manage_superuser.py @@ -0,0 +1,42 @@ +import logging +import os + +from django.contrib.auth.models import User +from django.core.management.base import BaseCommand, CommandError + + +logger = logging.getLogger("paperless.management.superuser") + + +class Command(BaseCommand): + + help = """ + Creates a Django superuser based on env variables. + """.replace(" ", "") + + def handle(self, *args, **options): + + username = os.getenv('PAPERLESS_ADMIN_USER') + if not username: + return + + mail = os.getenv('PAPERLESS_ADMIN_MAIL', 'root@localhost') + password = os.getenv('PAPERLESS_ADMIN_PASSWORD') + + # Check if user exists already, leave as is if it does + if User.objects.filter(username=username).exists(): + user: User = User.objects.get_by_natural_key(username) + user.set_password(password) + user.save() + self.stdout.write(f"Changed password of user {username}.") + elif password: + # Create superuser based on env variables + User.objects.create_superuser(username, mail, password) + self.stdout.write( + f'Created superuser "{username}" with provided password.') + else: + self.stdout.write( + f'Did not create superuser "{username}".') + self.stdout.write( + 'Make sure you specified "PAPERLESS_ADMIN_PASSWORD" in your ' + '"docker-compose.env" file.') diff --git a/src/documents/matching.py b/src/documents/matching.py index 212698ad3..a1f3896e5 100644 --- a/src/documents/matching.py +++ b/src/documents/matching.py @@ -1,53 +1,63 @@ +import logging import re -from fuzzywuzzy import fuzz from documents.models import MatchingModel, Correspondent, DocumentType, Tag -def match_correspondents(document_content, classifier): +logger = logging.getLogger("paperless.matching") + + +def log_reason(matching_model, document, reason): + class_name = type(matching_model).__name__ + logger.debug( + f"{class_name} {matching_model.name} matched on document " + f"{document} because {reason}") + + +def match_correspondents(document, classifier): if classifier: - pred_id = classifier.predict_correspondent(document_content) + pred_id = classifier.predict_correspondent(document.content) else: pred_id = None correspondents = Correspondent.objects.all() return list(filter( - lambda o: matches(o, document_content) or o.pk == pred_id, + lambda o: matches(o, document) or o.pk == pred_id, correspondents)) -def match_document_types(document_content, classifier): +def match_document_types(document, classifier): if classifier: - pred_id = classifier.predict_document_type(document_content) + pred_id = classifier.predict_document_type(document.content) else: pred_id = None document_types = DocumentType.objects.all() return list(filter( - lambda o: matches(o, document_content) or o.pk == pred_id, + lambda o: matches(o, document) or o.pk == pred_id, document_types)) -def match_tags(document_content, classifier): +def match_tags(document, classifier): if classifier: - predicted_tag_ids = classifier.predict_tags(document_content) + predicted_tag_ids = classifier.predict_tags(document.content) else: predicted_tag_ids = [] tags = Tag.objects.all() return list(filter( - lambda o: matches(o, document_content) or o.pk in predicted_tag_ids, + lambda o: matches(o, document) or o.pk in predicted_tag_ids, tags)) -def matches(matching_model, document_content): +def matches(matching_model, document): search_kwargs = {} - document_content = document_content.lower() + document_content = document.content.lower() # Check that match is not empty if matching_model.match.strip() == "": @@ -62,35 +72,73 @@ def matches(matching_model, document_content): rf"\b{word}\b", document_content, **search_kwargs) if not search_result: return False + log_reason( + matching_model, document, + f"it contains all of these words: {matching_model.match}" + ) return True elif matching_model.matching_algorithm == MatchingModel.MATCH_ANY: for word in _split_match(matching_model): if re.search(rf"\b{word}\b", document_content, **search_kwargs): + log_reason( + matching_model, document, + f"it contains this word: {word}" + ) return True return False elif matching_model.matching_algorithm == MatchingModel.MATCH_LITERAL: - return bool(re.search( - rf"\b{matching_model.match}\b", + result = bool(re.search( + rf"\b{re.escape(matching_model.match)}\b", document_content, **search_kwargs )) + if result: + log_reason( + matching_model, document, + f"it contains this string: \"{matching_model.match}\"" + ) + return result elif matching_model.matching_algorithm == MatchingModel.MATCH_REGEX: - return bool(re.search( - re.compile(matching_model.match, **search_kwargs), - document_content - )) + try: + match = re.search( + re.compile(matching_model.match, **search_kwargs), + document_content + ) + except re.error: + logger.error( + f"Error while processing regular expression " + f"{matching_model.match}" + ) + return False + if match: + log_reason( + matching_model, document, + f"the string {match.group()} matches the regular expression " + f"{matching_model.match}" + ) + return bool(match) elif matching_model.matching_algorithm == MatchingModel.MATCH_FUZZY: + from fuzzywuzzy import fuzz + match = re.sub(r'[^\w\s]', '', matching_model.match) text = re.sub(r'[^\w\s]', '', document_content) if matching_model.is_insensitive: match = match.lower() text = text.lower() - - return fuzz.partial_ratio(match, text) >= 90 + if fuzz.partial_ratio(match, text) >= 90: + # TODO: make this better + log_reason( + matching_model, document, + f"parts of the document content somehow match the string " + f"{matching_model.match}" + ) + return True + else: + return False elif matching_model.matching_algorithm == MatchingModel.MATCH_AUTO: # this is done elsewhere. @@ -113,6 +161,9 @@ def _split_match(matching_model): findterms = re.compile(r'"([^"]+)"|(\S+)').findall normspace = re.compile(r"\s+").sub return [ - normspace(" ", (t[0] or t[1]).strip()).replace(" ", r"\s+") + # normspace(" ", (t[0] or t[1]).strip()).replace(" ", r"\s+") + re.escape( + normspace(" ", (t[0] or t[1]).strip()) + ).replace(r"\ ", r"\s+") for t in findterms(matching_model.match) ] diff --git a/src/documents/migrations/1003_mime_types.py b/src/documents/migrations/1003_mime_types.py index 1038d57b3..c196f29f4 100644 --- a/src/documents/migrations/1003_mime_types.py +++ b/src/documents/migrations/1003_mime_types.py @@ -6,13 +6,18 @@ import magic from django.conf import settings from django.db import migrations, models +from paperless.db import GnuPG + +STORAGE_TYPE_UNENCRYPTED = "unencrypted" +STORAGE_TYPE_GPG = "gpg" + def source_path(self): if self.filename: fname = str(self.filename) else: fname = "{:07}.{}".format(self.pk, self.file_type) - if self.storage_type == self.STORAGE_TYPE_GPG: + if self.storage_type == STORAGE_TYPE_GPG: fname += ".gpg" return os.path.join( @@ -26,9 +31,18 @@ def add_mime_types(apps, schema_editor): documents = Document.objects.all() for d in documents: - d.mime_type = magic.from_file(source_path(d), mime=True) + f = open(source_path(d), "rb") + if d.storage_type == STORAGE_TYPE_GPG: + + data = GnuPG.decrypted(f) + else: + data = f.read(1024) + + d.mime_type = magic.from_buffer(data, mime=True) d.save() + f.close() + def add_file_extensions(apps, schema_editor): Document = apps.get_model("documents", "Document") diff --git a/src/documents/migrations/1007_savedview_savedviewfilterrule.py b/src/documents/migrations/1007_savedview_savedviewfilterrule.py new file mode 100644 index 000000000..664def5f1 --- /dev/null +++ b/src/documents/migrations/1007_savedview_savedviewfilterrule.py @@ -0,0 +1,37 @@ +# Generated by Django 3.1.4 on 2020-12-12 14:41 + +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ('documents', '1006_auto_20201208_2209'), + ] + + operations = [ + migrations.CreateModel( + name='SavedView', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('name', models.CharField(max_length=128)), + ('show_on_dashboard', models.BooleanField()), + ('show_in_sidebar', models.BooleanField()), + ('sort_field', models.CharField(max_length=128)), + ('sort_reverse', models.BooleanField(default=False)), + ('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL)), + ], + ), + migrations.CreateModel( + name='SavedViewFilterRule', + fields=[ + ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), + ('rule_type', models.PositiveIntegerField(choices=[(0, 'Title contains'), (1, 'Content contains'), (2, 'ASN is'), (3, 'Correspondent is'), (4, 'Document type is'), (5, 'Is in inbox'), (6, 'Has tag'), (7, 'Has any tag'), (8, 'Created before'), (9, 'Created after'), (10, 'Created year is'), (11, 'Created month is'), (12, 'Created day is'), (13, 'Added before'), (14, 'Added after'), (15, 'Modified before'), (16, 'Modified after'), (17, 'Does not have tag')])), + ('value', models.CharField(max_length=128)), + ('saved_view', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='filter_rules', to='documents.savedview')), + ], + ), + ] diff --git a/src/documents/migrations/1008_auto_20201216_1736.py b/src/documents/migrations/1008_auto_20201216_1736.py new file mode 100644 index 000000000..d94f4767f --- /dev/null +++ b/src/documents/migrations/1008_auto_20201216_1736.py @@ -0,0 +1,34 @@ +# Generated by Django 3.1.4 on 2020-12-16 17:36 + +from django.db import migrations +import django.db.models.functions.text + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1007_savedview_savedviewfilterrule'), + ] + + operations = [ + migrations.AlterModelOptions( + name='correspondent', + options={'ordering': (django.db.models.functions.text.Lower('name'),)}, + ), + migrations.AlterModelOptions( + name='document', + options={'ordering': ('-created',)}, + ), + migrations.AlterModelOptions( + name='documenttype', + options={'ordering': (django.db.models.functions.text.Lower('name'),)}, + ), + migrations.AlterModelOptions( + name='savedview', + options={'ordering': (django.db.models.functions.text.Lower('name'),)}, + ), + migrations.AlterModelOptions( + name='tag', + options={'ordering': (django.db.models.functions.text.Lower('name'),)}, + ), + ] diff --git a/src/documents/migrations/1009_auto_20201216_2005.py b/src/documents/migrations/1009_auto_20201216_2005.py new file mode 100644 index 000000000..5e8302bb0 --- /dev/null +++ b/src/documents/migrations/1009_auto_20201216_2005.py @@ -0,0 +1,29 @@ +# Generated by Django 3.1.4 on 2020-12-16 20:05 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1008_auto_20201216_1736'), + ] + + operations = [ + migrations.AlterModelOptions( + name='correspondent', + options={'ordering': ('name',)}, + ), + migrations.AlterModelOptions( + name='documenttype', + options={'ordering': ('name',)}, + ), + migrations.AlterModelOptions( + name='savedview', + options={'ordering': ('name',)}, + ), + migrations.AlterModelOptions( + name='tag', + options={'ordering': ('name',)}, + ), + ] diff --git a/src/documents/migrations/1010_auto_20210101_2159.py b/src/documents/migrations/1010_auto_20210101_2159.py new file mode 100644 index 000000000..4c30125be --- /dev/null +++ b/src/documents/migrations/1010_auto_20210101_2159.py @@ -0,0 +1,18 @@ +# Generated by Django 3.1.4 on 2021-01-01 21:59 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1009_auto_20201216_2005'), + ] + + operations = [ + migrations.AlterField( + model_name='savedviewfilterrule', + name='value', + field=models.CharField(blank=True, max_length=128, null=True), + ), + ] diff --git a/src/documents/migrations/1011_auto_20210101_2340.py b/src/documents/migrations/1011_auto_20210101_2340.py new file mode 100644 index 000000000..9405a5210 --- /dev/null +++ b/src/documents/migrations/1011_auto_20210101_2340.py @@ -0,0 +1,250 @@ +# Generated by Django 3.1.4 on 2021-01-01 23:40 + +from django.conf import settings +from django.db import migrations, models +import django.db.models.deletion +import django.utils.timezone + + +class Migration(migrations.Migration): + + dependencies = [ + migrations.swappable_dependency(settings.AUTH_USER_MODEL), + ('documents', '1010_auto_20210101_2159'), + ] + + operations = [ + migrations.AlterModelOptions( + name='correspondent', + options={'ordering': ('name',), 'verbose_name': 'correspondent', 'verbose_name_plural': 'correspondents'}, + ), + migrations.AlterModelOptions( + name='document', + options={'ordering': ('-created',), 'verbose_name': 'document', 'verbose_name_plural': 'documents'}, + ), + migrations.AlterModelOptions( + name='documenttype', + options={'verbose_name': 'document type', 'verbose_name_plural': 'document types'}, + ), + migrations.AlterModelOptions( + name='log', + options={'ordering': ('-created',), 'verbose_name': 'log', 'verbose_name_plural': 'logs'}, + ), + migrations.AlterModelOptions( + name='savedview', + options={'ordering': ('name',), 'verbose_name': 'saved view', 'verbose_name_plural': 'saved views'}, + ), + migrations.AlterModelOptions( + name='savedviewfilterrule', + options={'verbose_name': 'filter rule', 'verbose_name_plural': 'filter rules'}, + ), + migrations.AlterModelOptions( + name='tag', + options={'verbose_name': 'tag', 'verbose_name_plural': 'tags'}, + ), + migrations.AlterField( + model_name='correspondent', + name='is_insensitive', + field=models.BooleanField(default=True, verbose_name='is insensitive'), + ), + migrations.AlterField( + model_name='correspondent', + name='match', + field=models.CharField(blank=True, max_length=256, verbose_name='match'), + ), + migrations.AlterField( + model_name='correspondent', + name='matching_algorithm', + field=models.PositiveIntegerField(choices=[(1, 'Any word'), (2, 'All words'), (3, 'Exact match'), (4, 'Regular expression'), (5, 'Fuzzy word'), (6, 'Automatic')], default=1, verbose_name='matching algorithm'), + ), + migrations.AlterField( + model_name='correspondent', + name='name', + field=models.CharField(max_length=128, unique=True, verbose_name='name'), + ), + migrations.AlterField( + model_name='document', + name='added', + field=models.DateTimeField(db_index=True, default=django.utils.timezone.now, editable=False, verbose_name='added'), + ), + migrations.AlterField( + model_name='document', + name='archive_checksum', + field=models.CharField(blank=True, editable=False, help_text='The checksum of the archived document.', max_length=32, null=True, verbose_name='archive checksum'), + ), + migrations.AlterField( + model_name='document', + name='archive_serial_number', + field=models.IntegerField(blank=True, db_index=True, help_text='The position of this document in your physical document archive.', null=True, unique=True, verbose_name='archive serial number'), + ), + migrations.AlterField( + model_name='document', + name='checksum', + field=models.CharField(editable=False, help_text='The checksum of the original document.', max_length=32, unique=True, verbose_name='checksum'), + ), + migrations.AlterField( + model_name='document', + name='content', + field=models.TextField(blank=True, help_text='The raw, text-only data of the document. This field is primarily used for searching.', verbose_name='content'), + ), + migrations.AlterField( + model_name='document', + name='correspondent', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='documents', to='documents.correspondent', verbose_name='correspondent'), + ), + migrations.AlterField( + model_name='document', + name='created', + field=models.DateTimeField(db_index=True, default=django.utils.timezone.now, verbose_name='created'), + ), + migrations.AlterField( + model_name='document', + name='document_type', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='documents', to='documents.documenttype', verbose_name='document type'), + ), + migrations.AlterField( + model_name='document', + name='filename', + field=models.FilePathField(default=None, editable=False, help_text='Current filename in storage', max_length=1024, null=True, verbose_name='filename'), + ), + migrations.AlterField( + model_name='document', + name='mime_type', + field=models.CharField(editable=False, max_length=256, verbose_name='mime type'), + ), + migrations.AlterField( + model_name='document', + name='modified', + field=models.DateTimeField(auto_now=True, db_index=True, verbose_name='modified'), + ), + migrations.AlterField( + model_name='document', + name='storage_type', + field=models.CharField(choices=[('unencrypted', 'Unencrypted'), ('gpg', 'Encrypted with GNU Privacy Guard')], default='unencrypted', editable=False, max_length=11, verbose_name='storage type'), + ), + migrations.AlterField( + model_name='document', + name='tags', + field=models.ManyToManyField(blank=True, related_name='documents', to='documents.Tag', verbose_name='tags'), + ), + migrations.AlterField( + model_name='document', + name='title', + field=models.CharField(blank=True, db_index=True, max_length=128, verbose_name='title'), + ), + migrations.AlterField( + model_name='documenttype', + name='is_insensitive', + field=models.BooleanField(default=True, verbose_name='is insensitive'), + ), + migrations.AlterField( + model_name='documenttype', + name='match', + field=models.CharField(blank=True, max_length=256, verbose_name='match'), + ), + migrations.AlterField( + model_name='documenttype', + name='matching_algorithm', + field=models.PositiveIntegerField(choices=[(1, 'Any word'), (2, 'All words'), (3, 'Exact match'), (4, 'Regular expression'), (5, 'Fuzzy word'), (6, 'Automatic')], default=1, verbose_name='matching algorithm'), + ), + migrations.AlterField( + model_name='documenttype', + name='name', + field=models.CharField(max_length=128, unique=True, verbose_name='name'), + ), + migrations.AlterField( + model_name='log', + name='created', + field=models.DateTimeField(auto_now_add=True, verbose_name='created'), + ), + migrations.AlterField( + model_name='log', + name='group', + field=models.UUIDField(blank=True, null=True, verbose_name='group'), + ), + migrations.AlterField( + model_name='log', + name='level', + field=models.PositiveIntegerField(choices=[(10, 'debug'), (20, 'information'), (30, 'warning'), (40, 'error'), (50, 'critical')], default=20, verbose_name='level'), + ), + migrations.AlterField( + model_name='log', + name='message', + field=models.TextField(verbose_name='message'), + ), + migrations.AlterField( + model_name='savedview', + name='name', + field=models.CharField(max_length=128, verbose_name='name'), + ), + migrations.AlterField( + model_name='savedview', + name='show_in_sidebar', + field=models.BooleanField(verbose_name='show in sidebar'), + ), + migrations.AlterField( + model_name='savedview', + name='show_on_dashboard', + field=models.BooleanField(verbose_name='show on dashboard'), + ), + migrations.AlterField( + model_name='savedview', + name='sort_field', + field=models.CharField(max_length=128, verbose_name='sort field'), + ), + migrations.AlterField( + model_name='savedview', + name='sort_reverse', + field=models.BooleanField(default=False, verbose_name='sort reverse'), + ), + migrations.AlterField( + model_name='savedview', + name='user', + field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL, verbose_name='user'), + ), + migrations.AlterField( + model_name='savedviewfilterrule', + name='rule_type', + field=models.PositiveIntegerField(choices=[(0, 'title contains'), (1, 'content contains'), (2, 'ASN is'), (3, 'correspondent is'), (4, 'document type is'), (5, 'is in inbox'), (6, 'has tag'), (7, 'has any tag'), (8, 'created before'), (9, 'created after'), (10, 'created year is'), (11, 'created month is'), (12, 'created day is'), (13, 'added before'), (14, 'added after'), (15, 'modified before'), (16, 'modified after'), (17, 'does not have tag')], verbose_name='rule type'), + ), + migrations.AlterField( + model_name='savedviewfilterrule', + name='saved_view', + field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='filter_rules', to='documents.savedview', verbose_name='saved view'), + ), + migrations.AlterField( + model_name='savedviewfilterrule', + name='value', + field=models.CharField(blank=True, max_length=128, null=True, verbose_name='value'), + ), + migrations.AlterField( + model_name='tag', + name='colour', + field=models.PositiveIntegerField(choices=[(1, '#a6cee3'), (2, '#1f78b4'), (3, '#b2df8a'), (4, '#33a02c'), (5, '#fb9a99'), (6, '#e31a1c'), (7, '#fdbf6f'), (8, '#ff7f00'), (9, '#cab2d6'), (10, '#6a3d9a'), (11, '#b15928'), (12, '#000000'), (13, '#cccccc')], default=1, verbose_name='color'), + ), + migrations.AlterField( + model_name='tag', + name='is_inbox_tag', + field=models.BooleanField(default=False, help_text='Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags.', verbose_name='is inbox tag'), + ), + migrations.AlterField( + model_name='tag', + name='is_insensitive', + field=models.BooleanField(default=True, verbose_name='is insensitive'), + ), + migrations.AlterField( + model_name='tag', + name='match', + field=models.CharField(blank=True, max_length=256, verbose_name='match'), + ), + migrations.AlterField( + model_name='tag', + name='matching_algorithm', + field=models.PositiveIntegerField(choices=[(1, 'Any word'), (2, 'All words'), (3, 'Exact match'), (4, 'Regular expression'), (5, 'Fuzzy word'), (6, 'Automatic')], default=1, verbose_name='matching algorithm'), + ), + migrations.AlterField( + model_name='tag', + name='name', + field=models.CharField(max_length=128, unique=True, verbose_name='name'), + ), + ] diff --git a/src/documents/migrations/1012_fix_archive_files.py b/src/documents/migrations/1012_fix_archive_files.py new file mode 100644 index 000000000..5f5064396 --- /dev/null +++ b/src/documents/migrations/1012_fix_archive_files.py @@ -0,0 +1,330 @@ +# Generated by Django 3.1.6 on 2021-02-07 22:26 +import datetime +import hashlib +import logging +import os +import shutil +from time import sleep + +import pathvalidate +from django.conf import settings +from django.db import migrations, models +from django.template.defaultfilters import slugify + +from documents.file_handling import defaultdictNoStr, many_to_dictionary + + +logger = logging.getLogger("paperless.migrations") + +############################################################################### +# This is code copied straight paperless before the change. +############################################################################### + +def archive_name_from_filename(filename): + return os.path.splitext(filename)[0] + ".pdf" + + +def archive_path_old(doc): + if doc.filename: + fname = archive_name_from_filename(doc.filename) + else: + fname = "{:07}.pdf".format(doc.pk) + + return os.path.join( + settings.ARCHIVE_DIR, + fname + ) + + +STORAGE_TYPE_GPG = "gpg" + + +def archive_path_new(doc): + if doc.archive_filename is not None: + return os.path.join( + settings.ARCHIVE_DIR, + str(doc.archive_filename) + ) + else: + return None + + +def source_path(doc): + if doc.filename: + fname = str(doc.filename) + else: + fname = "{:07}{}".format(doc.pk, doc.file_type) + if doc.storage_type == STORAGE_TYPE_GPG: + fname += ".gpg" # pragma: no cover + + return os.path.join( + settings.ORIGINALS_DIR, + fname + ) + + +def generate_unique_filename(doc, archive_filename=False): + if archive_filename: + old_filename = doc.archive_filename + root = settings.ARCHIVE_DIR + else: + old_filename = doc.filename + root = settings.ORIGINALS_DIR + + counter = 0 + + while True: + new_filename = generate_filename( + doc, counter, archive_filename=archive_filename) + if new_filename == old_filename: + # still the same as before. + return new_filename + + if os.path.exists(os.path.join(root, new_filename)): + counter += 1 + else: + return new_filename + + +def generate_filename(doc, counter=0, append_gpg=True, archive_filename=False): + path = "" + + try: + if settings.PAPERLESS_FILENAME_FORMAT is not None: + tags = defaultdictNoStr(lambda: slugify(None), + many_to_dictionary(doc.tags)) + + tag_list = pathvalidate.sanitize_filename( + ",".join(sorted( + [tag.name for tag in doc.tags.all()] + )), + replacement_text="-" + ) + + if doc.correspondent: + correspondent = pathvalidate.sanitize_filename( + doc.correspondent.name, replacement_text="-" + ) + else: + correspondent = "none" + + if doc.document_type: + document_type = pathvalidate.sanitize_filename( + doc.document_type.name, replacement_text="-" + ) + else: + document_type = "none" + + path = settings.PAPERLESS_FILENAME_FORMAT.format( + title=pathvalidate.sanitize_filename( + doc.title, replacement_text="-"), + correspondent=correspondent, + document_type=document_type, + created=datetime.date.isoformat(doc.created), + created_year=doc.created.year if doc.created else "none", + created_month=f"{doc.created.month:02}" if doc.created else "none", # NOQA: E501 + created_day=f"{doc.created.day:02}" if doc.created else "none", + added=datetime.date.isoformat(doc.added), + added_year=doc.added.year if doc.added else "none", + added_month=f"{doc.added.month:02}" if doc.added else "none", + added_day=f"{doc.added.day:02}" if doc.added else "none", + tags=tags, + tag_list=tag_list + ).strip() + + path = path.strip(os.sep) + + except (ValueError, KeyError, IndexError): + logger.warning( + f"Invalid PAPERLESS_FILENAME_FORMAT: " + f"{settings.PAPERLESS_FILENAME_FORMAT}, falling back to default") + + counter_str = f"_{counter:02}" if counter else "" + + filetype_str = ".pdf" if archive_filename else doc.file_type + + if len(path) > 0: + filename = f"{path}{counter_str}{filetype_str}" + else: + filename = f"{doc.pk:07}{counter_str}{filetype_str}" + + # Append .gpg for encrypted files + if append_gpg and doc.storage_type == STORAGE_TYPE_GPG: + filename += ".gpg" + + return filename + + +############################################################################### +# This code performs bidirection archive file transformation. +############################################################################### + + +def parse_wrapper(parser, path, mime_type, file_name): + # this is here so that I can mock this out for testing. + parser.parse(path, mime_type, file_name) + + +def create_archive_version(doc, retry_count=3): + from documents.parsers import get_parser_class_for_mime_type, \ + DocumentParser, \ + ParseError + + logger.info( + f"Regenerating archive document for document ID:{doc.id}" + ) + parser_class = get_parser_class_for_mime_type(doc.mime_type) + for try_num in range(retry_count): + parser: DocumentParser = parser_class(None, None) + try: + parse_wrapper(parser, source_path(doc), doc.mime_type, + os.path.basename(doc.filename)) + doc.content = parser.get_text() + + if parser.get_archive_path() and os.path.isfile( + parser.get_archive_path()): + doc.archive_filename = generate_unique_filename( + doc, archive_filename=True) + with open(parser.get_archive_path(), "rb") as f: + doc.archive_checksum = hashlib.md5(f.read()).hexdigest() + os.makedirs(os.path.dirname(archive_path_new(doc)), + exist_ok=True) + shutil.copy2(parser.get_archive_path(), archive_path_new(doc)) + else: + doc.archive_checksum = None + logger.error( + f"Parser did not return an archive document for document " + f"ID:{doc.id}. Removing archive document." + ) + doc.save() + return + except ParseError: + if try_num + 1 == retry_count: + logger.exception( + f"Unable to regenerate archive document for ID:{doc.id}. You " + f"need to invoke the document_archiver management command " + f"manually for that document." + ) + doc.archive_checksum = None + doc.save() + return + else: + # This is mostly here for the tika parser in docker + # environemnts. The servers for parsing need to come up first, + # and the docker setup doesn't ensure that tika is running + # before attempting migrations. + logger.error("Parse error, will try again in 5 seconds...") + sleep(5) + finally: + parser.cleanup() + + +def move_old_to_new_locations(apps, schema_editor): + Document = apps.get_model("documents", "Document") + + affected_document_ids = set() + + old_archive_path_to_id = {} + + # check for documents that have incorrect archive versions + for doc in Document.objects.filter(archive_checksum__isnull=False): + old_path = archive_path_old(doc) + + if old_path in old_archive_path_to_id: + affected_document_ids.add(doc.id) + affected_document_ids.add(old_archive_path_to_id[old_path]) + else: + old_archive_path_to_id[old_path] = doc.id + + # check that archive files of all unaffected documents are in place + for doc in Document.objects.filter(archive_checksum__isnull=False): + old_path = archive_path_old(doc) + if doc.id not in affected_document_ids and not os.path.isfile(old_path): + raise ValueError( + f"Archived document ID:{doc.id} does not exist at: " + f"{old_path}") + + # check that we can regenerate affected archive versions + for doc_id in affected_document_ids: + from documents.parsers import get_parser_class_for_mime_type + + doc = Document.objects.get(id=doc_id) + parser_class = get_parser_class_for_mime_type(doc.mime_type) + if not parser_class: + raise ValueError( + f"Document ID:{doc.id} has an invalid archived document, " + f"but no parsers are available. Cannot migrate.") + + for doc in Document.objects.filter(archive_checksum__isnull=False): + + if doc.id in affected_document_ids: + old_path = archive_path_old(doc) + # remove affected archive versions + if os.path.isfile(old_path): + logger.debug( + f"Removing {old_path}" + ) + os.unlink(old_path) + else: + # Set archive path for unaffected files + doc.archive_filename = archive_name_from_filename(doc.filename) + Document.objects.filter(id=doc.id).update( + archive_filename=doc.archive_filename + ) + + # regenerate archive documents + for doc_id in affected_document_ids: + doc = Document.objects.get(id=doc_id) + create_archive_version(doc) + + +def move_new_to_old_locations(apps, schema_editor): + Document = apps.get_model("documents", "Document") + + old_archive_paths = set() + + for doc in Document.objects.filter(archive_checksum__isnull=False): + new_archive_path = archive_path_new(doc) + old_archive_path = archive_path_old(doc) + if old_archive_path in old_archive_paths: + raise ValueError( + f"Cannot migrate: Archive file name {old_archive_path} of " + f"document {doc.filename} would clash with another archive " + f"filename.") + old_archive_paths.add(old_archive_path) + if new_archive_path != old_archive_path and os.path.isfile(old_archive_path): + raise ValueError( + f"Cannot migrate: Cannot move {new_archive_path} to " + f"{old_archive_path}: file already exists." + ) + + for doc in Document.objects.filter(archive_checksum__isnull=False): + new_archive_path = archive_path_new(doc) + old_archive_path = archive_path_old(doc) + if new_archive_path != old_archive_path: + logger.debug(f"Moving {new_archive_path} to {old_archive_path}") + shutil.move(new_archive_path, old_archive_path) + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1011_auto_20210101_2340'), + ] + + operations = [ + migrations.AddField( + model_name='document', + name='archive_filename', + field=models.FilePathField(default=None, editable=False, help_text='Current archive filename in storage', max_length=1024, null=True, unique=True, verbose_name='archive filename'), + ), + migrations.AlterField( + model_name='document', + name='filename', + field=models.FilePathField(default=None, editable=False, help_text='Current filename in storage', max_length=1024, null=True, unique=True, verbose_name='filename'), + ), + migrations.RunPython( + move_old_to_new_locations, + move_new_to_old_locations + ), + ] diff --git a/src/documents/migrations/1013_migrate_tag_colour.py b/src/documents/migrations/1013_migrate_tag_colour.py new file mode 100644 index 000000000..323ff2bfb --- /dev/null +++ b/src/documents/migrations/1013_migrate_tag_colour.py @@ -0,0 +1,70 @@ +# Generated by Django 3.1.4 on 2020-12-02 21:43 + +from django.db import migrations, models + +COLOURS_OLD = { + 1: "#a6cee3", + 2: "#1f78b4", + 3: "#b2df8a", + 4: "#33a02c", + 5: "#fb9a99", + 6: "#e31a1c", + 7: "#fdbf6f", + 8: "#ff7f00", + 9: "#cab2d6", + 10: "#6a3d9a", + 11: "#b15928", + 12: "#000000", + 13: "#cccccc", +} + + +def forward(apps, schema_editor): + Tag = apps.get_model('documents', 'Tag') + + for tag in Tag.objects.all(): + colour_old_id = tag.colour_old + rgb = COLOURS_OLD[colour_old_id] + tag.color = rgb + tag.save() + + +def reverse(apps, schema_editor): + Tag = apps.get_model('documents', 'Tag') + + def _get_colour_id(rdb): + for idx, rdbx in COLOURS_OLD.items(): + if rdbx == rdb: + return idx + # Return colour 1 if we can't match anything + return 1 + + for tag in Tag.objects.all(): + colour_id = _get_colour_id(tag.color) + tag.colour_old = colour_id + tag.save() + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1012_fix_archive_files'), + ] + + operations = [ + migrations.RenameField( + model_name='tag', + old_name='colour', + new_name='colour_old', + ), + migrations.AddField( + model_name='tag', + name='color', + field=models.CharField(default='#a6cee3', max_length=7, verbose_name='color'), + ), + migrations.RunPython(forward, reverse), + migrations.RemoveField( + model_name='tag', + name='colour_old', + ) + ] diff --git a/src/documents/migrations/1014_auto_20210228_1614.py b/src/documents/migrations/1014_auto_20210228_1614.py new file mode 100644 index 000000000..cb716fa82 --- /dev/null +++ b/src/documents/migrations/1014_auto_20210228_1614.py @@ -0,0 +1,18 @@ +# Generated by Django 3.1.7 on 2021-02-28 15:14 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1013_migrate_tag_colour'), + ] + + operations = [ + migrations.AlterField( + model_name='savedviewfilterrule', + name='rule_type', + field=models.PositiveIntegerField(choices=[(0, 'title contains'), (1, 'content contains'), (2, 'ASN is'), (3, 'correspondent is'), (4, 'document type is'), (5, 'is in inbox'), (6, 'has tag'), (7, 'has any tag'), (8, 'created before'), (9, 'created after'), (10, 'created year is'), (11, 'created month is'), (12, 'created day is'), (13, 'added before'), (14, 'added after'), (15, 'modified before'), (16, 'modified after'), (17, 'does not have tag'), (18, 'does not have ASN'), (19, 'title or content contains')], verbose_name='rule type'), + ), + ] diff --git a/src/documents/migrations/1015_remove_null_characters.py b/src/documents/migrations/1015_remove_null_characters.py new file mode 100644 index 000000000..2f7ee99b6 --- /dev/null +++ b/src/documents/migrations/1015_remove_null_characters.py @@ -0,0 +1,29 @@ +# Generated by Django 3.1.7 on 2021-04-04 18:28 +import logging + +from django.db import migrations + + +logger = logging.getLogger("paperless.migrations") + + +def remove_null_characters(apps, schema_editor): + Document = apps.get_model('documents', 'Document') + + for doc in Document.objects.all(): + content: str = doc.content + if '\0' in content: + logger.info(f"Removing null characters from document {doc}...") + doc.content = content.replace('\0', ' ') + doc.save() + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1014_auto_20210228_1614'), + ] + + operations = [ + migrations.RunPython(remove_null_characters, migrations.RunPython.noop) + ] diff --git a/src/documents/migrations/1016_auto_20210317_1351.py b/src/documents/migrations/1016_auto_20210317_1351.py new file mode 100644 index 000000000..733c1bb33 --- /dev/null +++ b/src/documents/migrations/1016_auto_20210317_1351.py @@ -0,0 +1,23 @@ +# Generated by Django 3.1.7 on 2021-03-17 12:51 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1015_remove_null_characters'), + ] + + operations = [ + migrations.AlterField( + model_name='savedview', + name='sort_field', + field=models.CharField(blank=True, max_length=128, null=True, verbose_name='sort field'), + ), + migrations.AlterField( + model_name='savedviewfilterrule', + name='rule_type', + field=models.PositiveIntegerField(choices=[(0, 'title contains'), (1, 'content contains'), (2, 'ASN is'), (3, 'correspondent is'), (4, 'document type is'), (5, 'is in inbox'), (6, 'has tag'), (7, 'has any tag'), (8, 'created before'), (9, 'created after'), (10, 'created year is'), (11, 'created month is'), (12, 'created day is'), (13, 'added before'), (14, 'added after'), (15, 'modified before'), (16, 'modified after'), (17, 'does not have tag'), (18, 'does not have ASN'), (19, 'title or content contains'), (20, 'fulltext query'), (21, 'more like this')], verbose_name='rule type'), + ), + ] diff --git a/src/documents/mixins.py b/src/documents/mixins.py deleted file mode 100755 index 949be02e8..000000000 --- a/src/documents/mixins.py +++ /dev/null @@ -1,9 +0,0 @@ -class Renderable: - """ - A handy mixin to make it easier/cleaner to print output based on a - verbosity value. - """ - - def _render(self, text, verbosity): - if self.verbosity >= verbosity: - print(text) diff --git a/src/documents/models.py b/src/documents/models.py old mode 100755 new mode 100644 index f0678a843..2a763c602 --- a/src/documents/models.py +++ b/src/documents/models.py @@ -9,11 +9,13 @@ import pathvalidate import dateutil.parser from django.conf import settings +from django.contrib.auth.models import User from django.db import models from django.utils import timezone -from django.utils.text import slugify +from django.utils.timezone import is_aware + +from django.utils.translation import gettext_lazy as _ -from documents.file_handling import archive_name_from_filename from documents.parsers import get_default_file_extension @@ -27,36 +29,31 @@ class MatchingModel(models.Model): MATCH_AUTO = 6 MATCHING_ALGORITHMS = ( - (MATCH_ANY, "Any"), - (MATCH_ALL, "All"), - (MATCH_LITERAL, "Literal"), - (MATCH_REGEX, "Regular Expression"), - (MATCH_FUZZY, "Fuzzy Match"), - (MATCH_AUTO, "Automatic Classification"), + (MATCH_ANY, _("Any word")), + (MATCH_ALL, _("All words")), + (MATCH_LITERAL, _("Exact match")), + (MATCH_REGEX, _("Regular expression")), + (MATCH_FUZZY, _("Fuzzy word")), + (MATCH_AUTO, _("Automatic")), ) - name = models.CharField(max_length=128, unique=True) + name = models.CharField( + _("name"), + max_length=128, unique=True) + + match = models.CharField( + _("match"), + max_length=256, blank=True) - match = models.CharField(max_length=256, blank=True) matching_algorithm = models.PositiveIntegerField( + _("matching algorithm"), choices=MATCHING_ALGORITHMS, - default=MATCH_ANY, - help_text=( - "Which algorithm you want to use when matching text to the OCR'd " - "PDF. Here, \"any\" looks for any occurrence of any word " - "provided in the PDF, while \"all\" requires that every word " - "provided appear in the PDF, albeit not in the order provided. A " - "\"literal\" match means that the text you enter must appear in " - "the PDF exactly as you've entered it, and \"regular expression\" " - "uses a regex to match the PDF. (If you don't know what a regex " - "is, you probably don't want this option.) Finally, a \"fuzzy " - "match\" looks for words or phrases that are mostly—but not " - "exactly—the same, which can be useful for matching against " - "documents containg imperfections that foil accurate OCR." - ) + default=MATCH_ANY ) - is_insensitive = models.BooleanField(default=True) + is_insensitive = models.BooleanField( + _("is insensitive"), + default=True) class Meta: abstract = True @@ -65,53 +62,40 @@ class MatchingModel(models.Model): def __str__(self): return self.name - def save(self, *args, **kwargs): - - self.match = self.match.lower() - - models.Model.save(self, *args, **kwargs) - class Correspondent(MatchingModel): - # This regex is probably more restrictive than it needs to be, but it's - # better safe than sorry. - SAFE_REGEX = re.compile(r"^[\w\- ,.']+$") - class Meta: ordering = ("name",) + verbose_name = _("correspondent") + verbose_name_plural = _("correspondents") class Tag(MatchingModel): - COLOURS = ( - (1, "#a6cee3"), - (2, "#1f78b4"), - (3, "#b2df8a"), - (4, "#33a02c"), - (5, "#fb9a99"), - (6, "#e31a1c"), - (7, "#fdbf6f"), - (8, "#ff7f00"), - (9, "#cab2d6"), - (10, "#6a3d9a"), - (11, "#b15928"), - (12, "#000000"), - (13, "#cccccc") + color = models.CharField( + _("color"), + max_length=7, + default="#a6cee3" ) - colour = models.PositiveIntegerField(choices=COLOURS, default=1) - is_inbox_tag = models.BooleanField( + _("is inbox tag"), default=False, - help_text="Marks this tag as an inbox tag: All newly consumed " - "documents will be tagged with inbox tags." + help_text=_("Marks this tag as an inbox tag: All newly consumed " + "documents will be tagged with inbox tags.") ) + class Meta: + verbose_name = _("tag") + verbose_name_plural = _("tags") + class DocumentType(MatchingModel): - pass + class Meta: + verbose_name = _("document type") + verbose_name_plural = _("document types") class Document(models.Model): @@ -119,8 +103,8 @@ class Document(models.Model): STORAGE_TYPE_UNENCRYPTED = "unencrypted" STORAGE_TYPE_GPG = "gpg" STORAGE_TYPES = ( - (STORAGE_TYPE_UNENCRYPTED, "Unencrypted"), - (STORAGE_TYPE_GPG, "Encrypted with GNU Privacy Guard") + (STORAGE_TYPE_UNENCRYPTED, _("Unencrypted")), + (STORAGE_TYPE_GPG, _("Encrypted with GNU Privacy Guard")) ) correspondent = models.ForeignKey( @@ -128,55 +112,68 @@ class Document(models.Model): blank=True, null=True, related_name="documents", - on_delete=models.SET_NULL + on_delete=models.SET_NULL, + verbose_name=_("correspondent") ) - title = models.CharField(max_length=128, blank=True, db_index=True) + title = models.CharField( + _("title"), + max_length=128, blank=True, db_index=True) document_type = models.ForeignKey( DocumentType, blank=True, null=True, related_name="documents", - on_delete=models.SET_NULL + on_delete=models.SET_NULL, + verbose_name=_("document type") ) content = models.TextField( + _("content"), blank=True, - help_text="The raw, text-only data of the document. This field is " - "primarily used for searching." + help_text=_("The raw, text-only data of the document. This field is " + "primarily used for searching.") ) mime_type = models.CharField( + _("mime type"), max_length=256, editable=False ) tags = models.ManyToManyField( - Tag, related_name="documents", blank=True) + Tag, related_name="documents", blank=True, + verbose_name=_("tags") + ) checksum = models.CharField( + _("checksum"), max_length=32, editable=False, unique=True, - help_text="The checksum of the original document." + help_text=_("The checksum of the original document.") ) archive_checksum = models.CharField( + _("archive checksum"), max_length=32, editable=False, blank=True, null=True, - help_text="The checksum of the archived document." + help_text=_("The checksum of the archived document.") ) created = models.DateTimeField( + _("created"), default=timezone.now, db_index=True) modified = models.DateTimeField( + _("modified"), auto_now=True, editable=False, db_index=True) storage_type = models.CharField( + _("storage type"), max_length=11, choices=STORAGE_TYPES, default=STORAGE_TYPE_UNENCRYPTED, @@ -184,30 +181,49 @@ class Document(models.Model): ) added = models.DateTimeField( + _("added"), default=timezone.now, editable=False, db_index=True) filename = models.FilePathField( + _("filename"), max_length=1024, editable=False, default=None, + unique=True, null=True, - help_text="Current filename in storage" + help_text=_("Current filename in storage") + ) + + archive_filename = models.FilePathField( + _("archive filename"), + max_length=1024, + editable=False, + default=None, + unique=True, + null=True, + help_text=_("Current archive filename in storage") ) archive_serial_number = models.IntegerField( + _("archive serial number"), blank=True, null=True, unique=True, db_index=True, - help_text="The position of this document in your physical document " - "archive." + help_text=_("The position of this document in your physical document " + "archive.") ) class Meta: - ordering = ("correspondent", "title") + ordering = ("-created",) + verbose_name = _("document") + verbose_name_plural = _("documents") def __str__(self): - created = datetime.date.isoformat(self.created) + if is_aware(self.created): + created = timezone.localdate(self.created).isoformat() + else: + created = datetime.date.isoformat(self.created) if self.correspondent and self.title: return f"{created} {self.correspondent} {self.title}" else: @@ -220,7 +236,7 @@ class Document(models.Model): else: fname = "{:07}{}".format(self.pk, self.file_type) if self.storage_type == self.STORAGE_TYPE_GPG: - fname += ".gpg" + fname += ".gpg" # pragma: no cover return os.path.join( settings.ORIGINALS_DIR, @@ -232,16 +248,18 @@ class Document(models.Model): return open(self.source_path, "rb") @property - def archive_path(self): - if self.filename: - fname = archive_name_from_filename(self.filename) - else: - fname = "{:07}.pdf".format(self.pk) + def has_archive_version(self): + return self.archive_filename is not None - return os.path.join( - settings.ARCHIVE_DIR, - fname - ) + @property + def archive_path(self): + if self.has_archive_version: + return os.path.join( + settings.ARCHIVE_DIR, + str(self.archive_filename) + ) + else: + return None @property def archive_file(self): @@ -286,76 +304,123 @@ class Document(models.Model): class Log(models.Model): LEVELS = ( - (logging.DEBUG, "Debugging"), - (logging.INFO, "Informational"), - (logging.WARNING, "Warning"), - (logging.ERROR, "Error"), - (logging.CRITICAL, "Critical"), + (logging.DEBUG, _("debug")), + (logging.INFO, _("information")), + (logging.WARNING, _("warning")), + (logging.ERROR, _("error")), + (logging.CRITICAL, _("critical")), ) - group = models.UUIDField(blank=True, null=True) - message = models.TextField() - level = models.PositiveIntegerField(choices=LEVELS, default=logging.INFO) - created = models.DateTimeField(auto_now_add=True) + group = models.UUIDField( + _("group"), + blank=True, null=True) + + message = models.TextField(_("message")) + + level = models.PositiveIntegerField( + _("level"), + choices=LEVELS, default=logging.INFO) + + created = models.DateTimeField(_("created"), auto_now_add=True) class Meta: ordering = ("-created",) + verbose_name = _("log") + verbose_name_plural = _("logs") def __str__(self): return self.message +class SavedView(models.Model): + + class Meta: + + ordering = ("name",) + verbose_name = _("saved view") + verbose_name_plural = _("saved views") + + user = models.ForeignKey(User, on_delete=models.CASCADE, + verbose_name=_("user")) + name = models.CharField( + _("name"), + max_length=128) + + show_on_dashboard = models.BooleanField( + _("show on dashboard"), + ) + show_in_sidebar = models.BooleanField( + _("show in sidebar"), + ) + + sort_field = models.CharField( + _("sort field"), + max_length=128, + null=True, + blank=True + ) + sort_reverse = models.BooleanField( + _("sort reverse"), + default=False) + + +class SavedViewFilterRule(models.Model): + RULE_TYPES = [ + (0, _("title contains")), + (1, _("content contains")), + (2, _("ASN is")), + (3, _("correspondent is")), + (4, _("document type is")), + (5, _("is in inbox")), + (6, _("has tag")), + (7, _("has any tag")), + (8, _("created before")), + (9, _("created after")), + (10, _("created year is")), + (11, _("created month is")), + (12, _("created day is")), + (13, _("added before")), + (14, _("added after")), + (15, _("modified before")), + (16, _("modified after")), + (17, _("does not have tag")), + (18, _("does not have ASN")), + (19, _("title or content contains")), + (20, _("fulltext query")), + (21, _("more like this")) + ] + + saved_view = models.ForeignKey( + SavedView, + on_delete=models.CASCADE, + related_name="filter_rules", + verbose_name=_("saved view") + ) + + rule_type = models.PositiveIntegerField( + _("rule type"), + choices=RULE_TYPES) + + value = models.CharField( + _("value"), + max_length=128, + blank=True, + null=True) + + class Meta: + verbose_name = _("filter rule") + verbose_name_plural = _("filter rules") + + # TODO: why is this in the models file? class FileInfo: - # This epic regex *almost* worked for our needs, so I'm keeping it here for - # posterity, in the hopes that we might find a way to make it work one day. - ALMOST_REGEX = re.compile( - r"^((?P\d\d\d\d\d\d\d\d\d\d\d\d\d\dZ){separator})?" - r"((?P{non_separated_word}+){separator})??" - r"(?P{non_separated_word}+)" - r"({separator}(?P<tags>[a-z,0-9-]+))?" - r"\.(?P<extension>[a-zA-Z.-]+)$".format( - separator=r"\s+-\s+", - non_separated_word=r"([\w,. ]|([^\s]-))" - ) - ) REGEXES = OrderedDict([ - ("created-correspondent-title-tags", re.compile( - r"^(?P<created>\d\d\d\d\d\d\d\d(\d\d\d\d\d\d)?Z) - " - r"(?P<correspondent>.*) - " - r"(?P<title>.*) - " - r"(?P<tags>[a-z0-9\-,]*)$", - flags=re.IGNORECASE - )), - ("created-title-tags", re.compile( - r"^(?P<created>\d\d\d\d\d\d\d\d(\d\d\d\d\d\d)?Z) - " - r"(?P<title>.*) - " - r"(?P<tags>[a-z0-9\-,]*)$", - flags=re.IGNORECASE - )), - ("created-correspondent-title", re.compile( - r"^(?P<created>\d\d\d\d\d\d\d\d(\d\d\d\d\d\d)?Z) - " - r"(?P<correspondent>.*) - " - r"(?P<title>.*)$", - flags=re.IGNORECASE - )), ("created-title", re.compile( r"^(?P<created>\d\d\d\d\d\d\d\d(\d\d\d\d\d\d)?Z) - " r"(?P<title>.*)$", flags=re.IGNORECASE )), - ("correspondent-title-tags", re.compile( - r"(?P<correspondent>.*) - " - r"(?P<title>.*) - " - r"(?P<tags>[a-z0-9\-,]*)$", - flags=re.IGNORECASE - )), - ("correspondent-title", re.compile( - r"(?P<correspondent>.*) - " - r"(?P<title>.*)?$", - flags=re.IGNORECASE - )), ("title", re.compile( r"(?P<title>.*)$", flags=re.IGNORECASE @@ -378,23 +443,10 @@ class FileInfo: except ValueError: return None - @classmethod - def _get_correspondent(cls, name): - if not name: - return None - return Correspondent.objects.get_or_create(name=name)[0] - @classmethod def _get_title(cls, title): return title - @classmethod - def _get_tags(cls, tags): - r = [] - for t in tags.split(","): - r.append(Tag.objects.get_or_create(name=t)[0]) - return tuple(r) - @classmethod def _mangle_property(cls, properties, name): if name in properties: @@ -404,15 +456,6 @@ class FileInfo: @classmethod def from_filename(cls, filename): - """ - We use a crude naming convention to make handling the correspondent, - title, and tags easier: - "<date> - <correspondent> - <title> - <tags>" - "<correspondent> - <title> - <tags>" - "<correspondent> - <title>" - "<title>" - """ - # Mutate filename in-place before parsing its components # by applying at most one of the configured transformations. for (pattern, repl) in settings.FILENAME_PARSE_TRANSFORMS: @@ -443,7 +486,5 @@ class FileInfo: if m: properties = m.groupdict() cls._mangle_property(properties, "created") - cls._mangle_property(properties, "correspondent") cls._mangle_property(properties, "title") - cls._mangle_property(properties, "tags") return cls(**properties) diff --git a/src/documents/parsers.py b/src/documents/parsers.py index 228e2c86e..8cb8f5399 100644 --- a/src/documents/parsers.py +++ b/src/documents/parsers.py @@ -6,7 +6,6 @@ import shutil import subprocess import tempfile -import dateparser import magic from django.conf import settings from django.utils import timezone @@ -36,7 +35,7 @@ DATE_REGEX = re.compile( ) -logger = logging.getLogger(__name__) +logger = logging.getLogger("paperless.parsing") def is_mime_type_supported(mime_type): @@ -117,6 +116,7 @@ def run_convert(input_file, trim=False, type=None, depth=None, + auto_orient=False, extra=None, logging_group=None): @@ -134,6 +134,7 @@ def run_convert(input_file, args += ['-trim'] if trim else [] args += ['-type', str(type)] if type else [] args += ['-depth', str(depth)] if depth else [] + args += ['-auto-orient'] if auto_orient else [] args += [input_file, output_file] logger.debug("Execute: " + " ".join(args), extra={'group': logging_group}) @@ -142,6 +143,70 @@ def run_convert(input_file, raise ParseError("Convert failed at {}".format(args)) +def get_default_thumbnail(): + return os.path.join(os.path.dirname(__file__), "resources", "document.png") + + +def make_thumbnail_from_pdf_gs_fallback(in_path, temp_dir, logging_group=None): + out_path = os.path.join(temp_dir, "convert_gs.png") + + # if convert fails, fall back to extracting + # the first PDF page as a PNG using Ghostscript + logger.warning( + "Thumbnail generation with ImageMagick failed, falling back " + "to ghostscript. Check your /etc/ImageMagick-x/policy.xml!", + extra={'group': logging_group} + ) + gs_out_path = os.path.join(temp_dir, "gs_out.png") + cmd = [settings.GS_BINARY, + "-q", + "-sDEVICE=pngalpha", + "-o", gs_out_path, + in_path] + try: + if not subprocess.Popen(cmd).wait() == 0: + raise ParseError("Thumbnail (gs) failed at {}".format(cmd)) + # then run convert on the output from gs + run_convert(density=300, + scale="500x5000>", + alpha="remove", + strip=True, + trim=False, + auto_orient=True, + input_file=gs_out_path, + output_file=out_path, + logging_group=logging_group) + + return out_path + + except ParseError: + return get_default_thumbnail() + + +def make_thumbnail_from_pdf(in_path, temp_dir, logging_group=None): + """ + The thumbnail of a PDF is just a 500px wide image of the first page. + """ + out_path = os.path.join(temp_dir, "convert.png") + + # Run convert to get a decent thumbnail + try: + run_convert(density=300, + scale="500x5000>", + alpha="remove", + strip=True, + trim=False, + auto_orient=True, + input_file="{}[0]".format(in_path), + output_file=out_path, + logging_group=logging_group) + except ParseError: + out_path = make_thumbnail_from_pdf_gs_fallback( + in_path, temp_dir, logging_group) + + return out_path + + def parse_date(filename, text): """ Returns the date of the document. @@ -151,6 +216,8 @@ def parse_date(filename, text): """ Call dateparser.parse with a particular date ordering """ + import dateparser + return dateparser.parse( ds, settings={ @@ -161,9 +228,14 @@ def parse_date(filename, text): } ) - date = None + def __filter(date): + if date and date.year > 1900 and \ + date <= timezone.now() and \ + date.date() not in settings.IGNORE_DATES: + return date + return None - next_year = timezone.now().year + 5 # Arbitrary 5 year future limit + date = None # if filename date parsing is enabled, search there first: if settings.FILENAME_DATE_ORDER: @@ -176,7 +248,8 @@ def parse_date(filename, text): # Skip all matches that do not parse to a proper date continue - if date is not None and next_year > date.year > 1900: + date = __filter(date) + if date is not None: return date # Iterate through all regex matches in text and try to parse the date @@ -189,10 +262,9 @@ def parse_date(filename, text): # Skip all matches that do not parse to a proper date continue - if date is not None and next_year > date.year > 1900: + date = __filter(date) + if date is not None: break - else: - date = None return date @@ -207,7 +279,9 @@ class DocumentParser(LoggingMixin): `paperless_tesseract.parsers` for inspiration. """ - def __init__(self, logging_group): + logging_name = "paperless.parsing" + + def __init__(self, logging_group, progress_callback=None): super().__init__() self.logging_group = logging_group os.makedirs(settings.SCRATCH_DIR, exist_ok=True) @@ -217,24 +291,32 @@ class DocumentParser(LoggingMixin): self.archive_path = None self.text = None self.date = None + self.progress_callback = progress_callback + + def progress(self, current_progress, max_progress): + if self.progress_callback: + self.progress_callback(current_progress, max_progress) def extract_metadata(self, document_path, mime_type): return [] - def parse(self, document_path, mime_type): + def parse(self, document_path, mime_type, file_name=None): raise NotImplementedError() def get_archive_path(self): return self.archive_path - def get_thumbnail(self, document_path, mime_type): + def get_thumbnail(self, document_path, mime_type, file_name=None): """ Returns the path to a file we can use as a thumbnail for this document. """ raise NotImplementedError() - def get_optimised_thumbnail(self, document_path, mime_type): - thumbnail = self.get_thumbnail(document_path, mime_type) + def get_optimised_thumbnail(self, + document_path, + mime_type, + file_name=None): + thumbnail = self.get_thumbnail(document_path, mime_type, file_name) if settings.OPTIMIZE_THUMBNAILS: out_path = os.path.join(self.tempdir, "thumb_optipng.png") @@ -257,5 +339,5 @@ class DocumentParser(LoggingMixin): return self.date def cleanup(self): - self.log("debug", "Deleting directory {}".format(self.tempdir)) + self.log("debug", f"Deleting directory {self.tempdir}") shutil.rmtree(self.tempdir) diff --git a/src/documents/resources/document.png b/src/documents/resources/document.png new file mode 100644 index 000000000..164afd14d Binary files /dev/null and b/src/documents/resources/document.png differ diff --git a/src/documents/sanity_checker.py b/src/documents/sanity_checker.py index bc0b689d4..26467d3cf 100644 --- a/src/documents/sanity_checker.py +++ b/src/documents/sanity_checker.py @@ -1,45 +1,55 @@ import hashlib +import logging import os from django.conf import settings +from tqdm import tqdm from documents.models import Document -class SanityMessage: - message = None +class SanityCheckMessages: + + def __init__(self): + self._messages = [] + + def error(self, message): + self._messages.append({"level": logging.ERROR, "message": message}) + + def warning(self, message): + self._messages.append({"level": logging.WARNING, "message": message}) + + def info(self, message): + self._messages.append({"level": logging.INFO, "message": message}) + + def log_messages(self): + logger = logging.getLogger("paperless.sanity_checker") + + if len(self._messages) == 0: + logger.info("Sanity checker detected no issues.") + else: + for msg in self._messages: + logger.log(msg['level'], msg['message']) + + def __len__(self): + return len(self._messages) + + def __getitem__(self, item): + return self._messages[item] + + def has_error(self): + return any([msg['level'] == logging.ERROR for msg in self._messages]) + + def has_warning(self): + return any([msg['level'] == logging.WARNING for msg in self._messages]) -class SanityWarning(SanityMessage): - def __init__(self, message): - self.message = message - - def __str__(self): - return f"Warning: {self.message}" +class SanityCheckFailedException(Exception): + pass -class SanityError(SanityMessage): - def __init__(self, message): - self.message = message - - def __str__(self): - return f"ERROR: {self.message}" - - -class SanityFailedError(Exception): - - def __init__(self, messages): - self.messages = messages - - def __str__(self): - message_string = "\n".join([str(m) for m in self.messages]) - return ( - f"The following issuse were found by the sanity checker:\n" - f"{message_string}\n\n===============\n\n") - - -def check_sanity(): - messages = [] +def check_sanity(progress=False): + messages = SanityCheckMessages() present_files = [] for root, subdirs, files in os.walk(settings.MEDIA_ROOT): @@ -50,72 +60,81 @@ def check_sanity(): if lockfile in present_files: present_files.remove(lockfile) - for doc in Document.objects.all(): + for doc in tqdm(Document.objects.all(), disable=not progress): # Check sanity of the thumbnail if not os.path.isfile(doc.thumbnail_path): - messages.append(SanityError( - f"Thumbnail of document {doc.pk} does not exist.")) + messages.error(f"Thumbnail of document {doc.pk} does not exist.") else: - present_files.remove(os.path.normpath(doc.thumbnail_path)) + if os.path.normpath(doc.thumbnail_path) in present_files: + present_files.remove(os.path.normpath(doc.thumbnail_path)) try: with doc.thumbnail_file as f: f.read() except OSError as e: - messages.append(SanityError( + messages.error( f"Cannot read thumbnail file of document {doc.pk}: {e}" - )) + ) # Check sanity of the original file # TODO: extract method if not os.path.isfile(doc.source_path): - messages.append(SanityError( - f"Original of document {doc.pk} does not exist.")) + messages.error(f"Original of document {doc.pk} does not exist.") else: - present_files.remove(os.path.normpath(doc.source_path)) + if os.path.normpath(doc.source_path) in present_files: + present_files.remove(os.path.normpath(doc.source_path)) try: with doc.source_file as f: checksum = hashlib.md5(f.read()).hexdigest() except OSError as e: - messages.append(SanityError( - f"Cannot read original file of document {doc.pk}: {e}")) + messages.error( + f"Cannot read original file of document {doc.pk}: {e}") else: if not checksum == doc.checksum: - messages.append(SanityError( + messages.error( f"Checksum mismatch of document {doc.pk}. " f"Stored: {doc.checksum}, actual: {checksum}." - )) + ) # Check sanity of the archive file. - if doc.archive_checksum: + if doc.archive_checksum and not doc.archive_filename: + messages.error( + f"Document {doc.pk} has an archive file checksum, but no " + f"archive filename." + ) + elif not doc.archive_checksum and doc.archive_filename: + messages.error( + f"Document {doc.pk} has an archive file, but its checksum is " + f"missing." + ) + elif doc.has_archive_version: if not os.path.isfile(doc.archive_path): - messages.append(SanityError( + messages.error( f"Archived version of document {doc.pk} does not exist." - )) + ) else: - present_files.remove(os.path.normpath(doc.archive_path)) + if os.path.normpath(doc.archive_path) in present_files: + present_files.remove(os.path.normpath(doc.archive_path)) try: with doc.archive_file as f: checksum = hashlib.md5(f.read()).hexdigest() except OSError as e: - messages.append(SanityError( + messages.error( f"Cannot read archive file of document {doc.pk}: {e}" - )) + ) else: if not checksum == doc.archive_checksum: - messages.append(SanityError( - f"Checksum mismatch of archive {doc.pk}. " - f"Stored: {doc.checksum}, actual: {checksum}." - )) + messages.error( + f"Checksum mismatch of archived document " + f"{doc.pk}. " + f"Stored: {doc.archive_checksum}, " + f"actual: {checksum}." + ) # other document checks if not doc.content: - messages.append(SanityWarning( - f"Document {doc.pk} has no content." - )) + messages.info(f"Document {doc.pk} has no content.") for extra_file in present_files: - messages.append(SanityWarning( - f"Orphaned file in media dir: {extra_file}" - )) + messages.warning(f"Orphaned file in media dir: {extra_file}") return messages diff --git a/src/documents/serialisers.py b/src/documents/serialisers.py index db0e610d1..3641c73a5 100644 --- a/src/documents/serialisers.py +++ b/src/documents/serialisers.py @@ -1,22 +1,65 @@ +import re + import magic +import math from django.utils.text import slugify from rest_framework import serializers from rest_framework.fields import SerializerMethodField -from .models import Correspondent, Tag, Document, Log, DocumentType +from . import bulk_edit +from .models import Correspondent, Tag, Document, DocumentType, \ + SavedView, SavedViewFilterRule, MatchingModel from .parsers import is_mime_type_supported +from django.utils.translation import gettext as _ -class CorrespondentSerializer(serializers.ModelSerializer): + +# https://www.django-rest-framework.org/api-guide/serializers/#example +class DynamicFieldsModelSerializer(serializers.ModelSerializer): + """ + A ModelSerializer that takes an additional `fields` argument that + controls which fields should be displayed. + """ + + def __init__(self, *args, **kwargs): + # Don't pass the 'fields' arg up to the superclass + fields = kwargs.pop('fields', None) + + # Instantiate the superclass normally + super(DynamicFieldsModelSerializer, self).__init__(*args, **kwargs) + + if fields is not None: + # Drop any fields that are not specified in the `fields` argument. + allowed = set(fields) + existing = set(self.fields) + for field_name in existing - allowed: + self.fields.pop(field_name) + + +class MatchingModelSerializer(serializers.ModelSerializer): document_count = serializers.IntegerField(read_only=True) - last_correspondence = serializers.DateTimeField(read_only=True) - def get_slug(self, obj): return slugify(obj.name) slug = SerializerMethodField() + def validate_match(self, match): + if 'matching_algorithm' in self.initial_data and self.initial_data['matching_algorithm'] == MatchingModel.MATCH_REGEX: # NOQA: E501 + try: + re.compile(match) + except Exception as e: + raise serializers.ValidationError( + _("Invalid regular expression: %(error)s") % + {'error': str(e)} + ) + return match + + +class CorrespondentSerializer(MatchingModelSerializer): + + last_correspondence = serializers.DateTimeField(read_only=True) + class Meta: model = Correspondent fields = ( @@ -31,13 +74,7 @@ class CorrespondentSerializer(serializers.ModelSerializer): ) -class DocumentTypeSerializer(serializers.ModelSerializer): - - document_count = serializers.IntegerField(read_only=True) - - def get_slug(self, obj): - return slugify(obj.name) - slug = SerializerMethodField() +class DocumentTypeSerializer(MatchingModelSerializer): class Meta: model = DocumentType @@ -52,13 +89,40 @@ class DocumentTypeSerializer(serializers.ModelSerializer): ) -class TagSerializer(serializers.ModelSerializer): +class ColorField(serializers.Field): - document_count = serializers.IntegerField(read_only=True) + COLOURS = ( + (1, "#a6cee3"), + (2, "#1f78b4"), + (3, "#b2df8a"), + (4, "#33a02c"), + (5, "#fb9a99"), + (6, "#e31a1c"), + (7, "#fdbf6f"), + (8, "#ff7f00"), + (9, "#cab2d6"), + (10, "#6a3d9a"), + (11, "#b15928"), + (12, "#000000"), + (13, "#cccccc") + ) - def get_slug(self, obj): - return slugify(obj.name) - slug = SerializerMethodField() + def to_internal_value(self, data): + for id, color in self.COLOURS: + if id == data: + return color + raise serializers.ValidationError() + + def to_representation(self, value): + for id, color in self.COLOURS: + if color == value: + return id + return 1 + + +class TagSerializerVersion1(MatchingModelSerializer): + + colour = ColorField(source='color', default="#a6cee3") class Meta: model = Tag @@ -75,6 +139,45 @@ class TagSerializer(serializers.ModelSerializer): ) +class TagSerializer(MatchingModelSerializer): + + def get_text_color(self, obj): + try: + h = obj.color.lstrip('#') + rgb = tuple(int(h[i:i + 2], 16)/256 for i in (0, 2, 4)) + luminance = math.sqrt( + 0.299 * math.pow(rgb[0], 2) + + 0.587 * math.pow(rgb[1], 2) + + 0.114 * math.pow(rgb[2], 2) + ) + return "#ffffff" if luminance < 0.53 else "#000000" + except ValueError: + return "#000000" + + text_color = serializers.SerializerMethodField() + + class Meta: + model = Tag + fields = ( + "id", + "slug", + "name", + "color", + "text_color", + "match", + "matching_algorithm", + "is_insensitive", + "is_inbox_tag", + "document_count" + ) + + def validate_color(self, color): + regex = r"#[0-9a-fA-F]{6}" + if not re.match(regex, color): + raise serializers.ValidationError(_("Invalid color.")) + return color + + class CorrespondentField(serializers.PrimaryKeyRelatedField): def get_queryset(self): return Correspondent.objects.all() @@ -90,7 +193,7 @@ class DocumentTypeField(serializers.PrimaryKeyRelatedField): return DocumentType.objects.all() -class DocumentSerializer(serializers.ModelSerializer): +class DocumentSerializer(DynamicFieldsModelSerializer): correspondent = CorrespondentField(allow_null=True) tags = TagsField(many=True) @@ -103,7 +206,7 @@ class DocumentSerializer(serializers.ModelSerializer): return obj.get_public_filename() def get_archived_file_name(self, obj): - if obj.archive_checksum: + if obj.has_archive_version: return obj.get_public_filename(archive=True) else: return None @@ -127,17 +230,179 @@ class DocumentSerializer(serializers.ModelSerializer): ) -class LogSerializer(serializers.ModelSerializer): +class SavedViewFilterRuleSerializer(serializers.ModelSerializer): class Meta: - model = Log - fields = ( - "id", - "created", - "message", - "group", - "level" - ) + model = SavedViewFilterRule + fields = ["rule_type", "value"] + + +class SavedViewSerializer(serializers.ModelSerializer): + + filter_rules = SavedViewFilterRuleSerializer(many=True) + + class Meta: + model = SavedView + depth = 1 + fields = ["id", "name", "show_on_dashboard", "show_in_sidebar", + "sort_field", "sort_reverse", "filter_rules"] + + def update(self, instance, validated_data): + if 'filter_rules' in validated_data: + rules_data = validated_data.pop('filter_rules') + else: + rules_data = None + super(SavedViewSerializer, self).update(instance, validated_data) + if rules_data is not None: + SavedViewFilterRule.objects.filter(saved_view=instance).delete() + for rule_data in rules_data: + SavedViewFilterRule.objects.create( + saved_view=instance, **rule_data) + return instance + + def create(self, validated_data): + rules_data = validated_data.pop('filter_rules') + saved_view = SavedView.objects.create(**validated_data) + for rule_data in rules_data: + SavedViewFilterRule.objects.create( + saved_view=saved_view, **rule_data) + return saved_view + + +class DocumentListSerializer(serializers.Serializer): + + documents = serializers.ListField( + required=True, + label="Documents", + write_only=True, + child=serializers.IntegerField() + ) + + def _validate_document_id_list(self, documents, name="documents"): + if not type(documents) == list: + raise serializers.ValidationError(f"{name} must be a list") + if not all([type(i) == int for i in documents]): + raise serializers.ValidationError( + f"{name} must be a list of integers") + count = Document.objects.filter(id__in=documents).count() + if not count == len(documents): + raise serializers.ValidationError( + f"Some documents in {name} don't exist or were " + f"specified twice.") + + def validate_documents(self, documents): + self._validate_document_id_list(documents) + return documents + + +class BulkEditSerializer(DocumentListSerializer): + + method = serializers.ChoiceField( + choices=[ + "set_correspondent", + "set_document_type", + "add_tag", + "remove_tag", + "modify_tags", + "delete" + ], + label="Method", + write_only=True, + ) + + parameters = serializers.DictField(allow_empty=True) + + def _validate_tag_id_list(self, tags, name="tags"): + if not type(tags) == list: + raise serializers.ValidationError(f"{name} must be a list") + if not all([type(i) == int for i in tags]): + raise serializers.ValidationError( + f"{name} must be a list of integers") + count = Tag.objects.filter(id__in=tags).count() + if not count == len(tags): + raise serializers.ValidationError( + f"Some tags in {name} don't exist or were specified twice.") + + def validate_method(self, method): + if method == "set_correspondent": + return bulk_edit.set_correspondent + elif method == "set_document_type": + return bulk_edit.set_document_type + elif method == "add_tag": + return bulk_edit.add_tag + elif method == "remove_tag": + return bulk_edit.remove_tag + elif method == "modify_tags": + return bulk_edit.modify_tags + elif method == "delete": + return bulk_edit.delete + else: + raise serializers.ValidationError("Unsupported method.") + + def _validate_parameters_tags(self, parameters): + if 'tag' in parameters: + tag_id = parameters['tag'] + try: + Tag.objects.get(id=tag_id) + except Tag.DoesNotExist: + raise serializers.ValidationError("Tag does not exist") + else: + raise serializers.ValidationError("tag not specified") + + def _validate_parameters_document_type(self, parameters): + if 'document_type' in parameters: + document_type_id = parameters['document_type'] + if document_type_id is None: + # None is ok + return + try: + DocumentType.objects.get(id=document_type_id) + except DocumentType.DoesNotExist: + raise serializers.ValidationError( + "Document type does not exist") + else: + raise serializers.ValidationError("document_type not specified") + + def _validate_parameters_correspondent(self, parameters): + if 'correspondent' in parameters: + correspondent_id = parameters['correspondent'] + if correspondent_id is None: + return + try: + Correspondent.objects.get(id=correspondent_id) + except Correspondent.DoesNotExist: + raise serializers.ValidationError( + "Correspondent does not exist") + else: + raise serializers.ValidationError("correspondent not specified") + + def _validate_parameters_modify_tags(self, parameters): + if "add_tags" in parameters: + self._validate_tag_id_list(parameters['add_tags'], "add_tags") + else: + raise serializers.ValidationError("add_tags not specified") + + if "remove_tags" in parameters: + self._validate_tag_id_list(parameters['remove_tags'], + "remove_tags") + else: + raise serializers.ValidationError("remove_tags not specified") + + def validate(self, attrs): + + method = attrs['method'] + parameters = attrs['parameters'] + + if method == bulk_edit.set_correspondent: + self._validate_parameters_correspondent(parameters) + elif method == bulk_edit.set_document_type: + self._validate_parameters_document_type(parameters) + elif method == bulk_edit.add_tag or method == bulk_edit.remove_tag: + self._validate_parameters_tags(parameters) + elif method == bulk_edit.modify_tags: + self._validate_parameters_modify_tags(parameters) + + return attrs class PostDocumentSerializer(serializers.Serializer): @@ -183,17 +448,12 @@ class PostDocumentSerializer(serializers.Serializer): if not is_mime_type_supported(mime_type): raise serializers.ValidationError( - "This file type is not supported.") + _("File type %(type)s not supported") % + {'type': mime_type} + ) return document.name, document_data - def validate_title(self, title): - if title: - return title - else: - # do not return empty strings. - return None - def validate_correspondent(self, correspondent): if correspondent: return correspondent.id @@ -211,3 +471,26 @@ class PostDocumentSerializer(serializers.Serializer): return [tag.id for tag in tags] else: return None + + +class BulkDownloadSerializer(DocumentListSerializer): + + content = serializers.ChoiceField( + choices=["archive", "originals", "both"], + default="archive" + ) + + compression = serializers.ChoiceField( + choices=["none", "deflated", "bzip2", "lzma"], + default="none" + ) + + def validate_compression(self, compression): + import zipfile + + return { + "none": zipfile.ZIP_STORED, + "deflated": zipfile.ZIP_DEFLATED, + "bzip2": zipfile.ZIP_BZIP2, + "lzma": zipfile.ZIP_LZMA + }[compression] diff --git a/src/documents/signals/handlers.py b/src/documents/signals/handlers.py old mode 100755 new mode 100644 index 586897585..8fb8faf51 --- a/src/documents/signals/handlers.py +++ b/src/documents/signals/handlers.py @@ -1,7 +1,7 @@ import logging import os -from subprocess import Popen +from django.utils import termcolors from django.conf import settings from django.contrib.admin.models import ADDITION, LogEntry from django.contrib.auth.models import User @@ -9,19 +9,17 @@ from django.contrib.contenttypes.models import ContentType from django.db import models, DatabaseError from django.db.models import Q from django.dispatch import receiver -from django.utils import timezone +from django.utils import termcolors, timezone from filelock import FileLock -from rest_framework.reverse import reverse -from .. import index, matching +from .. import matching from ..file_handling import delete_empty_directories, \ - create_source_path_directory, archive_name_from_filename, \ + create_source_path_directory, \ generate_unique_filename -from ..models import Document, Tag +from ..models import Document, Tag, MatchingModel -def logger(message, group): - logging.getLogger(__name__).debug(message, extra={"group": group}) +logger = logging.getLogger("paperless.handlers") def add_inbox_tags(sender, document=None, logging_group=None, **kwargs): @@ -35,11 +33,14 @@ def set_correspondent(sender, classifier=None, replace=False, use_first=True, + suggest=False, + base_url=None, + color=False, **kwargs): if document.correspondent and not replace: return - potential_correspondents = matching.match_correspondents(document.content, + potential_correspondents = matching.match_correspondents(document, classifier) potential_count = len(potential_correspondents) @@ -49,27 +50,45 @@ def set_correspondent(sender, selected = None if potential_count > 1: if use_first: - logger( + logger.debug( f"Detected {potential_count} potential correspondents, " f"so we've opted for {selected}", - logging_group + extra={'group': logging_group} ) else: - logger( + logger.debug( f"Detected {potential_count} potential correspondents, " f"not assigning any correspondent", - logging_group + extra={'group': logging_group} ) return if selected or replace: - logger( - f"Assigning correspondent {selected} to {document}", - logging_group - ) + if suggest: + if base_url: + print( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + print(f"{base_url}/documents/{document.pk}") + else: + print( + ( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + f" [{document.pk}]" + ) + print(f"Suggest correspondent {selected}") + else: + logger.info( + f"Assigning correspondent {selected} to {document}", + extra={'group': logging_group} + ) - document.correspondent = selected - document.save(update_fields=("correspondent",)) + document.correspondent = selected + document.save(update_fields=("correspondent",)) def set_document_type(sender, @@ -78,11 +97,14 @@ def set_document_type(sender, classifier=None, replace=False, use_first=True, + suggest=False, + base_url=None, + color=False, **kwargs): if document.document_type and not replace: return - potential_document_type = matching.match_document_types(document.content, + potential_document_type = matching.match_document_types(document, classifier) potential_count = len(potential_document_type) @@ -93,27 +115,45 @@ def set_document_type(sender, if potential_count > 1: if use_first: - logger( + logger.info( f"Detected {potential_count} potential document types, " f"so we've opted for {selected}", - logging_group + extra={'group': logging_group} ) else: - logger( + logger.info( f"Detected {potential_count} potential document types, " f"not assigning any document type", - logging_group + extra={'group': logging_group} ) return if selected or replace: - logger( - f"Assigning document type {selected} to {document}", - logging_group - ) + if suggest: + if base_url: + print( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + print(f"{base_url}/documents/{document.pk}") + else: + print( + ( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + f" [{document.pk}]" + ) + print(f"Sugest document type {selected}") + else: + logger.info( + f"Assigning document type {selected} to {document}", + extra={'group': logging_group} + ) - document.document_type = selected - document.save(update_fields=("document_type",)) + document.document_type = selected + document.save(update_fields=("document_type",)) def set_tags(sender, @@ -121,6 +161,9 @@ def set_tags(sender, logging_group=None, classifier=None, replace=False, + suggest=False, + base_url=None, + color=False, **kwargs): if replace: @@ -131,63 +174,69 @@ def set_tags(sender, current_tags = set(document.tags.all()) - matched_tags = matching.match_tags(document.content, classifier) + matched_tags = matching.match_tags(document, classifier) relevant_tags = set(matched_tags) - current_tags - if not relevant_tags: - return + if suggest: + extra_tags = current_tags - set(matched_tags) + extra_tags = [ + t for t in extra_tags + if t.matching_algorithm == MatchingModel.MATCH_AUTO + ] + if not relevant_tags and not extra_tags: + return + if base_url: + print( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + print(f"{base_url}/documents/{document.pk}") + else: + print( + ( + termcolors.colorize(str(document), fg='green') + if color + else str(document) + ) + f" [{document.pk}]" + ) + if relevant_tags: + print( + "Suggest tags: " + ", ".join([t.name for t in relevant_tags]) + ) + if extra_tags: + print("Extra tags: " + ", ".join([t.name for t in extra_tags])) + else: + if not relevant_tags: + return - message = 'Tagging "{}" with "{}"' - logger( - message.format(document, ", ".join([t.name for t in relevant_tags])), - logging_group - ) + message = 'Tagging "{}" with "{}"' + logger.info( + message.format( + document, ", ".join([t.name for t in relevant_tags]) + ), + extra={'group': logging_group} + ) - document.tags.add(*relevant_tags) - - -def run_pre_consume_script(sender, filename, **kwargs): - - if not settings.PRE_CONSUME_SCRIPT: - return - - Popen((settings.PRE_CONSUME_SCRIPT, filename)).wait() - - -def run_post_consume_script(sender, document, **kwargs): - - if not settings.POST_CONSUME_SCRIPT: - return - - Popen(( - settings.POST_CONSUME_SCRIPT, - str(document.pk), - document.get_public_filename(), - os.path.normpath(document.source_path), - os.path.normpath(document.thumbnail_path), - reverse("document-download", kwargs={"pk": document.pk}), - reverse("document-thumb", kwargs={"pk": document.pk}), - str(document.correspondent), - str(",".join(document.tags.all().values_list("name", flat=True))) - )).wait() + document.tags.add(*relevant_tags) @receiver(models.signals.post_delete, sender=Document) def cleanup_document_deletion(sender, instance, using, **kwargs): with FileLock(settings.MEDIA_LOCK): - for f in (instance.source_path, - instance.archive_path, - instance.thumbnail_path): - if os.path.isfile(f): + for filename in (instance.source_path, + instance.archive_path, + instance.thumbnail_path): + if filename and os.path.isfile(filename): try: - os.unlink(f) - logging.getLogger(__name__).debug( - f"Deleted file {f}.") + os.unlink(filename) + logger.debug( + f"Deleted file {filename}.") except OSError as e: - logging.getLogger(__name__).warning( + logger.warning( f"While deleting document {str(instance)}, the file " - f"{f} could not be deleted: {e}" + f"{filename} could not be deleted: {e}" ) delete_empty_directories( @@ -195,27 +244,30 @@ def cleanup_document_deletion(sender, instance, using, **kwargs): root=settings.ORIGINALS_DIR ) - delete_empty_directories( - os.path.dirname(instance.archive_path), - root=settings.ARCHIVE_DIR - ) + if instance.has_archive_version: + delete_empty_directories( + os.path.dirname(instance.archive_path), + root=settings.ARCHIVE_DIR + ) + + +class CannotMoveFilesException(Exception): + pass def validate_move(instance, old_path, new_path): if not os.path.isfile(old_path): # Can't do anything if the old file does not exist anymore. - logging.getLogger(__name__).fatal( + logger.fatal( f"Document {str(instance)}: File {old_path} has gone.") - return False + raise CannotMoveFilesException() if os.path.isfile(new_path): # Can't do anything if the new file already exists. Skip updating file. - logging.getLogger(__name__).warning( + logger.warning( f"Document {str(instance)}: Cannot rename file " f"since target path {new_path} already exists.") - return False - - return True + raise CannotMoveFilesException() @receiver(models.signals.m2m_changed, sender=Document.tags.through) @@ -234,63 +286,61 @@ def update_filename_and_move_files(sender, instance, **kwargs): return with FileLock(settings.MEDIA_LOCK): - old_filename = instance.filename - new_filename = generate_unique_filename( - instance, settings.ORIGINALS_DIR) + try: + old_filename = instance.filename + old_source_path = instance.source_path - if new_filename == instance.filename: - # Don't do anything if its the same. - return + instance.filename = generate_unique_filename(instance) + move_original = old_filename != instance.filename - old_source_path = instance.source_path - new_source_path = os.path.join(settings.ORIGINALS_DIR, new_filename) - - if not validate_move(instance, old_source_path, new_source_path): - return - - # archive files are optional, archive checksum tells us if we have one, - # since this is None for documents without archived files. - if instance.archive_checksum: - new_archive_filename = archive_name_from_filename(new_filename) + old_archive_filename = instance.archive_filename old_archive_path = instance.archive_path - new_archive_path = os.path.join(settings.ARCHIVE_DIR, - new_archive_filename) - if not validate_move(instance, old_archive_path, new_archive_path): + if instance.has_archive_version: + + instance.archive_filename = generate_unique_filename( + instance, archive_filename=True + ) + + move_archive = old_archive_filename != instance.archive_filename # NOQA: E501 + else: + move_archive = False + + if not move_original and not move_archive: + # Don't do anything if filenames did not change. return - create_source_path_directory(new_archive_path) - else: - old_archive_path = None - new_archive_path = None + if move_original: + validate_move(instance, old_source_path, instance.source_path) + create_source_path_directory(instance.source_path) + os.rename(old_source_path, instance.source_path) - create_source_path_directory(new_source_path) - - try: - os.rename(old_source_path, new_source_path) - if instance.archive_checksum: - os.rename(old_archive_path, new_archive_path) - instance.filename = new_filename + if move_archive: + validate_move( + instance, old_archive_path, instance.archive_path) + create_source_path_directory(instance.archive_path) + os.rename(old_archive_path, instance.archive_path) # Don't save() here to prevent infinite recursion. Document.objects.filter(pk=instance.pk).update( - filename=new_filename) + filename=instance.filename, + archive_filename=instance.archive_filename, + ) - logging.getLogger(__name__).debug( - f"Moved file {old_source_path} to {new_source_path}.") + except (OSError, DatabaseError, CannotMoveFilesException): + # This happens when either: + # - moving the files failed due to file system errors + # - saving to the database failed due to database errors + # In both cases, we need to revert to the original state. - if instance.archive_checksum: - logging.getLogger(__name__).debug( - f"Moved file {old_archive_path} to {new_archive_path}.") - - except OSError as e: - instance.filename = old_filename - # this happens when we can't move a file. If that's the case for - # the archive file, we try our best to revert the changes. - # no need to save the instance, the update() has not happened yet. + # Try to move files to their original location. try: - os.rename(new_source_path, old_source_path) - os.rename(new_archive_path, old_archive_path) + if move_original and os.path.isfile(instance.source_path): + os.rename(instance.source_path, old_source_path) + + if move_archive and os.path.isfile(instance.archive_path): + os.rename(instance.archive_path, old_archive_path) + except Exception as e: # This is fine, since: # A: if we managed to move source from A to B, we will also @@ -301,16 +351,10 @@ def update_filename_and_move_files(sender, instance, **kwargs): # B: if moving the orignal file failed, nothing has changed # anyway. pass - except DatabaseError as e: - # this happens after moving files, so move them back into place. - # since moving them once succeeded, it's very likely going to - # succeed again. - os.rename(new_source_path, old_source_path) - if instance.archive_checksum: - os.rename(new_archive_path, old_archive_path) + + # restore old values on the instance instance.filename = old_filename - # again, no need to save the instance, since the actual update() - # operation failed. + instance.archive_filename = old_archive_filename # finally, remove any empty sub folders. This will do nothing if # something has failed above. @@ -318,7 +362,7 @@ def update_filename_and_move_files(sender, instance, **kwargs): delete_empty_directories(os.path.dirname(old_source_path), root=settings.ORIGINALS_DIR) - if old_archive_path and not os.path.isfile(old_archive_path): + if instance.has_archive_version and not os.path.isfile(old_archive_path): # NOQA: E501 delete_empty_directories(os.path.dirname(old_archive_path), root=settings.ARCHIVE_DIR) @@ -339,4 +383,6 @@ def set_log_entry(sender, document=None, logging_group=None, **kwargs): def add_to_index(sender, document, **kwargs): + from documents import index + index.add_or_update_document(document) diff --git a/src/documents/static/bootstrap.min.css b/src/documents/static/bootstrap.min.css index 286cde4c0..6bcea10ad 100644 --- a/src/documents/static/bootstrap.min.css +++ b/src/documents/static/bootstrap.min.css @@ -4,4 +4,4 @@ * Copyright 2011-2020 Twitter, Inc. * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) */:root{--blue:#007bff;--indigo:#6610f2;--purple:#6f42c1;--pink:#e83e8c;--red:#dc3545;--orange:#fd7e14;--yellow:#ffc107;--green:#28a745;--teal:#20c997;--cyan:#17a2b8;--white:#fff;--gray:#6c757d;--gray-dark:#343a40;--primary:#007bff;--secondary:#6c757d;--success:#28a745;--info:#17a2b8;--warning:#ffc107;--danger:#dc3545;--light:#f8f9fa;--dark:#343a40;--breakpoint-xs:0;--breakpoint-sm:576px;--breakpoint-md:768px;--breakpoint-lg:992px;--breakpoint-xl:1200px;--font-family-sans-serif:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";--font-family-monospace:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace}*,::after,::before{box-sizing:border-box}html{font-family:sans-serif;line-height:1.15;-webkit-text-size-adjust:100%;-webkit-tap-highlight-color:transparent}article,aside,figcaption,figure,footer,header,hgroup,main,nav,section{display:block}body{margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-size:1rem;font-weight:400;line-height:1.5;color:#212529;text-align:left;background-color:#fff}[tabindex="-1"]:focus:not(:focus-visible){outline:0!important}hr{box-sizing:content-box;height:0;overflow:visible}h1,h2,h3,h4,h5,h6{margin-top:0;margin-bottom:.5rem}p{margin-top:0;margin-bottom:1rem}abbr[data-original-title],abbr[title]{text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted;cursor:help;border-bottom:0;-webkit-text-decoration-skip-ink:none;text-decoration-skip-ink:none}address{margin-bottom:1rem;font-style:normal;line-height:inherit}dl,ol,ul{margin-top:0;margin-bottom:1rem}ol ol,ol ul,ul ol,ul ul{margin-bottom:0}dt{font-weight:700}dd{margin-bottom:.5rem;margin-left:0}blockquote{margin:0 0 1rem}b,strong{font-weight:bolder}small{font-size:80%}sub,sup{position:relative;font-size:75%;line-height:0;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}a{color:#007bff;text-decoration:none;background-color:transparent}a:hover{color:#0056b3;text-decoration:underline}a:not([href]):not([class]){color:inherit;text-decoration:none}a:not([href]):not([class]):hover{color:inherit;text-decoration:none}code,kbd,pre,samp{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace;font-size:1em}pre{margin-top:0;margin-bottom:1rem;overflow:auto;-ms-overflow-style:scrollbar}figure{margin:0 0 1rem}img{vertical-align:middle;border-style:none}svg{overflow:hidden;vertical-align:middle}table{border-collapse:collapse}caption{padding-top:.75rem;padding-bottom:.75rem;color:#6c757d;text-align:left;caption-side:bottom}th{text-align:inherit;text-align:-webkit-match-parent}label{display:inline-block;margin-bottom:.5rem}button{border-radius:0}button:focus{outline:1px dotted;outline:5px auto -webkit-focus-ring-color}button,input,optgroup,select,textarea{margin:0;font-family:inherit;font-size:inherit;line-height:inherit}button,input{overflow:visible}button,select{text-transform:none}[role=button]{cursor:pointer}select{word-wrap:normal}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]:not(:disabled),[type=reset]:not(:disabled),[type=submit]:not(:disabled),button:not(:disabled){cursor:pointer}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{padding:0;border-style:none}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}textarea{overflow:auto;resize:vertical}fieldset{min-width:0;padding:0;margin:0;border:0}legend{display:block;width:100%;max-width:100%;padding:0;margin-bottom:.5rem;font-size:1.5rem;line-height:inherit;color:inherit;white-space:normal}progress{vertical-align:baseline}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{outline-offset:-2px;-webkit-appearance:none}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{font:inherit;-webkit-appearance:button}output{display:inline-block}summary{display:list-item;cursor:pointer}template{display:none}[hidden]{display:none!important}.h1,.h2,.h3,.h4,.h5,.h6,h1,h2,h3,h4,h5,h6{margin-bottom:.5rem;font-weight:500;line-height:1.2}.h1,h1{font-size:2.5rem}.h2,h2{font-size:2rem}.h3,h3{font-size:1.75rem}.h4,h4{font-size:1.5rem}.h5,h5{font-size:1.25rem}.h6,h6{font-size:1rem}.lead{font-size:1.25rem;font-weight:300}.display-1{font-size:6rem;font-weight:300;line-height:1.2}.display-2{font-size:5.5rem;font-weight:300;line-height:1.2}.display-3{font-size:4.5rem;font-weight:300;line-height:1.2}.display-4{font-size:3.5rem;font-weight:300;line-height:1.2}hr{margin-top:1rem;margin-bottom:1rem;border:0;border-top:1px solid rgba(0,0,0,.1)}.small,small{font-size:80%;font-weight:400}.mark,mark{padding:.2em;background-color:#fcf8e3}.list-unstyled{padding-left:0;list-style:none}.list-inline{padding-left:0;list-style:none}.list-inline-item{display:inline-block}.list-inline-item:not(:last-child){margin-right:.5rem}.initialism{font-size:90%;text-transform:uppercase}.blockquote{margin-bottom:1rem;font-size:1.25rem}.blockquote-footer{display:block;font-size:80%;color:#6c757d}.blockquote-footer::before{content:"\2014\00A0"}.img-fluid{max-width:100%;height:auto}.img-thumbnail{padding:.25rem;background-color:#fff;border:1px solid #dee2e6;border-radius:.25rem;max-width:100%;height:auto}.figure{display:inline-block}.figure-img{margin-bottom:.5rem;line-height:1}.figure-caption{font-size:90%;color:#6c757d}code{font-size:87.5%;color:#e83e8c;word-wrap:break-word}a>code{color:inherit}kbd{padding:.2rem .4rem;font-size:87.5%;color:#fff;background-color:#212529;border-radius:.2rem}kbd kbd{padding:0;font-size:100%;font-weight:700}pre{display:block;font-size:87.5%;color:#212529}pre code{font-size:inherit;color:inherit;word-break:normal}.pre-scrollable{max-height:340px;overflow-y:scroll}.container,.container-fluid,.container-lg,.container-md,.container-sm,.container-xl{width:100%;padding-right:15px;padding-left:15px;margin-right:auto;margin-left:auto}@media (min-width:576px){.container,.container-sm{max-width:540px}}@media (min-width:768px){.container,.container-md,.container-sm{max-width:720px}}@media (min-width:992px){.container,.container-lg,.container-md,.container-sm{max-width:960px}}@media (min-width:1200px){.container,.container-lg,.container-md,.container-sm,.container-xl{max-width:1140px}}.row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-15px;margin-left:-15px}.no-gutters{margin-right:0;margin-left:0}.no-gutters>.col,.no-gutters>[class*=col-]{padding-right:0;padding-left:0}.col,.col-1,.col-10,.col-11,.col-12,.col-2,.col-3,.col-4,.col-5,.col-6,.col-7,.col-8,.col-9,.col-auto,.col-lg,.col-lg-1,.col-lg-10,.col-lg-11,.col-lg-12,.col-lg-2,.col-lg-3,.col-lg-4,.col-lg-5,.col-lg-6,.col-lg-7,.col-lg-8,.col-lg-9,.col-lg-auto,.col-md,.col-md-1,.col-md-10,.col-md-11,.col-md-12,.col-md-2,.col-md-3,.col-md-4,.col-md-5,.col-md-6,.col-md-7,.col-md-8,.col-md-9,.col-md-auto,.col-sm,.col-sm-1,.col-sm-10,.col-sm-11,.col-sm-12,.col-sm-2,.col-sm-3,.col-sm-4,.col-sm-5,.col-sm-6,.col-sm-7,.col-sm-8,.col-sm-9,.col-sm-auto,.col-xl,.col-xl-1,.col-xl-10,.col-xl-11,.col-xl-12,.col-xl-2,.col-xl-3,.col-xl-4,.col-xl-5,.col-xl-6,.col-xl-7,.col-xl-8,.col-xl-9,.col-xl-auto{position:relative;width:100%;padding-right:15px;padding-left:15px}.col{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-first{-ms-flex-order:-1;order:-1}.order-last{-ms-flex-order:13;order:13}.order-0{-ms-flex-order:0;order:0}.order-1{-ms-flex-order:1;order:1}.order-2{-ms-flex-order:2;order:2}.order-3{-ms-flex-order:3;order:3}.order-4{-ms-flex-order:4;order:4}.order-5{-ms-flex-order:5;order:5}.order-6{-ms-flex-order:6;order:6}.order-7{-ms-flex-order:7;order:7}.order-8{-ms-flex-order:8;order:8}.order-9{-ms-flex-order:9;order:9}.order-10{-ms-flex-order:10;order:10}.order-11{-ms-flex-order:11;order:11}.order-12{-ms-flex-order:12;order:12}.offset-1{margin-left:8.333333%}.offset-2{margin-left:16.666667%}.offset-3{margin-left:25%}.offset-4{margin-left:33.333333%}.offset-5{margin-left:41.666667%}.offset-6{margin-left:50%}.offset-7{margin-left:58.333333%}.offset-8{margin-left:66.666667%}.offset-9{margin-left:75%}.offset-10{margin-left:83.333333%}.offset-11{margin-left:91.666667%}@media (min-width:576px){.col-sm{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-sm-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-sm-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-sm-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-sm-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-sm-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-sm-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-sm-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-sm-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-sm-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-sm-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-sm-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-sm-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-sm-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-sm-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-sm-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-sm-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-sm-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-sm-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-sm-first{-ms-flex-order:-1;order:-1}.order-sm-last{-ms-flex-order:13;order:13}.order-sm-0{-ms-flex-order:0;order:0}.order-sm-1{-ms-flex-order:1;order:1}.order-sm-2{-ms-flex-order:2;order:2}.order-sm-3{-ms-flex-order:3;order:3}.order-sm-4{-ms-flex-order:4;order:4}.order-sm-5{-ms-flex-order:5;order:5}.order-sm-6{-ms-flex-order:6;order:6}.order-sm-7{-ms-flex-order:7;order:7}.order-sm-8{-ms-flex-order:8;order:8}.order-sm-9{-ms-flex-order:9;order:9}.order-sm-10{-ms-flex-order:10;order:10}.order-sm-11{-ms-flex-order:11;order:11}.order-sm-12{-ms-flex-order:12;order:12}.offset-sm-0{margin-left:0}.offset-sm-1{margin-left:8.333333%}.offset-sm-2{margin-left:16.666667%}.offset-sm-3{margin-left:25%}.offset-sm-4{margin-left:33.333333%}.offset-sm-5{margin-left:41.666667%}.offset-sm-6{margin-left:50%}.offset-sm-7{margin-left:58.333333%}.offset-sm-8{margin-left:66.666667%}.offset-sm-9{margin-left:75%}.offset-sm-10{margin-left:83.333333%}.offset-sm-11{margin-left:91.666667%}}@media (min-width:768px){.col-md{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-md-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-md-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-md-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-md-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-md-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-md-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-md-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-md-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-md-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-md-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-md-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-md-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-md-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-md-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-md-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-md-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-md-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-md-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-md-first{-ms-flex-order:-1;order:-1}.order-md-last{-ms-flex-order:13;order:13}.order-md-0{-ms-flex-order:0;order:0}.order-md-1{-ms-flex-order:1;order:1}.order-md-2{-ms-flex-order:2;order:2}.order-md-3{-ms-flex-order:3;order:3}.order-md-4{-ms-flex-order:4;order:4}.order-md-5{-ms-flex-order:5;order:5}.order-md-6{-ms-flex-order:6;order:6}.order-md-7{-ms-flex-order:7;order:7}.order-md-8{-ms-flex-order:8;order:8}.order-md-9{-ms-flex-order:9;order:9}.order-md-10{-ms-flex-order:10;order:10}.order-md-11{-ms-flex-order:11;order:11}.order-md-12{-ms-flex-order:12;order:12}.offset-md-0{margin-left:0}.offset-md-1{margin-left:8.333333%}.offset-md-2{margin-left:16.666667%}.offset-md-3{margin-left:25%}.offset-md-4{margin-left:33.333333%}.offset-md-5{margin-left:41.666667%}.offset-md-6{margin-left:50%}.offset-md-7{margin-left:58.333333%}.offset-md-8{margin-left:66.666667%}.offset-md-9{margin-left:75%}.offset-md-10{margin-left:83.333333%}.offset-md-11{margin-left:91.666667%}}@media (min-width:992px){.col-lg{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-lg-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-lg-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-lg-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-lg-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-lg-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-lg-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-lg-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-lg-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-lg-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-lg-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-lg-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-lg-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-lg-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-lg-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-lg-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-lg-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-lg-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-lg-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-lg-first{-ms-flex-order:-1;order:-1}.order-lg-last{-ms-flex-order:13;order:13}.order-lg-0{-ms-flex-order:0;order:0}.order-lg-1{-ms-flex-order:1;order:1}.order-lg-2{-ms-flex-order:2;order:2}.order-lg-3{-ms-flex-order:3;order:3}.order-lg-4{-ms-flex-order:4;order:4}.order-lg-5{-ms-flex-order:5;order:5}.order-lg-6{-ms-flex-order:6;order:6}.order-lg-7{-ms-flex-order:7;order:7}.order-lg-8{-ms-flex-order:8;order:8}.order-lg-9{-ms-flex-order:9;order:9}.order-lg-10{-ms-flex-order:10;order:10}.order-lg-11{-ms-flex-order:11;order:11}.order-lg-12{-ms-flex-order:12;order:12}.offset-lg-0{margin-left:0}.offset-lg-1{margin-left:8.333333%}.offset-lg-2{margin-left:16.666667%}.offset-lg-3{margin-left:25%}.offset-lg-4{margin-left:33.333333%}.offset-lg-5{margin-left:41.666667%}.offset-lg-6{margin-left:50%}.offset-lg-7{margin-left:58.333333%}.offset-lg-8{margin-left:66.666667%}.offset-lg-9{margin-left:75%}.offset-lg-10{margin-left:83.333333%}.offset-lg-11{margin-left:91.666667%}}@media (min-width:1200px){.col-xl{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;max-width:100%}.row-cols-xl-1>*{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.row-cols-xl-2>*{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.row-cols-xl-3>*{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.row-cols-xl-4>*{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.row-cols-xl-5>*{-ms-flex:0 0 20%;flex:0 0 20%;max-width:20%}.row-cols-xl-6>*{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-auto{-ms-flex:0 0 auto;flex:0 0 auto;width:auto;max-width:100%}.col-xl-1{-ms-flex:0 0 8.333333%;flex:0 0 8.333333%;max-width:8.333333%}.col-xl-2{-ms-flex:0 0 16.666667%;flex:0 0 16.666667%;max-width:16.666667%}.col-xl-3{-ms-flex:0 0 25%;flex:0 0 25%;max-width:25%}.col-xl-4{-ms-flex:0 0 33.333333%;flex:0 0 33.333333%;max-width:33.333333%}.col-xl-5{-ms-flex:0 0 41.666667%;flex:0 0 41.666667%;max-width:41.666667%}.col-xl-6{-ms-flex:0 0 50%;flex:0 0 50%;max-width:50%}.col-xl-7{-ms-flex:0 0 58.333333%;flex:0 0 58.333333%;max-width:58.333333%}.col-xl-8{-ms-flex:0 0 66.666667%;flex:0 0 66.666667%;max-width:66.666667%}.col-xl-9{-ms-flex:0 0 75%;flex:0 0 75%;max-width:75%}.col-xl-10{-ms-flex:0 0 83.333333%;flex:0 0 83.333333%;max-width:83.333333%}.col-xl-11{-ms-flex:0 0 91.666667%;flex:0 0 91.666667%;max-width:91.666667%}.col-xl-12{-ms-flex:0 0 100%;flex:0 0 100%;max-width:100%}.order-xl-first{-ms-flex-order:-1;order:-1}.order-xl-last{-ms-flex-order:13;order:13}.order-xl-0{-ms-flex-order:0;order:0}.order-xl-1{-ms-flex-order:1;order:1}.order-xl-2{-ms-flex-order:2;order:2}.order-xl-3{-ms-flex-order:3;order:3}.order-xl-4{-ms-flex-order:4;order:4}.order-xl-5{-ms-flex-order:5;order:5}.order-xl-6{-ms-flex-order:6;order:6}.order-xl-7{-ms-flex-order:7;order:7}.order-xl-8{-ms-flex-order:8;order:8}.order-xl-9{-ms-flex-order:9;order:9}.order-xl-10{-ms-flex-order:10;order:10}.order-xl-11{-ms-flex-order:11;order:11}.order-xl-12{-ms-flex-order:12;order:12}.offset-xl-0{margin-left:0}.offset-xl-1{margin-left:8.333333%}.offset-xl-2{margin-left:16.666667%}.offset-xl-3{margin-left:25%}.offset-xl-4{margin-left:33.333333%}.offset-xl-5{margin-left:41.666667%}.offset-xl-6{margin-left:50%}.offset-xl-7{margin-left:58.333333%}.offset-xl-8{margin-left:66.666667%}.offset-xl-9{margin-left:75%}.offset-xl-10{margin-left:83.333333%}.offset-xl-11{margin-left:91.666667%}}.table{width:100%;margin-bottom:1rem;color:#212529}.table td,.table th{padding:.75rem;vertical-align:top;border-top:1px solid #dee2e6}.table thead th{vertical-align:bottom;border-bottom:2px solid #dee2e6}.table tbody+tbody{border-top:2px solid #dee2e6}.table-sm td,.table-sm th{padding:.3rem}.table-bordered{border:1px solid #dee2e6}.table-bordered td,.table-bordered th{border:1px solid #dee2e6}.table-bordered thead td,.table-bordered thead th{border-bottom-width:2px}.table-borderless tbody+tbody,.table-borderless td,.table-borderless th,.table-borderless thead th{border:0}.table-striped tbody tr:nth-of-type(odd){background-color:rgba(0,0,0,.05)}.table-hover tbody tr:hover{color:#212529;background-color:rgba(0,0,0,.075)}.table-primary,.table-primary>td,.table-primary>th{background-color:#b8daff}.table-primary tbody+tbody,.table-primary td,.table-primary th,.table-primary thead th{border-color:#7abaff}.table-hover .table-primary:hover{background-color:#9fcdff}.table-hover .table-primary:hover>td,.table-hover .table-primary:hover>th{background-color:#9fcdff}.table-secondary,.table-secondary>td,.table-secondary>th{background-color:#d6d8db}.table-secondary tbody+tbody,.table-secondary td,.table-secondary th,.table-secondary thead th{border-color:#b3b7bb}.table-hover .table-secondary:hover{background-color:#c8cbcf}.table-hover .table-secondary:hover>td,.table-hover .table-secondary:hover>th{background-color:#c8cbcf}.table-success,.table-success>td,.table-success>th{background-color:#c3e6cb}.table-success tbody+tbody,.table-success td,.table-success th,.table-success thead th{border-color:#8fd19e}.table-hover .table-success:hover{background-color:#b1dfbb}.table-hover .table-success:hover>td,.table-hover .table-success:hover>th{background-color:#b1dfbb}.table-info,.table-info>td,.table-info>th{background-color:#bee5eb}.table-info tbody+tbody,.table-info td,.table-info th,.table-info thead th{border-color:#86cfda}.table-hover .table-info:hover{background-color:#abdde5}.table-hover .table-info:hover>td,.table-hover .table-info:hover>th{background-color:#abdde5}.table-warning,.table-warning>td,.table-warning>th{background-color:#ffeeba}.table-warning tbody+tbody,.table-warning td,.table-warning th,.table-warning thead th{border-color:#ffdf7e}.table-hover .table-warning:hover{background-color:#ffe8a1}.table-hover .table-warning:hover>td,.table-hover .table-warning:hover>th{background-color:#ffe8a1}.table-danger,.table-danger>td,.table-danger>th{background-color:#f5c6cb}.table-danger tbody+tbody,.table-danger td,.table-danger th,.table-danger thead th{border-color:#ed969e}.table-hover .table-danger:hover{background-color:#f1b0b7}.table-hover .table-danger:hover>td,.table-hover .table-danger:hover>th{background-color:#f1b0b7}.table-light,.table-light>td,.table-light>th{background-color:#fdfdfe}.table-light tbody+tbody,.table-light td,.table-light th,.table-light thead th{border-color:#fbfcfc}.table-hover .table-light:hover{background-color:#ececf6}.table-hover .table-light:hover>td,.table-hover .table-light:hover>th{background-color:#ececf6}.table-dark,.table-dark>td,.table-dark>th{background-color:#c6c8ca}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#95999c}.table-hover .table-dark:hover{background-color:#b9bbbe}.table-hover .table-dark:hover>td,.table-hover .table-dark:hover>th{background-color:#b9bbbe}.table-active,.table-active>td,.table-active>th{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover{background-color:rgba(0,0,0,.075)}.table-hover .table-active:hover>td,.table-hover .table-active:hover>th{background-color:rgba(0,0,0,.075)}.table .thead-dark th{color:#fff;background-color:#343a40;border-color:#454d55}.table .thead-light th{color:#495057;background-color:#e9ecef;border-color:#dee2e6}.table-dark{color:#fff;background-color:#343a40}.table-dark td,.table-dark th,.table-dark thead th{border-color:#454d55}.table-dark.table-bordered{border:0}.table-dark.table-striped tbody tr:nth-of-type(odd){background-color:rgba(255,255,255,.05)}.table-dark.table-hover tbody tr:hover{color:#fff;background-color:rgba(255,255,255,.075)}@media (max-width:575.98px){.table-responsive-sm{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-sm>.table-bordered{border:0}}@media (max-width:767.98px){.table-responsive-md{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-md>.table-bordered{border:0}}@media (max-width:991.98px){.table-responsive-lg{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-lg>.table-bordered{border:0}}@media (max-width:1199.98px){.table-responsive-xl{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive-xl>.table-bordered{border:0}}.table-responsive{display:block;width:100%;overflow-x:auto;-webkit-overflow-scrolling:touch}.table-responsive>.table-bordered{border:0}.form-control{display:block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;background-clip:padding-box;border:1px solid #ced4da;border-radius:.25rem;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.form-control{transition:none}}.form-control::-ms-expand{background-color:transparent;border:0}.form-control:-moz-focusring{color:transparent;text-shadow:0 0 0 #495057}.form-control:focus{color:#495057;background-color:#fff;border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.form-control::-webkit-input-placeholder{color:#6c757d;opacity:1}.form-control::-moz-placeholder{color:#6c757d;opacity:1}.form-control:-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::-ms-input-placeholder{color:#6c757d;opacity:1}.form-control::placeholder{color:#6c757d;opacity:1}.form-control:disabled,.form-control[readonly]{background-color:#e9ecef;opacity:1}input[type=date].form-control,input[type=datetime-local].form-control,input[type=month].form-control,input[type=time].form-control{-webkit-appearance:none;-moz-appearance:none;appearance:none}select.form-control:focus::-ms-value{color:#495057;background-color:#fff}.form-control-file,.form-control-range{display:block;width:100%}.col-form-label{padding-top:calc(.375rem + 1px);padding-bottom:calc(.375rem + 1px);margin-bottom:0;font-size:inherit;line-height:1.5}.col-form-label-lg{padding-top:calc(.5rem + 1px);padding-bottom:calc(.5rem + 1px);font-size:1.25rem;line-height:1.5}.col-form-label-sm{padding-top:calc(.25rem + 1px);padding-bottom:calc(.25rem + 1px);font-size:.875rem;line-height:1.5}.form-control-plaintext{display:block;width:100%;padding:.375rem 0;margin-bottom:0;font-size:1rem;line-height:1.5;color:#212529;background-color:transparent;border:solid transparent;border-width:1px 0}.form-control-plaintext.form-control-lg,.form-control-plaintext.form-control-sm{padding-right:0;padding-left:0}.form-control-sm{height:calc(1.5em + .5rem + 2px);padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.form-control-lg{height:calc(1.5em + 1rem + 2px);padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}select.form-control[multiple],select.form-control[size]{height:auto}textarea.form-control{height:auto}.form-group{margin-bottom:1rem}.form-text{display:block;margin-top:.25rem}.form-row{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;margin-right:-5px;margin-left:-5px}.form-row>.col,.form-row>[class*=col-]{padding-right:5px;padding-left:5px}.form-check{position:relative;display:block;padding-left:1.25rem}.form-check-input{position:absolute;margin-top:.3rem;margin-left:-1.25rem}.form-check-input:disabled~.form-check-label,.form-check-input[disabled]~.form-check-label{color:#6c757d}.form-check-label{margin-bottom:0}.form-check-inline{display:-ms-inline-flexbox;display:inline-flex;-ms-flex-align:center;align-items:center;padding-left:0;margin-right:.75rem}.form-check-inline .form-check-input{position:static;margin-top:0;margin-right:.3125rem;margin-left:0}.valid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#28a745}.valid-tooltip{position:absolute;top:100%;left:0;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(40,167,69,.9);border-radius:.25rem}.is-valid~.valid-feedback,.is-valid~.valid-tooltip,.was-validated :valid~.valid-feedback,.was-validated :valid~.valid-tooltip{display:block}.form-control.is-valid,.was-validated .form-control:valid{border-color:#28a745;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-valid:focus,.was-validated .form-control:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.was-validated textarea.form-control:valid,textarea.form-control.is-valid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-valid,.was-validated .custom-select:valid{border-color:#28a745;padding-right:calc(.75em + 2.3125rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%2328a745' d='M2.3 6.73L.6 4.53c-.4-1.04.46-1.4 1.1-.8l1.1 1.4 3.4-3.8c.6-.63 1.6-.27 1.2.7l-4 4.6c-.43.5-.8.4-1.1.1z'/%3e%3c/svg%3e") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-valid:focus,.was-validated .custom-select:valid:focus{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.form-check-input.is-valid~.form-check-label,.was-validated .form-check-input:valid~.form-check-label{color:#28a745}.form-check-input.is-valid~.valid-feedback,.form-check-input.is-valid~.valid-tooltip,.was-validated .form-check-input:valid~.valid-feedback,.was-validated .form-check-input:valid~.valid-tooltip{display:block}.custom-control-input.is-valid~.custom-control-label,.was-validated .custom-control-input:valid~.custom-control-label{color:#28a745}.custom-control-input.is-valid~.custom-control-label::before,.was-validated .custom-control-input:valid~.custom-control-label::before{border-color:#28a745}.custom-control-input.is-valid:checked~.custom-control-label::before,.was-validated .custom-control-input:valid:checked~.custom-control-label::before{border-color:#34ce57;background-color:#34ce57}.custom-control-input.is-valid:focus~.custom-control-label::before,.was-validated .custom-control-input:valid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.custom-control-input.is-valid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:valid:focus:not(:checked)~.custom-control-label::before{border-color:#28a745}.custom-file-input.is-valid~.custom-file-label,.was-validated .custom-file-input:valid~.custom-file-label{border-color:#28a745}.custom-file-input.is-valid:focus~.custom-file-label,.was-validated .custom-file-input:valid:focus~.custom-file-label{border-color:#28a745;box-shadow:0 0 0 .2rem rgba(40,167,69,.25)}.invalid-feedback{display:none;width:100%;margin-top:.25rem;font-size:80%;color:#dc3545}.invalid-tooltip{position:absolute;top:100%;left:0;z-index:5;display:none;max-width:100%;padding:.25rem .5rem;margin-top:.1rem;font-size:.875rem;line-height:1.5;color:#fff;background-color:rgba(220,53,69,.9);border-radius:.25rem}.is-invalid~.invalid-feedback,.is-invalid~.invalid-tooltip,.was-validated :invalid~.invalid-feedback,.was-validated :invalid~.invalid-tooltip{display:block}.form-control.is-invalid,.was-validated .form-control:invalid{border-color:#dc3545;padding-right:calc(1.5em + .75rem);background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23dc3545' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e");background-repeat:no-repeat;background-position:right calc(.375em + .1875rem) center;background-size:calc(.75em + .375rem) calc(.75em + .375rem)}.form-control.is-invalid:focus,.was-validated .form-control:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.was-validated textarea.form-control:invalid,textarea.form-control.is-invalid{padding-right:calc(1.5em + .75rem);background-position:top calc(.375em + .1875rem) right calc(.375em + .1875rem)}.custom-select.is-invalid,.was-validated .custom-select:invalid{border-color:#dc3545;padding-right:calc(.75em + 2.3125rem);background:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px,url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' fill='none' stroke='%23dc3545' viewBox='0 0 12 12'%3e%3ccircle cx='6' cy='6' r='4.5'/%3e%3cpath stroke-linejoin='round' d='M5.8 3.6h.4L6 6.5z'/%3e%3ccircle cx='6' cy='8.2' r='.6' fill='%23dc3545' stroke='none'/%3e%3c/svg%3e") #fff no-repeat center right 1.75rem/calc(.75em + .375rem) calc(.75em + .375rem)}.custom-select.is-invalid:focus,.was-validated .custom-select:invalid:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-check-input.is-invalid~.form-check-label,.was-validated .form-check-input:invalid~.form-check-label{color:#dc3545}.form-check-input.is-invalid~.invalid-feedback,.form-check-input.is-invalid~.invalid-tooltip,.was-validated .form-check-input:invalid~.invalid-feedback,.was-validated .form-check-input:invalid~.invalid-tooltip{display:block}.custom-control-input.is-invalid~.custom-control-label,.was-validated .custom-control-input:invalid~.custom-control-label{color:#dc3545}.custom-control-input.is-invalid~.custom-control-label::before,.was-validated .custom-control-input:invalid~.custom-control-label::before{border-color:#dc3545}.custom-control-input.is-invalid:checked~.custom-control-label::before,.was-validated .custom-control-input:invalid:checked~.custom-control-label::before{border-color:#e4606d;background-color:#e4606d}.custom-control-input.is-invalid:focus~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.custom-control-input.is-invalid:focus:not(:checked)~.custom-control-label::before,.was-validated .custom-control-input:invalid:focus:not(:checked)~.custom-control-label::before{border-color:#dc3545}.custom-file-input.is-invalid~.custom-file-label,.was-validated .custom-file-input:invalid~.custom-file-label{border-color:#dc3545}.custom-file-input.is-invalid:focus~.custom-file-label,.was-validated .custom-file-input:invalid:focus~.custom-file-label{border-color:#dc3545;box-shadow:0 0 0 .2rem rgba(220,53,69,.25)}.form-inline{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center}.form-inline .form-check{width:100%}@media (min-width:576px){.form-inline label{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;margin-bottom:0}.form-inline .form-group{display:-ms-flexbox;display:flex;-ms-flex:0 0 auto;flex:0 0 auto;-ms-flex-flow:row wrap;flex-flow:row wrap;-ms-flex-align:center;align-items:center;margin-bottom:0}.form-inline .form-control{display:inline-block;width:auto;vertical-align:middle}.form-inline .form-control-plaintext{display:inline-block}.form-inline .custom-select,.form-inline .input-group{width:auto}.form-inline .form-check{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:auto;padding-left:0}.form-inline .form-check-input{position:relative;-ms-flex-negative:0;flex-shrink:0;margin-top:0;margin-right:.25rem;margin-left:0}.form-inline .custom-control{-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center}.form-inline .custom-control-label{margin-bottom:0}}.btn{display:inline-block;font-weight:400;color:#212529;text-align:center;vertical-align:middle;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;background-color:transparent;border:1px solid transparent;padding:.375rem .75rem;font-size:1rem;line-height:1.5;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.btn{transition:none}}.btn:hover{color:#212529;text-decoration:none}.btn.focus,.btn:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.btn.disabled,.btn:disabled{opacity:.65}.btn:not(:disabled):not(.disabled){cursor:pointer}a.btn.disabled,fieldset:disabled a.btn{pointer-events:none}.btn-primary{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:hover{color:#fff;background-color:#0069d9;border-color:#0062cc}.btn-primary.focus,.btn-primary:focus{color:#fff;background-color:#0069d9;border-color:#0062cc;box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-primary.disabled,.btn-primary:disabled{color:#fff;background-color:#007bff;border-color:#007bff}.btn-primary:not(:disabled):not(.disabled).active,.btn-primary:not(:disabled):not(.disabled):active,.show>.btn-primary.dropdown-toggle{color:#fff;background-color:#0062cc;border-color:#005cbf}.btn-primary:not(:disabled):not(.disabled).active:focus,.btn-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(38,143,255,.5)}.btn-secondary{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:hover{color:#fff;background-color:#5a6268;border-color:#545b62}.btn-secondary.focus,.btn-secondary:focus{color:#fff;background-color:#5a6268;border-color:#545b62;box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-secondary.disabled,.btn-secondary:disabled{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-secondary:not(:disabled):not(.disabled).active,.btn-secondary:not(:disabled):not(.disabled):active,.show>.btn-secondary.dropdown-toggle{color:#fff;background-color:#545b62;border-color:#4e555b}.btn-secondary:not(:disabled):not(.disabled).active:focus,.btn-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(130,138,145,.5)}.btn-success{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:hover{color:#fff;background-color:#218838;border-color:#1e7e34}.btn-success.focus,.btn-success:focus{color:#fff;background-color:#218838;border-color:#1e7e34;box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-success.disabled,.btn-success:disabled{color:#fff;background-color:#28a745;border-color:#28a745}.btn-success:not(:disabled):not(.disabled).active,.btn-success:not(:disabled):not(.disabled):active,.show>.btn-success.dropdown-toggle{color:#fff;background-color:#1e7e34;border-color:#1c7430}.btn-success:not(:disabled):not(.disabled).active:focus,.btn-success:not(:disabled):not(.disabled):active:focus,.show>.btn-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(72,180,97,.5)}.btn-info{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:hover{color:#fff;background-color:#138496;border-color:#117a8b}.btn-info.focus,.btn-info:focus{color:#fff;background-color:#138496;border-color:#117a8b;box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-info.disabled,.btn-info:disabled{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-info:not(:disabled):not(.disabled).active,.btn-info:not(:disabled):not(.disabled):active,.show>.btn-info.dropdown-toggle{color:#fff;background-color:#117a8b;border-color:#10707f}.btn-info:not(:disabled):not(.disabled).active:focus,.btn-info:not(:disabled):not(.disabled):active:focus,.show>.btn-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(58,176,195,.5)}.btn-warning{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:hover{color:#212529;background-color:#e0a800;border-color:#d39e00}.btn-warning.focus,.btn-warning:focus{color:#212529;background-color:#e0a800;border-color:#d39e00;box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-warning.disabled,.btn-warning:disabled{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-warning:not(:disabled):not(.disabled).active,.btn-warning:not(:disabled):not(.disabled):active,.show>.btn-warning.dropdown-toggle{color:#212529;background-color:#d39e00;border-color:#c69500}.btn-warning:not(:disabled):not(.disabled).active:focus,.btn-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(222,170,12,.5)}.btn-danger{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:hover{color:#fff;background-color:#c82333;border-color:#bd2130}.btn-danger.focus,.btn-danger:focus{color:#fff;background-color:#c82333;border-color:#bd2130;box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-danger.disabled,.btn-danger:disabled{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-danger:not(:disabled):not(.disabled).active,.btn-danger:not(:disabled):not(.disabled):active,.show>.btn-danger.dropdown-toggle{color:#fff;background-color:#bd2130;border-color:#b21f2d}.btn-danger:not(:disabled):not(.disabled).active:focus,.btn-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(225,83,97,.5)}.btn-light{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:hover{color:#212529;background-color:#e2e6ea;border-color:#dae0e5}.btn-light.focus,.btn-light:focus{color:#212529;background-color:#e2e6ea;border-color:#dae0e5;box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-light.disabled,.btn-light:disabled{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-light:not(:disabled):not(.disabled).active,.btn-light:not(:disabled):not(.disabled):active,.show>.btn-light.dropdown-toggle{color:#212529;background-color:#dae0e5;border-color:#d3d9df}.btn-light:not(:disabled):not(.disabled).active:focus,.btn-light:not(:disabled):not(.disabled):active:focus,.show>.btn-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(216,217,219,.5)}.btn-dark{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:hover{color:#fff;background-color:#23272b;border-color:#1d2124}.btn-dark.focus,.btn-dark:focus{color:#fff;background-color:#23272b;border-color:#1d2124;box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-dark.disabled,.btn-dark:disabled{color:#fff;background-color:#343a40;border-color:#343a40}.btn-dark:not(:disabled):not(.disabled).active,.btn-dark:not(:disabled):not(.disabled):active,.show>.btn-dark.dropdown-toggle{color:#fff;background-color:#1d2124;border-color:#171a1d}.btn-dark:not(:disabled):not(.disabled).active:focus,.btn-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(82,88,93,.5)}.btn-outline-primary{color:#007bff;border-color:#007bff}.btn-outline-primary:hover{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary.focus,.btn-outline-primary:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-primary.disabled,.btn-outline-primary:disabled{color:#007bff;background-color:transparent}.btn-outline-primary:not(:disabled):not(.disabled).active,.btn-outline-primary:not(:disabled):not(.disabled):active,.show>.btn-outline-primary.dropdown-toggle{color:#fff;background-color:#007bff;border-color:#007bff}.btn-outline-primary:not(:disabled):not(.disabled).active:focus,.btn-outline-primary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-primary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.btn-outline-secondary{color:#6c757d;border-color:#6c757d}.btn-outline-secondary:hover{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary.focus,.btn-outline-secondary:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-secondary.disabled,.btn-outline-secondary:disabled{color:#6c757d;background-color:transparent}.btn-outline-secondary:not(:disabled):not(.disabled).active,.btn-outline-secondary:not(:disabled):not(.disabled):active,.show>.btn-outline-secondary.dropdown-toggle{color:#fff;background-color:#6c757d;border-color:#6c757d}.btn-outline-secondary:not(:disabled):not(.disabled).active:focus,.btn-outline-secondary:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-secondary.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.btn-outline-success{color:#28a745;border-color:#28a745}.btn-outline-success:hover{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success.focus,.btn-outline-success:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-success.disabled,.btn-outline-success:disabled{color:#28a745;background-color:transparent}.btn-outline-success:not(:disabled):not(.disabled).active,.btn-outline-success:not(:disabled):not(.disabled):active,.show>.btn-outline-success.dropdown-toggle{color:#fff;background-color:#28a745;border-color:#28a745}.btn-outline-success:not(:disabled):not(.disabled).active:focus,.btn-outline-success:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-success.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.btn-outline-info{color:#17a2b8;border-color:#17a2b8}.btn-outline-info:hover{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info.focus,.btn-outline-info:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-info.disabled,.btn-outline-info:disabled{color:#17a2b8;background-color:transparent}.btn-outline-info:not(:disabled):not(.disabled).active,.btn-outline-info:not(:disabled):not(.disabled):active,.show>.btn-outline-info.dropdown-toggle{color:#fff;background-color:#17a2b8;border-color:#17a2b8}.btn-outline-info:not(:disabled):not(.disabled).active:focus,.btn-outline-info:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-info.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.btn-outline-warning{color:#ffc107;border-color:#ffc107}.btn-outline-warning:hover{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning.focus,.btn-outline-warning:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-warning.disabled,.btn-outline-warning:disabled{color:#ffc107;background-color:transparent}.btn-outline-warning:not(:disabled):not(.disabled).active,.btn-outline-warning:not(:disabled):not(.disabled):active,.show>.btn-outline-warning.dropdown-toggle{color:#212529;background-color:#ffc107;border-color:#ffc107}.btn-outline-warning:not(:disabled):not(.disabled).active:focus,.btn-outline-warning:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-warning.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.btn-outline-danger{color:#dc3545;border-color:#dc3545}.btn-outline-danger:hover{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger.focus,.btn-outline-danger:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-danger.disabled,.btn-outline-danger:disabled{color:#dc3545;background-color:transparent}.btn-outline-danger:not(:disabled):not(.disabled).active,.btn-outline-danger:not(:disabled):not(.disabled):active,.show>.btn-outline-danger.dropdown-toggle{color:#fff;background-color:#dc3545;border-color:#dc3545}.btn-outline-danger:not(:disabled):not(.disabled).active:focus,.btn-outline-danger:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-danger.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.btn-outline-light{color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:hover{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light.focus,.btn-outline-light:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-light.disabled,.btn-outline-light:disabled{color:#f8f9fa;background-color:transparent}.btn-outline-light:not(:disabled):not(.disabled).active,.btn-outline-light:not(:disabled):not(.disabled):active,.show>.btn-outline-light.dropdown-toggle{color:#212529;background-color:#f8f9fa;border-color:#f8f9fa}.btn-outline-light:not(:disabled):not(.disabled).active:focus,.btn-outline-light:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-light.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.btn-outline-dark{color:#343a40;border-color:#343a40}.btn-outline-dark:hover{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark.focus,.btn-outline-dark:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-outline-dark.disabled,.btn-outline-dark:disabled{color:#343a40;background-color:transparent}.btn-outline-dark:not(:disabled):not(.disabled).active,.btn-outline-dark:not(:disabled):not(.disabled):active,.show>.btn-outline-dark.dropdown-toggle{color:#fff;background-color:#343a40;border-color:#343a40}.btn-outline-dark:not(:disabled):not(.disabled).active:focus,.btn-outline-dark:not(:disabled):not(.disabled):active:focus,.show>.btn-outline-dark.dropdown-toggle:focus{box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.btn-link{font-weight:400;color:#007bff;text-decoration:none}.btn-link:hover{color:#0056b3;text-decoration:underline}.btn-link.focus,.btn-link:focus{text-decoration:underline}.btn-link.disabled,.btn-link:disabled{color:#6c757d;pointer-events:none}.btn-group-lg>.btn,.btn-lg{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.btn-group-sm>.btn,.btn-sm{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.btn-block{display:block;width:100%}.btn-block+.btn-block{margin-top:.5rem}input[type=button].btn-block,input[type=reset].btn-block,input[type=submit].btn-block{width:100%}.fade{transition:opacity .15s linear}@media (prefers-reduced-motion:reduce){.fade{transition:none}}.fade:not(.show){opacity:0}.collapse:not(.show){display:none}.collapsing{position:relative;height:0;overflow:hidden;transition:height .35s ease}@media (prefers-reduced-motion:reduce){.collapsing{transition:none}}.dropdown,.dropleft,.dropright,.dropup{position:relative}.dropdown-toggle{white-space:nowrap}.dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid;border-right:.3em solid transparent;border-bottom:0;border-left:.3em solid transparent}.dropdown-toggle:empty::after{margin-left:0}.dropdown-menu{position:absolute;top:100%;left:0;z-index:1000;display:none;float:left;min-width:10rem;padding:.5rem 0;margin:.125rem 0 0;font-size:1rem;color:#212529;text-align:left;list-style:none;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.15);border-radius:.25rem}.dropdown-menu-left{right:auto;left:0}.dropdown-menu-right{right:0;left:auto}@media (min-width:576px){.dropdown-menu-sm-left{right:auto;left:0}.dropdown-menu-sm-right{right:0;left:auto}}@media (min-width:768px){.dropdown-menu-md-left{right:auto;left:0}.dropdown-menu-md-right{right:0;left:auto}}@media (min-width:992px){.dropdown-menu-lg-left{right:auto;left:0}.dropdown-menu-lg-right{right:0;left:auto}}@media (min-width:1200px){.dropdown-menu-xl-left{right:auto;left:0}.dropdown-menu-xl-right{right:0;left:auto}}.dropup .dropdown-menu{top:auto;bottom:100%;margin-top:0;margin-bottom:.125rem}.dropup .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:0;border-right:.3em solid transparent;border-bottom:.3em solid;border-left:.3em solid transparent}.dropup .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-menu{top:0;right:auto;left:100%;margin-top:0;margin-left:.125rem}.dropright .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:0;border-bottom:.3em solid transparent;border-left:.3em solid}.dropright .dropdown-toggle:empty::after{margin-left:0}.dropright .dropdown-toggle::after{vertical-align:0}.dropleft .dropdown-menu{top:0;right:100%;left:auto;margin-top:0;margin-right:.125rem}.dropleft .dropdown-toggle::after{display:inline-block;margin-left:.255em;vertical-align:.255em;content:""}.dropleft .dropdown-toggle::after{display:none}.dropleft .dropdown-toggle::before{display:inline-block;margin-right:.255em;vertical-align:.255em;content:"";border-top:.3em solid transparent;border-right:.3em solid;border-bottom:.3em solid transparent}.dropleft .dropdown-toggle:empty::after{margin-left:0}.dropleft .dropdown-toggle::before{vertical-align:0}.dropdown-menu[x-placement^=bottom],.dropdown-menu[x-placement^=left],.dropdown-menu[x-placement^=right],.dropdown-menu[x-placement^=top]{right:auto;bottom:auto}.dropdown-divider{height:0;margin:.5rem 0;overflow:hidden;border-top:1px solid #e9ecef}.dropdown-item{display:block;width:100%;padding:.25rem 1.5rem;clear:both;font-weight:400;color:#212529;text-align:inherit;white-space:nowrap;background-color:transparent;border:0}.dropdown-item:focus,.dropdown-item:hover{color:#16181b;text-decoration:none;background-color:#f8f9fa}.dropdown-item.active,.dropdown-item:active{color:#fff;text-decoration:none;background-color:#007bff}.dropdown-item.disabled,.dropdown-item:disabled{color:#6c757d;pointer-events:none;background-color:transparent}.dropdown-menu.show{display:block}.dropdown-header{display:block;padding:.5rem 1.5rem;margin-bottom:0;font-size:.875rem;color:#6c757d;white-space:nowrap}.dropdown-item-text{display:block;padding:.25rem 1.5rem;color:#212529}.btn-group,.btn-group-vertical{position:relative;display:-ms-inline-flexbox;display:inline-flex;vertical-align:middle}.btn-group-vertical>.btn,.btn-group>.btn{position:relative;-ms-flex:1 1 auto;flex:1 1 auto}.btn-group-vertical>.btn:hover,.btn-group>.btn:hover{z-index:1}.btn-group-vertical>.btn.active,.btn-group-vertical>.btn:active,.btn-group-vertical>.btn:focus,.btn-group>.btn.active,.btn-group>.btn:active,.btn-group>.btn:focus{z-index:1}.btn-toolbar{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-pack:start;justify-content:flex-start}.btn-toolbar .input-group{width:auto}.btn-group>.btn-group:not(:first-child),.btn-group>.btn:not(:first-child){margin-left:-1px}.btn-group>.btn-group:not(:last-child)>.btn,.btn-group>.btn:not(:last-child):not(.dropdown-toggle){border-top-right-radius:0;border-bottom-right-radius:0}.btn-group>.btn-group:not(:first-child)>.btn,.btn-group>.btn:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.dropdown-toggle-split{padding-right:.5625rem;padding-left:.5625rem}.dropdown-toggle-split::after,.dropright .dropdown-toggle-split::after,.dropup .dropdown-toggle-split::after{margin-left:0}.dropleft .dropdown-toggle-split::before{margin-right:0}.btn-group-sm>.btn+.dropdown-toggle-split,.btn-sm+.dropdown-toggle-split{padding-right:.375rem;padding-left:.375rem}.btn-group-lg>.btn+.dropdown-toggle-split,.btn-lg+.dropdown-toggle-split{padding-right:.75rem;padding-left:.75rem}.btn-group-vertical{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:center;justify-content:center}.btn-group-vertical>.btn,.btn-group-vertical>.btn-group{width:100%}.btn-group-vertical>.btn-group:not(:first-child),.btn-group-vertical>.btn:not(:first-child){margin-top:-1px}.btn-group-vertical>.btn-group:not(:last-child)>.btn,.btn-group-vertical>.btn:not(:last-child):not(.dropdown-toggle){border-bottom-right-radius:0;border-bottom-left-radius:0}.btn-group-vertical>.btn-group:not(:first-child)>.btn,.btn-group-vertical>.btn:not(:first-child){border-top-left-radius:0;border-top-right-radius:0}.btn-group-toggle>.btn,.btn-group-toggle>.btn-group>.btn{margin-bottom:0}.btn-group-toggle>.btn input[type=checkbox],.btn-group-toggle>.btn input[type=radio],.btn-group-toggle>.btn-group>.btn input[type=checkbox],.btn-group-toggle>.btn-group>.btn input[type=radio]{position:absolute;clip:rect(0,0,0,0);pointer-events:none}.input-group{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:stretch;align-items:stretch;width:100%}.input-group>.custom-file,.input-group>.custom-select,.input-group>.form-control,.input-group>.form-control-plaintext{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;width:1%;min-width:0;margin-bottom:0}.input-group>.custom-file+.custom-file,.input-group>.custom-file+.custom-select,.input-group>.custom-file+.form-control,.input-group>.custom-select+.custom-file,.input-group>.custom-select+.custom-select,.input-group>.custom-select+.form-control,.input-group>.form-control+.custom-file,.input-group>.form-control+.custom-select,.input-group>.form-control+.form-control,.input-group>.form-control-plaintext+.custom-file,.input-group>.form-control-plaintext+.custom-select,.input-group>.form-control-plaintext+.form-control{margin-left:-1px}.input-group>.custom-file .custom-file-input:focus~.custom-file-label,.input-group>.custom-select:focus,.input-group>.form-control:focus{z-index:3}.input-group>.custom-file .custom-file-input:focus{z-index:4}.input-group>.custom-select:not(:last-child),.input-group>.form-control:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-select:not(:first-child),.input-group>.form-control:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.input-group>.custom-file{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center}.input-group>.custom-file:not(:last-child) .custom-file-label,.input-group>.custom-file:not(:last-child) .custom-file-label::after{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.custom-file:not(:first-child) .custom-file-label{border-top-left-radius:0;border-bottom-left-radius:0}.input-group-append,.input-group-prepend{display:-ms-flexbox;display:flex}.input-group-append .btn,.input-group-prepend .btn{position:relative;z-index:2}.input-group-append .btn:focus,.input-group-prepend .btn:focus{z-index:3}.input-group-append .btn+.btn,.input-group-append .btn+.input-group-text,.input-group-append .input-group-text+.btn,.input-group-append .input-group-text+.input-group-text,.input-group-prepend .btn+.btn,.input-group-prepend .btn+.input-group-text,.input-group-prepend .input-group-text+.btn,.input-group-prepend .input-group-text+.input-group-text{margin-left:-1px}.input-group-prepend{margin-right:-1px}.input-group-append{margin-left:-1px}.input-group-text{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.375rem .75rem;margin-bottom:0;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;text-align:center;white-space:nowrap;background-color:#e9ecef;border:1px solid #ced4da;border-radius:.25rem}.input-group-text input[type=checkbox],.input-group-text input[type=radio]{margin-top:0}.input-group-lg>.custom-select,.input-group-lg>.form-control:not(textarea){height:calc(1.5em + 1rem + 2px)}.input-group-lg>.custom-select,.input-group-lg>.form-control,.input-group-lg>.input-group-append>.btn,.input-group-lg>.input-group-append>.input-group-text,.input-group-lg>.input-group-prepend>.btn,.input-group-lg>.input-group-prepend>.input-group-text{padding:.5rem 1rem;font-size:1.25rem;line-height:1.5;border-radius:.3rem}.input-group-sm>.custom-select,.input-group-sm>.form-control:not(textarea){height:calc(1.5em + .5rem + 2px)}.input-group-sm>.custom-select,.input-group-sm>.form-control,.input-group-sm>.input-group-append>.btn,.input-group-sm>.input-group-append>.input-group-text,.input-group-sm>.input-group-prepend>.btn,.input-group-sm>.input-group-prepend>.input-group-text{padding:.25rem .5rem;font-size:.875rem;line-height:1.5;border-radius:.2rem}.input-group-lg>.custom-select,.input-group-sm>.custom-select{padding-right:1.75rem}.input-group>.input-group-append:last-child>.btn:not(:last-child):not(.dropdown-toggle),.input-group>.input-group-append:last-child>.input-group-text:not(:last-child),.input-group>.input-group-append:not(:last-child)>.btn,.input-group>.input-group-append:not(:last-child)>.input-group-text,.input-group>.input-group-prepend>.btn,.input-group>.input-group-prepend>.input-group-text{border-top-right-radius:0;border-bottom-right-radius:0}.input-group>.input-group-append>.btn,.input-group>.input-group-append>.input-group-text,.input-group>.input-group-prepend:first-child>.btn:not(:first-child),.input-group>.input-group-prepend:first-child>.input-group-text:not(:first-child),.input-group>.input-group-prepend:not(:first-child)>.btn,.input-group>.input-group-prepend:not(:first-child)>.input-group-text{border-top-left-radius:0;border-bottom-left-radius:0}.custom-control{position:relative;z-index:1;display:block;min-height:1.5rem;padding-left:1.5rem;-webkit-print-color-adjust:exact;color-adjust:exact}.custom-control-inline{display:-ms-inline-flexbox;display:inline-flex;margin-right:1rem}.custom-control-input{position:absolute;left:0;z-index:-1;width:1rem;height:1.25rem;opacity:0}.custom-control-input:checked~.custom-control-label::before{color:#fff;border-color:#007bff;background-color:#007bff}.custom-control-input:focus~.custom-control-label::before{box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-control-input:focus:not(:checked)~.custom-control-label::before{border-color:#80bdff}.custom-control-input:not(:disabled):active~.custom-control-label::before{color:#fff;background-color:#b3d7ff;border-color:#b3d7ff}.custom-control-input:disabled~.custom-control-label,.custom-control-input[disabled]~.custom-control-label{color:#6c757d}.custom-control-input:disabled~.custom-control-label::before,.custom-control-input[disabled]~.custom-control-label::before{background-color:#e9ecef}.custom-control-label{position:relative;margin-bottom:0;vertical-align:top}.custom-control-label::before{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;pointer-events:none;content:"";background-color:#fff;border:#adb5bd solid 1px}.custom-control-label::after{position:absolute;top:.25rem;left:-1.5rem;display:block;width:1rem;height:1rem;content:"";background:no-repeat 50%/50% 50%}.custom-checkbox .custom-control-label::before{border-radius:.25rem}.custom-checkbox .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath fill='%23fff' d='M6.564.75l-3.59 3.612-1.538-1.55L0 4.26l2.974 2.99L8 2.193z'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::before{border-color:#007bff;background-color:#007bff}.custom-checkbox .custom-control-input:indeterminate~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='4' viewBox='0 0 4 4'%3e%3cpath stroke='%23fff' d='M0 2h4'/%3e%3c/svg%3e")}.custom-checkbox .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-checkbox .custom-control-input:disabled:indeterminate~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-radio .custom-control-label::before{border-radius:50%}.custom-radio .custom-control-input:checked~.custom-control-label::after{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='-4 -4 8 8'%3e%3ccircle r='3' fill='%23fff'/%3e%3c/svg%3e")}.custom-radio .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-switch{padding-left:2.25rem}.custom-switch .custom-control-label::before{left:-2.25rem;width:1.75rem;pointer-events:all;border-radius:.5rem}.custom-switch .custom-control-label::after{top:calc(.25rem + 2px);left:calc(-2.25rem + 2px);width:calc(1rem - 4px);height:calc(1rem - 4px);background-color:#adb5bd;border-radius:.5rem;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:transform .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-transform .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-switch .custom-control-label::after{transition:none}}.custom-switch .custom-control-input:checked~.custom-control-label::after{background-color:#fff;-webkit-transform:translateX(.75rem);transform:translateX(.75rem)}.custom-switch .custom-control-input:disabled:checked~.custom-control-label::before{background-color:rgba(0,123,255,.5)}.custom-select{display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);padding:.375rem 1.75rem .375rem .75rem;font-size:1rem;font-weight:400;line-height:1.5;color:#495057;vertical-align:middle;background:#fff url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='4' height='5' viewBox='0 0 4 5'%3e%3cpath fill='%23343a40' d='M2 0L0 2h4zm0 5L0 3h4z'/%3e%3c/svg%3e") no-repeat right .75rem center/8px 10px;border:1px solid #ced4da;border-radius:.25rem;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-select:focus{border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-select:focus::-ms-value{color:#495057;background-color:#fff}.custom-select[multiple],.custom-select[size]:not([size="1"]){height:auto;padding-right:.75rem;background-image:none}.custom-select:disabled{color:#6c757d;background-color:#e9ecef}.custom-select::-ms-expand{display:none}.custom-select:-moz-focusring{color:transparent;text-shadow:0 0 0 #495057}.custom-select-sm{height:calc(1.5em + .5rem + 2px);padding-top:.25rem;padding-bottom:.25rem;padding-left:.5rem;font-size:.875rem}.custom-select-lg{height:calc(1.5em + 1rem + 2px);padding-top:.5rem;padding-bottom:.5rem;padding-left:1rem;font-size:1.25rem}.custom-file{position:relative;display:inline-block;width:100%;height:calc(1.5em + .75rem + 2px);margin-bottom:0}.custom-file-input{position:relative;z-index:2;width:100%;height:calc(1.5em + .75rem + 2px);margin:0;opacity:0}.custom-file-input:focus~.custom-file-label{border-color:#80bdff;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.custom-file-input:disabled~.custom-file-label,.custom-file-input[disabled]~.custom-file-label{background-color:#e9ecef}.custom-file-input:lang(en)~.custom-file-label::after{content:"Browse"}.custom-file-input~.custom-file-label[data-browse]::after{content:attr(data-browse)}.custom-file-label{position:absolute;top:0;right:0;left:0;z-index:1;height:calc(1.5em + .75rem + 2px);padding:.375rem .75rem;font-weight:400;line-height:1.5;color:#495057;background-color:#fff;border:1px solid #ced4da;border-radius:.25rem}.custom-file-label::after{position:absolute;top:0;right:0;bottom:0;z-index:3;display:block;height:calc(1.5em + .75rem);padding:.375rem .75rem;line-height:1.5;color:#495057;content:"Browse";background-color:#e9ecef;border-left:inherit;border-radius:0 .25rem .25rem 0}.custom-range{width:100%;height:1.4rem;padding:0;background-color:transparent;-webkit-appearance:none;-moz-appearance:none;appearance:none}.custom-range:focus{outline:0}.custom-range:focus::-webkit-slider-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-moz-range-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range:focus::-ms-thumb{box-shadow:0 0 0 1px #fff,0 0 0 .2rem rgba(0,123,255,.25)}.custom-range::-moz-focus-outer{border:0}.custom-range::-webkit-slider-thumb{width:1rem;height:1rem;margin-top:-.25rem;background-color:#007bff;border:0;border-radius:1rem;-webkit-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-webkit-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-webkit-slider-thumb{-webkit-transition:none;transition:none}}.custom-range::-webkit-slider-thumb:active{background-color:#b3d7ff}.custom-range::-webkit-slider-runnable-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-moz-range-thumb{width:1rem;height:1rem;background-color:#007bff;border:0;border-radius:1rem;-moz-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;-moz-appearance:none;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-moz-range-thumb{-moz-transition:none;transition:none}}.custom-range::-moz-range-thumb:active{background-color:#b3d7ff}.custom-range::-moz-range-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:#dee2e6;border-color:transparent;border-radius:1rem}.custom-range::-ms-thumb{width:1rem;height:1rem;margin-top:0;margin-right:.2rem;margin-left:.2rem;background-color:#007bff;border:0;border-radius:1rem;-ms-transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out;appearance:none}@media (prefers-reduced-motion:reduce){.custom-range::-ms-thumb{-ms-transition:none;transition:none}}.custom-range::-ms-thumb:active{background-color:#b3d7ff}.custom-range::-ms-track{width:100%;height:.5rem;color:transparent;cursor:pointer;background-color:transparent;border-color:transparent;border-width:.5rem}.custom-range::-ms-fill-lower{background-color:#dee2e6;border-radius:1rem}.custom-range::-ms-fill-upper{margin-right:15px;background-color:#dee2e6;border-radius:1rem}.custom-range:disabled::-webkit-slider-thumb{background-color:#adb5bd}.custom-range:disabled::-webkit-slider-runnable-track{cursor:default}.custom-range:disabled::-moz-range-thumb{background-color:#adb5bd}.custom-range:disabled::-moz-range-track{cursor:default}.custom-range:disabled::-ms-thumb{background-color:#adb5bd}.custom-control-label::before,.custom-file-label,.custom-select{transition:background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.custom-control-label::before,.custom-file-label,.custom-select{transition:none}}.nav{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding-left:0;margin-bottom:0;list-style:none}.nav-link{display:block;padding:.5rem 1rem}.nav-link:focus,.nav-link:hover{text-decoration:none}.nav-link.disabled{color:#6c757d;pointer-events:none;cursor:default}.nav-tabs{border-bottom:1px solid #dee2e6}.nav-tabs .nav-item{margin-bottom:-1px}.nav-tabs .nav-link{border:1px solid transparent;border-top-left-radius:.25rem;border-top-right-radius:.25rem}.nav-tabs .nav-link:focus,.nav-tabs .nav-link:hover{border-color:#e9ecef #e9ecef #dee2e6}.nav-tabs .nav-link.disabled{color:#6c757d;background-color:transparent;border-color:transparent}.nav-tabs .nav-item.show .nav-link,.nav-tabs .nav-link.active{color:#495057;background-color:#fff;border-color:#dee2e6 #dee2e6 #fff}.nav-tabs .dropdown-menu{margin-top:-1px;border-top-left-radius:0;border-top-right-radius:0}.nav-pills .nav-link{border-radius:.25rem}.nav-pills .nav-link.active,.nav-pills .show>.nav-link{color:#fff;background-color:#007bff}.nav-fill .nav-item,.nav-fill>.nav-link{-ms-flex:1 1 auto;flex:1 1 auto;text-align:center}.nav-justified .nav-item,.nav-justified>.nav-link{-ms-flex-preferred-size:0;flex-basis:0;-ms-flex-positive:1;flex-grow:1;text-align:center}.tab-content>.tab-pane{display:none}.tab-content>.active{display:block}.navbar{position:relative;display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between;padding:.5rem 1rem}.navbar .container,.navbar .container-fluid,.navbar .container-lg,.navbar .container-md,.navbar .container-sm,.navbar .container-xl{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:justify;justify-content:space-between}.navbar-brand{display:inline-block;padding-top:.3125rem;padding-bottom:.3125rem;margin-right:1rem;font-size:1.25rem;line-height:inherit;white-space:nowrap}.navbar-brand:focus,.navbar-brand:hover{text-decoration:none}.navbar-nav{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;list-style:none}.navbar-nav .nav-link{padding-right:0;padding-left:0}.navbar-nav .dropdown-menu{position:static;float:none}.navbar-text{display:inline-block;padding-top:.5rem;padding-bottom:.5rem}.navbar-collapse{-ms-flex-preferred-size:100%;flex-basis:100%;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}.navbar-toggler{padding:.25rem .75rem;font-size:1.25rem;line-height:1;background-color:transparent;border:1px solid transparent;border-radius:.25rem}.navbar-toggler:focus,.navbar-toggler:hover{text-decoration:none}.navbar-toggler-icon{display:inline-block;width:1.5em;height:1.5em;vertical-align:middle;content:"";background:no-repeat center center;background-size:100% 100%}@media (max-width:575.98px){.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid,.navbar-expand-sm>.container-lg,.navbar-expand-sm>.container-md,.navbar-expand-sm>.container-sm,.navbar-expand-sm>.container-xl{padding-right:0;padding-left:0}}@media (min-width:576px){.navbar-expand-sm{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-sm .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-sm .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-sm .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-sm>.container,.navbar-expand-sm>.container-fluid,.navbar-expand-sm>.container-lg,.navbar-expand-sm>.container-md,.navbar-expand-sm>.container-sm,.navbar-expand-sm>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-sm .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-sm .navbar-toggler{display:none}}@media (max-width:767.98px){.navbar-expand-md>.container,.navbar-expand-md>.container-fluid,.navbar-expand-md>.container-lg,.navbar-expand-md>.container-md,.navbar-expand-md>.container-sm,.navbar-expand-md>.container-xl{padding-right:0;padding-left:0}}@media (min-width:768px){.navbar-expand-md{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-md .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-md .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-md .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-md>.container,.navbar-expand-md>.container-fluid,.navbar-expand-md>.container-lg,.navbar-expand-md>.container-md,.navbar-expand-md>.container-sm,.navbar-expand-md>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-md .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-md .navbar-toggler{display:none}}@media (max-width:991.98px){.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid,.navbar-expand-lg>.container-lg,.navbar-expand-lg>.container-md,.navbar-expand-lg>.container-sm,.navbar-expand-lg>.container-xl{padding-right:0;padding-left:0}}@media (min-width:992px){.navbar-expand-lg{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-lg .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-lg .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-lg .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-lg>.container,.navbar-expand-lg>.container-fluid,.navbar-expand-lg>.container-lg,.navbar-expand-lg>.container-md,.navbar-expand-lg>.container-sm,.navbar-expand-lg>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-lg .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-lg .navbar-toggler{display:none}}@media (max-width:1199.98px){.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid,.navbar-expand-xl>.container-lg,.navbar-expand-xl>.container-md,.navbar-expand-xl>.container-sm,.navbar-expand-xl>.container-xl{padding-right:0;padding-left:0}}@media (min-width:1200px){.navbar-expand-xl{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand-xl .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand-xl .navbar-nav .dropdown-menu{position:absolute}.navbar-expand-xl .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand-xl>.container,.navbar-expand-xl>.container-fluid,.navbar-expand-xl>.container-lg,.navbar-expand-xl>.container-md,.navbar-expand-xl>.container-sm,.navbar-expand-xl>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand-xl .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand-xl .navbar-toggler{display:none}}.navbar-expand{-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-pack:start;justify-content:flex-start}.navbar-expand>.container,.navbar-expand>.container-fluid,.navbar-expand>.container-lg,.navbar-expand>.container-md,.navbar-expand>.container-sm,.navbar-expand>.container-xl{padding-right:0;padding-left:0}.navbar-expand .navbar-nav{-ms-flex-direction:row;flex-direction:row}.navbar-expand .navbar-nav .dropdown-menu{position:absolute}.navbar-expand .navbar-nav .nav-link{padding-right:.5rem;padding-left:.5rem}.navbar-expand>.container,.navbar-expand>.container-fluid,.navbar-expand>.container-lg,.navbar-expand>.container-md,.navbar-expand>.container-sm,.navbar-expand>.container-xl{-ms-flex-wrap:nowrap;flex-wrap:nowrap}.navbar-expand .navbar-collapse{display:-ms-flexbox!important;display:flex!important;-ms-flex-preferred-size:auto;flex-basis:auto}.navbar-expand .navbar-toggler{display:none}.navbar-light .navbar-brand{color:rgba(0,0,0,.9)}.navbar-light .navbar-brand:focus,.navbar-light .navbar-brand:hover{color:rgba(0,0,0,.9)}.navbar-light .navbar-nav .nav-link{color:rgba(0,0,0,.5)}.navbar-light .navbar-nav .nav-link:focus,.navbar-light .navbar-nav .nav-link:hover{color:rgba(0,0,0,.7)}.navbar-light .navbar-nav .nav-link.disabled{color:rgba(0,0,0,.3)}.navbar-light .navbar-nav .active>.nav-link,.navbar-light .navbar-nav .nav-link.active,.navbar-light .navbar-nav .nav-link.show,.navbar-light .navbar-nav .show>.nav-link{color:rgba(0,0,0,.9)}.navbar-light .navbar-toggler{color:rgba(0,0,0,.5);border-color:rgba(0,0,0,.1)}.navbar-light .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%280, 0, 0, 0.5%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-light .navbar-text{color:rgba(0,0,0,.5)}.navbar-light .navbar-text a{color:rgba(0,0,0,.9)}.navbar-light .navbar-text a:focus,.navbar-light .navbar-text a:hover{color:rgba(0,0,0,.9)}.navbar-dark .navbar-brand{color:#fff}.navbar-dark .navbar-brand:focus,.navbar-dark .navbar-brand:hover{color:#fff}.navbar-dark .navbar-nav .nav-link{color:rgba(255,255,255,.5)}.navbar-dark .navbar-nav .nav-link:focus,.navbar-dark .navbar-nav .nav-link:hover{color:rgba(255,255,255,.75)}.navbar-dark .navbar-nav .nav-link.disabled{color:rgba(255,255,255,.25)}.navbar-dark .navbar-nav .active>.nav-link,.navbar-dark .navbar-nav .nav-link.active,.navbar-dark .navbar-nav .nav-link.show,.navbar-dark .navbar-nav .show>.nav-link{color:#fff}.navbar-dark .navbar-toggler{color:rgba(255,255,255,.5);border-color:rgba(255,255,255,.1)}.navbar-dark .navbar-toggler-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='30' height='30' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.5%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e")}.navbar-dark .navbar-text{color:rgba(255,255,255,.5)}.navbar-dark .navbar-text a{color:#fff}.navbar-dark .navbar-text a:focus,.navbar-dark .navbar-text a:hover{color:#fff}.card{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;min-width:0;word-wrap:break-word;background-color:#fff;background-clip:border-box;border:1px solid rgba(0,0,0,.125);border-radius:.25rem}.card>hr{margin-right:0;margin-left:0}.card>.list-group{border-top:inherit;border-bottom:inherit}.card>.list-group:first-child{border-top-width:0;border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card>.list-group:last-child{border-bottom-width:0;border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card>.card-header+.list-group,.card>.list-group+.card-footer{border-top:0}.card-body{-ms-flex:1 1 auto;flex:1 1 auto;min-height:1px;padding:1.25rem}.card-title{margin-bottom:.75rem}.card-subtitle{margin-top:-.375rem;margin-bottom:0}.card-text:last-child{margin-bottom:0}.card-link:hover{text-decoration:none}.card-link+.card-link{margin-left:1.25rem}.card-header{padding:.75rem 1.25rem;margin-bottom:0;background-color:rgba(0,0,0,.03);border-bottom:1px solid rgba(0,0,0,.125)}.card-header:first-child{border-radius:calc(.25rem - 1px) calc(.25rem - 1px) 0 0}.card-footer{padding:.75rem 1.25rem;background-color:rgba(0,0,0,.03);border-top:1px solid rgba(0,0,0,.125)}.card-footer:last-child{border-radius:0 0 calc(.25rem - 1px) calc(.25rem - 1px)}.card-header-tabs{margin-right:-.625rem;margin-bottom:-.75rem;margin-left:-.625rem;border-bottom:0}.card-header-pills{margin-right:-.625rem;margin-left:-.625rem}.card-img-overlay{position:absolute;top:0;right:0;bottom:0;left:0;padding:1.25rem;border-radius:calc(.25rem - 1px)}.card-img,.card-img-bottom,.card-img-top{-ms-flex-negative:0;flex-shrink:0;width:100%}.card-img,.card-img-top{border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.card-img,.card-img-bottom{border-bottom-right-radius:calc(.25rem - 1px);border-bottom-left-radius:calc(.25rem - 1px)}.card-deck .card{margin-bottom:15px}@media (min-width:576px){.card-deck{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap;margin-right:-15px;margin-left:-15px}.card-deck .card{-ms-flex:1 0 0%;flex:1 0 0%;margin-right:15px;margin-bottom:0;margin-left:15px}}.card-group>.card{margin-bottom:15px}@media (min-width:576px){.card-group{display:-ms-flexbox;display:flex;-ms-flex-flow:row wrap;flex-flow:row wrap}.card-group>.card{-ms-flex:1 0 0%;flex:1 0 0%;margin-bottom:0}.card-group>.card+.card{margin-left:0;border-left:0}.card-group>.card:not(:last-child){border-top-right-radius:0;border-bottom-right-radius:0}.card-group>.card:not(:last-child) .card-header,.card-group>.card:not(:last-child) .card-img-top{border-top-right-radius:0}.card-group>.card:not(:last-child) .card-footer,.card-group>.card:not(:last-child) .card-img-bottom{border-bottom-right-radius:0}.card-group>.card:not(:first-child){border-top-left-radius:0;border-bottom-left-radius:0}.card-group>.card:not(:first-child) .card-header,.card-group>.card:not(:first-child) .card-img-top{border-top-left-radius:0}.card-group>.card:not(:first-child) .card-footer,.card-group>.card:not(:first-child) .card-img-bottom{border-bottom-left-radius:0}}.card-columns .card{margin-bottom:.75rem}@media (min-width:576px){.card-columns{-webkit-column-count:3;-moz-column-count:3;column-count:3;-webkit-column-gap:1.25rem;-moz-column-gap:1.25rem;column-gap:1.25rem;orphans:1;widows:1}.card-columns .card{display:inline-block;width:100%}}.accordion{overflow-anchor:none}.accordion>.card{overflow:hidden}.accordion>.card:not(:last-of-type){border-bottom:0;border-bottom-right-radius:0;border-bottom-left-radius:0}.accordion>.card:not(:first-of-type){border-top-left-radius:0;border-top-right-radius:0}.accordion>.card>.card-header{border-radius:0;margin-bottom:-1px}.breadcrumb{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;padding:.75rem 1rem;margin-bottom:1rem;list-style:none;background-color:#e9ecef;border-radius:.25rem}.breadcrumb-item{display:-ms-flexbox;display:flex}.breadcrumb-item+.breadcrumb-item{padding-left:.5rem}.breadcrumb-item+.breadcrumb-item::before{display:inline-block;padding-right:.5rem;color:#6c757d;content:"/"}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:underline}.breadcrumb-item+.breadcrumb-item:hover::before{text-decoration:none}.breadcrumb-item.active{color:#6c757d}.pagination{display:-ms-flexbox;display:flex;padding-left:0;list-style:none;border-radius:.25rem}.page-link{position:relative;display:block;padding:.5rem .75rem;margin-left:-1px;line-height:1.25;color:#007bff;background-color:#fff;border:1px solid #dee2e6}.page-link:hover{z-index:2;color:#0056b3;text-decoration:none;background-color:#e9ecef;border-color:#dee2e6}.page-link:focus{z-index:3;outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.25)}.page-item:first-child .page-link{margin-left:0;border-top-left-radius:.25rem;border-bottom-left-radius:.25rem}.page-item:last-child .page-link{border-top-right-radius:.25rem;border-bottom-right-radius:.25rem}.page-item.active .page-link{z-index:3;color:#fff;background-color:#007bff;border-color:#007bff}.page-item.disabled .page-link{color:#6c757d;pointer-events:none;cursor:auto;background-color:#fff;border-color:#dee2e6}.pagination-lg .page-link{padding:.75rem 1.5rem;font-size:1.25rem;line-height:1.5}.pagination-lg .page-item:first-child .page-link{border-top-left-radius:.3rem;border-bottom-left-radius:.3rem}.pagination-lg .page-item:last-child .page-link{border-top-right-radius:.3rem;border-bottom-right-radius:.3rem}.pagination-sm .page-link{padding:.25rem .5rem;font-size:.875rem;line-height:1.5}.pagination-sm .page-item:first-child .page-link{border-top-left-radius:.2rem;border-bottom-left-radius:.2rem}.pagination-sm .page-item:last-child .page-link{border-top-right-radius:.2rem;border-bottom-right-radius:.2rem}.badge{display:inline-block;padding:.25em .4em;font-size:75%;font-weight:700;line-height:1;text-align:center;white-space:nowrap;vertical-align:baseline;border-radius:.25rem;transition:color .15s ease-in-out,background-color .15s ease-in-out,border-color .15s ease-in-out,box-shadow .15s ease-in-out}@media (prefers-reduced-motion:reduce){.badge{transition:none}}a.badge:focus,a.badge:hover{text-decoration:none}.badge:empty{display:none}.btn .badge{position:relative;top:-1px}.badge-pill{padding-right:.6em;padding-left:.6em;border-radius:10rem}.badge-primary{color:#fff;background-color:#007bff}a.badge-primary:focus,a.badge-primary:hover{color:#fff;background-color:#0062cc}a.badge-primary.focus,a.badge-primary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(0,123,255,.5)}.badge-secondary{color:#fff;background-color:#6c757d}a.badge-secondary:focus,a.badge-secondary:hover{color:#fff;background-color:#545b62}a.badge-secondary.focus,a.badge-secondary:focus{outline:0;box-shadow:0 0 0 .2rem rgba(108,117,125,.5)}.badge-success{color:#fff;background-color:#28a745}a.badge-success:focus,a.badge-success:hover{color:#fff;background-color:#1e7e34}a.badge-success.focus,a.badge-success:focus{outline:0;box-shadow:0 0 0 .2rem rgba(40,167,69,.5)}.badge-info{color:#fff;background-color:#17a2b8}a.badge-info:focus,a.badge-info:hover{color:#fff;background-color:#117a8b}a.badge-info.focus,a.badge-info:focus{outline:0;box-shadow:0 0 0 .2rem rgba(23,162,184,.5)}.badge-warning{color:#212529;background-color:#ffc107}a.badge-warning:focus,a.badge-warning:hover{color:#212529;background-color:#d39e00}a.badge-warning.focus,a.badge-warning:focus{outline:0;box-shadow:0 0 0 .2rem rgba(255,193,7,.5)}.badge-danger{color:#fff;background-color:#dc3545}a.badge-danger:focus,a.badge-danger:hover{color:#fff;background-color:#bd2130}a.badge-danger.focus,a.badge-danger:focus{outline:0;box-shadow:0 0 0 .2rem rgba(220,53,69,.5)}.badge-light{color:#212529;background-color:#f8f9fa}a.badge-light:focus,a.badge-light:hover{color:#212529;background-color:#dae0e5}a.badge-light.focus,a.badge-light:focus{outline:0;box-shadow:0 0 0 .2rem rgba(248,249,250,.5)}.badge-dark{color:#fff;background-color:#343a40}a.badge-dark:focus,a.badge-dark:hover{color:#fff;background-color:#1d2124}a.badge-dark.focus,a.badge-dark:focus{outline:0;box-shadow:0 0 0 .2rem rgba(52,58,64,.5)}.jumbotron{padding:2rem 1rem;margin-bottom:2rem;background-color:#e9ecef;border-radius:.3rem}@media (min-width:576px){.jumbotron{padding:4rem 2rem}}.jumbotron-fluid{padding-right:0;padding-left:0;border-radius:0}.alert{position:relative;padding:.75rem 1.25rem;margin-bottom:1rem;border:1px solid transparent;border-radius:.25rem}.alert-heading{color:inherit}.alert-link{font-weight:700}.alert-dismissible{padding-right:4rem}.alert-dismissible .close{position:absolute;top:0;right:0;z-index:2;padding:.75rem 1.25rem;color:inherit}.alert-primary{color:#004085;background-color:#cce5ff;border-color:#b8daff}.alert-primary hr{border-top-color:#9fcdff}.alert-primary .alert-link{color:#002752}.alert-secondary{color:#383d41;background-color:#e2e3e5;border-color:#d6d8db}.alert-secondary hr{border-top-color:#c8cbcf}.alert-secondary .alert-link{color:#202326}.alert-success{color:#155724;background-color:#d4edda;border-color:#c3e6cb}.alert-success hr{border-top-color:#b1dfbb}.alert-success .alert-link{color:#0b2e13}.alert-info{color:#0c5460;background-color:#d1ecf1;border-color:#bee5eb}.alert-info hr{border-top-color:#abdde5}.alert-info .alert-link{color:#062c33}.alert-warning{color:#856404;background-color:#fff3cd;border-color:#ffeeba}.alert-warning hr{border-top-color:#ffe8a1}.alert-warning .alert-link{color:#533f03}.alert-danger{color:#721c24;background-color:#f8d7da;border-color:#f5c6cb}.alert-danger hr{border-top-color:#f1b0b7}.alert-danger .alert-link{color:#491217}.alert-light{color:#818182;background-color:#fefefe;border-color:#fdfdfe}.alert-light hr{border-top-color:#ececf6}.alert-light .alert-link{color:#686868}.alert-dark{color:#1b1e21;background-color:#d6d8d9;border-color:#c6c8ca}.alert-dark hr{border-top-color:#b9bbbe}.alert-dark .alert-link{color:#040505}@-webkit-keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}@keyframes progress-bar-stripes{from{background-position:1rem 0}to{background-position:0 0}}.progress{display:-ms-flexbox;display:flex;height:1rem;overflow:hidden;line-height:0;font-size:.75rem;background-color:#e9ecef;border-radius:.25rem}.progress-bar{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;overflow:hidden;color:#fff;text-align:center;white-space:nowrap;background-color:#007bff;transition:width .6s ease}@media (prefers-reduced-motion:reduce){.progress-bar{transition:none}}.progress-bar-striped{background-image:linear-gradient(45deg,rgba(255,255,255,.15) 25%,transparent 25%,transparent 50%,rgba(255,255,255,.15) 50%,rgba(255,255,255,.15) 75%,transparent 75%,transparent);background-size:1rem 1rem}.progress-bar-animated{-webkit-animation:progress-bar-stripes 1s linear infinite;animation:progress-bar-stripes 1s linear infinite}@media (prefers-reduced-motion:reduce){.progress-bar-animated{-webkit-animation:none;animation:none}}.media{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start}.media-body{-ms-flex:1;flex:1}.list-group{display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;padding-left:0;margin-bottom:0;border-radius:.25rem}.list-group-item-action{width:100%;color:#495057;text-align:inherit}.list-group-item-action:focus,.list-group-item-action:hover{z-index:1;color:#495057;text-decoration:none;background-color:#f8f9fa}.list-group-item-action:active{color:#212529;background-color:#e9ecef}.list-group-item{position:relative;display:block;padding:.75rem 1.25rem;background-color:#fff;border:1px solid rgba(0,0,0,.125)}.list-group-item:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.list-group-item:last-child{border-bottom-right-radius:inherit;border-bottom-left-radius:inherit}.list-group-item.disabled,.list-group-item:disabled{color:#6c757d;pointer-events:none;background-color:#fff}.list-group-item.active{z-index:2;color:#fff;background-color:#007bff;border-color:#007bff}.list-group-item+.list-group-item{border-top-width:0}.list-group-item+.list-group-item.active{margin-top:-1px;border-top-width:1px}.list-group-horizontal{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal>.list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal>.list-group-item.active{margin-top:0}.list-group-horizontal>.list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}@media (min-width:576px){.list-group-horizontal-sm{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-sm>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-sm>.list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-sm>.list-group-item.active{margin-top:0}.list-group-horizontal-sm>.list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-sm>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:768px){.list-group-horizontal-md{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-md>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-md>.list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-md>.list-group-item.active{margin-top:0}.list-group-horizontal-md>.list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-md>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:992px){.list-group-horizontal-lg{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-lg>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-lg>.list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-lg>.list-group-item.active{margin-top:0}.list-group-horizontal-lg>.list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-lg>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}@media (min-width:1200px){.list-group-horizontal-xl{-ms-flex-direction:row;flex-direction:row}.list-group-horizontal-xl>.list-group-item:first-child{border-bottom-left-radius:.25rem;border-top-right-radius:0}.list-group-horizontal-xl>.list-group-item:last-child{border-top-right-radius:.25rem;border-bottom-left-radius:0}.list-group-horizontal-xl>.list-group-item.active{margin-top:0}.list-group-horizontal-xl>.list-group-item+.list-group-item{border-top-width:1px;border-left-width:0}.list-group-horizontal-xl>.list-group-item+.list-group-item.active{margin-left:-1px;border-left-width:1px}}.list-group-flush{border-radius:0}.list-group-flush>.list-group-item{border-width:0 0 1px}.list-group-flush>.list-group-item:last-child{border-bottom-width:0}.list-group-item-primary{color:#004085;background-color:#b8daff}.list-group-item-primary.list-group-item-action:focus,.list-group-item-primary.list-group-item-action:hover{color:#004085;background-color:#9fcdff}.list-group-item-primary.list-group-item-action.active{color:#fff;background-color:#004085;border-color:#004085}.list-group-item-secondary{color:#383d41;background-color:#d6d8db}.list-group-item-secondary.list-group-item-action:focus,.list-group-item-secondary.list-group-item-action:hover{color:#383d41;background-color:#c8cbcf}.list-group-item-secondary.list-group-item-action.active{color:#fff;background-color:#383d41;border-color:#383d41}.list-group-item-success{color:#155724;background-color:#c3e6cb}.list-group-item-success.list-group-item-action:focus,.list-group-item-success.list-group-item-action:hover{color:#155724;background-color:#b1dfbb}.list-group-item-success.list-group-item-action.active{color:#fff;background-color:#155724;border-color:#155724}.list-group-item-info{color:#0c5460;background-color:#bee5eb}.list-group-item-info.list-group-item-action:focus,.list-group-item-info.list-group-item-action:hover{color:#0c5460;background-color:#abdde5}.list-group-item-info.list-group-item-action.active{color:#fff;background-color:#0c5460;border-color:#0c5460}.list-group-item-warning{color:#856404;background-color:#ffeeba}.list-group-item-warning.list-group-item-action:focus,.list-group-item-warning.list-group-item-action:hover{color:#856404;background-color:#ffe8a1}.list-group-item-warning.list-group-item-action.active{color:#fff;background-color:#856404;border-color:#856404}.list-group-item-danger{color:#721c24;background-color:#f5c6cb}.list-group-item-danger.list-group-item-action:focus,.list-group-item-danger.list-group-item-action:hover{color:#721c24;background-color:#f1b0b7}.list-group-item-danger.list-group-item-action.active{color:#fff;background-color:#721c24;border-color:#721c24}.list-group-item-light{color:#818182;background-color:#fdfdfe}.list-group-item-light.list-group-item-action:focus,.list-group-item-light.list-group-item-action:hover{color:#818182;background-color:#ececf6}.list-group-item-light.list-group-item-action.active{color:#fff;background-color:#818182;border-color:#818182}.list-group-item-dark{color:#1b1e21;background-color:#c6c8ca}.list-group-item-dark.list-group-item-action:focus,.list-group-item-dark.list-group-item-action:hover{color:#1b1e21;background-color:#b9bbbe}.list-group-item-dark.list-group-item-action.active{color:#fff;background-color:#1b1e21;border-color:#1b1e21}.close{float:right;font-size:1.5rem;font-weight:700;line-height:1;color:#000;text-shadow:0 1px 0 #fff;opacity:.5}.close:hover{color:#000;text-decoration:none}.close:not(:disabled):not(.disabled):focus,.close:not(:disabled):not(.disabled):hover{opacity:.75}button.close{padding:0;background-color:transparent;border:0}a.close.disabled{pointer-events:none}.toast{-ms-flex-preferred-size:350px;flex-basis:350px;max-width:350px;font-size:.875rem;background-color:rgba(255,255,255,.85);background-clip:padding-box;border:1px solid rgba(0,0,0,.1);box-shadow:0 .25rem .75rem rgba(0,0,0,.1);opacity:0;border-radius:.25rem}.toast:not(:last-child){margin-bottom:.75rem}.toast.showing{opacity:1}.toast.show{display:block;opacity:1}.toast.hide{display:none}.toast-header{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;padding:.25rem .75rem;color:#6c757d;background-color:rgba(255,255,255,.85);background-clip:padding-box;border-bottom:1px solid rgba(0,0,0,.05);border-top-left-radius:calc(.25rem - 1px);border-top-right-radius:calc(.25rem - 1px)}.toast-body{padding:.75rem}.modal-open{overflow:hidden}.modal-open .modal{overflow-x:hidden;overflow-y:auto}.modal{position:fixed;top:0;left:0;z-index:1050;display:none;width:100%;height:100%;overflow:hidden;outline:0}.modal-dialog{position:relative;width:auto;margin:.5rem;pointer-events:none}.modal.fade .modal-dialog{transition:-webkit-transform .3s ease-out;transition:transform .3s ease-out;transition:transform .3s ease-out,-webkit-transform .3s ease-out;-webkit-transform:translate(0,-50px);transform:translate(0,-50px)}@media (prefers-reduced-motion:reduce){.modal.fade .modal-dialog{transition:none}}.modal.show .modal-dialog{-webkit-transform:none;transform:none}.modal.modal-static .modal-dialog{-webkit-transform:scale(1.02);transform:scale(1.02)}.modal-dialog-scrollable{display:-ms-flexbox;display:flex;max-height:calc(100% - 1rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 1rem);overflow:hidden}.modal-dialog-scrollable .modal-footer,.modal-dialog-scrollable .modal-header{-ms-flex-negative:0;flex-shrink:0}.modal-dialog-scrollable .modal-body{overflow-y:auto}.modal-dialog-centered{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;min-height:calc(100% - 1rem)}.modal-dialog-centered::before{display:block;height:calc(100vh - 1rem);height:-webkit-min-content;height:-moz-min-content;height:min-content;content:""}.modal-dialog-centered.modal-dialog-scrollable{-ms-flex-direction:column;flex-direction:column;-ms-flex-pack:center;justify-content:center;height:100%}.modal-dialog-centered.modal-dialog-scrollable .modal-content{max-height:none}.modal-dialog-centered.modal-dialog-scrollable::before{content:none}.modal-content{position:relative;display:-ms-flexbox;display:flex;-ms-flex-direction:column;flex-direction:column;width:100%;pointer-events:auto;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem;outline:0}.modal-backdrop{position:fixed;top:0;left:0;z-index:1040;width:100vw;height:100vh;background-color:#000}.modal-backdrop.fade{opacity:0}.modal-backdrop.show{opacity:.5}.modal-header{display:-ms-flexbox;display:flex;-ms-flex-align:start;align-items:flex-start;-ms-flex-pack:justify;justify-content:space-between;padding:1rem 1rem;border-bottom:1px solid #dee2e6;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.modal-header .close{padding:1rem 1rem;margin:-1rem -1rem -1rem auto}.modal-title{margin-bottom:0;line-height:1.5}.modal-body{position:relative;-ms-flex:1 1 auto;flex:1 1 auto;padding:1rem}.modal-footer{display:-ms-flexbox;display:flex;-ms-flex-wrap:wrap;flex-wrap:wrap;-ms-flex-align:center;align-items:center;-ms-flex-pack:end;justify-content:flex-end;padding:.75rem;border-top:1px solid #dee2e6;border-bottom-right-radius:calc(.3rem - 1px);border-bottom-left-radius:calc(.3rem - 1px)}.modal-footer>*{margin:.25rem}.modal-scrollbar-measure{position:absolute;top:-9999px;width:50px;height:50px;overflow:scroll}@media (min-width:576px){.modal-dialog{max-width:500px;margin:1.75rem auto}.modal-dialog-scrollable{max-height:calc(100% - 3.5rem)}.modal-dialog-scrollable .modal-content{max-height:calc(100vh - 3.5rem)}.modal-dialog-centered{min-height:calc(100% - 3.5rem)}.modal-dialog-centered::before{height:calc(100vh - 3.5rem);height:-webkit-min-content;height:-moz-min-content;height:min-content}.modal-sm{max-width:300px}}@media (min-width:992px){.modal-lg,.modal-xl{max-width:800px}}@media (min-width:1200px){.modal-xl{max-width:1140px}}.tooltip{position:absolute;z-index:1070;display:block;margin:0;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;opacity:0}.tooltip.show{opacity:.9}.tooltip .arrow{position:absolute;display:block;width:.8rem;height:.4rem}.tooltip .arrow::before{position:absolute;content:"";border-color:transparent;border-style:solid}.bs-tooltip-auto[x-placement^=top],.bs-tooltip-top{padding:.4rem 0}.bs-tooltip-auto[x-placement^=top] .arrow,.bs-tooltip-top .arrow{bottom:0}.bs-tooltip-auto[x-placement^=top] .arrow::before,.bs-tooltip-top .arrow::before{top:0;border-width:.4rem .4rem 0;border-top-color:#000}.bs-tooltip-auto[x-placement^=right],.bs-tooltip-right{padding:0 .4rem}.bs-tooltip-auto[x-placement^=right] .arrow,.bs-tooltip-right .arrow{left:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=right] .arrow::before,.bs-tooltip-right .arrow::before{right:0;border-width:.4rem .4rem .4rem 0;border-right-color:#000}.bs-tooltip-auto[x-placement^=bottom],.bs-tooltip-bottom{padding:.4rem 0}.bs-tooltip-auto[x-placement^=bottom] .arrow,.bs-tooltip-bottom .arrow{top:0}.bs-tooltip-auto[x-placement^=bottom] .arrow::before,.bs-tooltip-bottom .arrow::before{bottom:0;border-width:0 .4rem .4rem;border-bottom-color:#000}.bs-tooltip-auto[x-placement^=left],.bs-tooltip-left{padding:0 .4rem}.bs-tooltip-auto[x-placement^=left] .arrow,.bs-tooltip-left .arrow{right:0;width:.4rem;height:.8rem}.bs-tooltip-auto[x-placement^=left] .arrow::before,.bs-tooltip-left .arrow::before{left:0;border-width:.4rem 0 .4rem .4rem;border-left-color:#000}.tooltip-inner{max-width:200px;padding:.25rem .5rem;color:#fff;text-align:center;background-color:#000;border-radius:.25rem}.popover{position:absolute;top:0;left:0;z-index:1060;display:block;max-width:276px;font-family:-apple-system,BlinkMacSystemFont,"Segoe UI",Roboto,"Helvetica Neue",Arial,"Noto Sans",sans-serif,"Apple Color Emoji","Segoe UI Emoji","Segoe UI Symbol","Noto Color Emoji";font-style:normal;font-weight:400;line-height:1.5;text-align:left;text-align:start;text-decoration:none;text-shadow:none;text-transform:none;letter-spacing:normal;word-break:normal;word-spacing:normal;white-space:normal;line-break:auto;font-size:.875rem;word-wrap:break-word;background-color:#fff;background-clip:padding-box;border:1px solid rgba(0,0,0,.2);border-radius:.3rem}.popover .arrow{position:absolute;display:block;width:1rem;height:.5rem;margin:0 .3rem}.popover .arrow::after,.popover .arrow::before{position:absolute;display:block;content:"";border-color:transparent;border-style:solid}.bs-popover-auto[x-placement^=top],.bs-popover-top{margin-bottom:.5rem}.bs-popover-auto[x-placement^=top]>.arrow,.bs-popover-top>.arrow{bottom:calc(-.5rem - 1px)}.bs-popover-auto[x-placement^=top]>.arrow::before,.bs-popover-top>.arrow::before{bottom:0;border-width:.5rem .5rem 0;border-top-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=top]>.arrow::after,.bs-popover-top>.arrow::after{bottom:1px;border-width:.5rem .5rem 0;border-top-color:#fff}.bs-popover-auto[x-placement^=right],.bs-popover-right{margin-left:.5rem}.bs-popover-auto[x-placement^=right]>.arrow,.bs-popover-right>.arrow{left:calc(-.5rem - 1px);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=right]>.arrow::before,.bs-popover-right>.arrow::before{left:0;border-width:.5rem .5rem .5rem 0;border-right-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=right]>.arrow::after,.bs-popover-right>.arrow::after{left:1px;border-width:.5rem .5rem .5rem 0;border-right-color:#fff}.bs-popover-auto[x-placement^=bottom],.bs-popover-bottom{margin-top:.5rem}.bs-popover-auto[x-placement^=bottom]>.arrow,.bs-popover-bottom>.arrow{top:calc(-.5rem - 1px)}.bs-popover-auto[x-placement^=bottom]>.arrow::before,.bs-popover-bottom>.arrow::before{top:0;border-width:0 .5rem .5rem .5rem;border-bottom-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=bottom]>.arrow::after,.bs-popover-bottom>.arrow::after{top:1px;border-width:0 .5rem .5rem .5rem;border-bottom-color:#fff}.bs-popover-auto[x-placement^=bottom] .popover-header::before,.bs-popover-bottom .popover-header::before{position:absolute;top:0;left:50%;display:block;width:1rem;margin-left:-.5rem;content:"";border-bottom:1px solid #f7f7f7}.bs-popover-auto[x-placement^=left],.bs-popover-left{margin-right:.5rem}.bs-popover-auto[x-placement^=left]>.arrow,.bs-popover-left>.arrow{right:calc(-.5rem - 1px);width:.5rem;height:1rem;margin:.3rem 0}.bs-popover-auto[x-placement^=left]>.arrow::before,.bs-popover-left>.arrow::before{right:0;border-width:.5rem 0 .5rem .5rem;border-left-color:rgba(0,0,0,.25)}.bs-popover-auto[x-placement^=left]>.arrow::after,.bs-popover-left>.arrow::after{right:1px;border-width:.5rem 0 .5rem .5rem;border-left-color:#fff}.popover-header{padding:.5rem .75rem;margin-bottom:0;font-size:1rem;background-color:#f7f7f7;border-bottom:1px solid #ebebeb;border-top-left-radius:calc(.3rem - 1px);border-top-right-radius:calc(.3rem - 1px)}.popover-header:empty{display:none}.popover-body{padding:.5rem .75rem;color:#212529}.carousel{position:relative}.carousel.pointer-event{-ms-touch-action:pan-y;touch-action:pan-y}.carousel-inner{position:relative;width:100%;overflow:hidden}.carousel-inner::after{display:block;clear:both;content:""}.carousel-item{position:relative;display:none;float:left;width:100%;margin-right:-100%;-webkit-backface-visibility:hidden;backface-visibility:hidden;transition:-webkit-transform .6s ease-in-out;transition:transform .6s ease-in-out;transition:transform .6s ease-in-out,-webkit-transform .6s ease-in-out}@media (prefers-reduced-motion:reduce){.carousel-item{transition:none}}.carousel-item-next,.carousel-item-prev,.carousel-item.active{display:block}.active.carousel-item-right,.carousel-item-next:not(.carousel-item-left){-webkit-transform:translateX(100%);transform:translateX(100%)}.active.carousel-item-left,.carousel-item-prev:not(.carousel-item-right){-webkit-transform:translateX(-100%);transform:translateX(-100%)}.carousel-fade .carousel-item{opacity:0;transition-property:opacity;-webkit-transform:none;transform:none}.carousel-fade .carousel-item-next.carousel-item-left,.carousel-fade .carousel-item-prev.carousel-item-right,.carousel-fade .carousel-item.active{z-index:1;opacity:1}.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{z-index:0;opacity:0;transition:opacity 0s .6s}@media (prefers-reduced-motion:reduce){.carousel-fade .active.carousel-item-left,.carousel-fade .active.carousel-item-right{transition:none}}.carousel-control-next,.carousel-control-prev{position:absolute;top:0;bottom:0;z-index:1;display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:15%;color:#fff;text-align:center;opacity:.5;transition:opacity .15s ease}@media (prefers-reduced-motion:reduce){.carousel-control-next,.carousel-control-prev{transition:none}}.carousel-control-next:focus,.carousel-control-next:hover,.carousel-control-prev:focus,.carousel-control-prev:hover{color:#fff;text-decoration:none;outline:0;opacity:.9}.carousel-control-prev{left:0}.carousel-control-next{right:0}.carousel-control-next-icon,.carousel-control-prev-icon{display:inline-block;width:20px;height:20px;background:no-repeat 50%/100% 100%}.carousel-control-prev-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M5.25 0l-4 4 4 4 1.5-1.5L4.25 4l2.5-2.5L5.25 0z'/%3e%3c/svg%3e")}.carousel-control-next-icon{background-image:url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' fill='%23fff' width='8' height='8' viewBox='0 0 8 8'%3e%3cpath d='M2.75 0l-1.5 1.5L3.75 4l-2.5 2.5L2.75 8l4-4-4-4z'/%3e%3c/svg%3e")}.carousel-indicators{position:absolute;right:0;bottom:0;left:0;z-index:15;display:-ms-flexbox;display:flex;-ms-flex-pack:center;justify-content:center;padding-left:0;margin-right:15%;margin-left:15%;list-style:none}.carousel-indicators li{box-sizing:content-box;-ms-flex:0 1 auto;flex:0 1 auto;width:30px;height:3px;margin-right:3px;margin-left:3px;text-indent:-999px;cursor:pointer;background-color:#fff;background-clip:padding-box;border-top:10px solid transparent;border-bottom:10px solid transparent;opacity:.5;transition:opacity .6s ease}@media (prefers-reduced-motion:reduce){.carousel-indicators li{transition:none}}.carousel-indicators .active{opacity:1}.carousel-caption{position:absolute;right:15%;bottom:20px;left:15%;z-index:10;padding-top:20px;padding-bottom:20px;color:#fff;text-align:center}@-webkit-keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spinner-border{to{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}.spinner-border{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;border:.25em solid currentColor;border-right-color:transparent;border-radius:50%;-webkit-animation:spinner-border .75s linear infinite;animation:spinner-border .75s linear infinite}.spinner-border-sm{width:1rem;height:1rem;border-width:.2em}@-webkit-keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1;-webkit-transform:none;transform:none}}@keyframes spinner-grow{0%{-webkit-transform:scale(0);transform:scale(0)}50%{opacity:1;-webkit-transform:none;transform:none}}.spinner-grow{display:inline-block;width:2rem;height:2rem;vertical-align:text-bottom;background-color:currentColor;border-radius:50%;opacity:0;-webkit-animation:spinner-grow .75s linear infinite;animation:spinner-grow .75s linear infinite}.spinner-grow-sm{width:1rem;height:1rem}.align-baseline{vertical-align:baseline!important}.align-top{vertical-align:top!important}.align-middle{vertical-align:middle!important}.align-bottom{vertical-align:bottom!important}.align-text-bottom{vertical-align:text-bottom!important}.align-text-top{vertical-align:text-top!important}.bg-primary{background-color:#007bff!important}a.bg-primary:focus,a.bg-primary:hover,button.bg-primary:focus,button.bg-primary:hover{background-color:#0062cc!important}.bg-secondary{background-color:#6c757d!important}a.bg-secondary:focus,a.bg-secondary:hover,button.bg-secondary:focus,button.bg-secondary:hover{background-color:#545b62!important}.bg-success{background-color:#28a745!important}a.bg-success:focus,a.bg-success:hover,button.bg-success:focus,button.bg-success:hover{background-color:#1e7e34!important}.bg-info{background-color:#17a2b8!important}a.bg-info:focus,a.bg-info:hover,button.bg-info:focus,button.bg-info:hover{background-color:#117a8b!important}.bg-warning{background-color:#ffc107!important}a.bg-warning:focus,a.bg-warning:hover,button.bg-warning:focus,button.bg-warning:hover{background-color:#d39e00!important}.bg-danger{background-color:#dc3545!important}a.bg-danger:focus,a.bg-danger:hover,button.bg-danger:focus,button.bg-danger:hover{background-color:#bd2130!important}.bg-light{background-color:#f8f9fa!important}a.bg-light:focus,a.bg-light:hover,button.bg-light:focus,button.bg-light:hover{background-color:#dae0e5!important}.bg-dark{background-color:#343a40!important}a.bg-dark:focus,a.bg-dark:hover,button.bg-dark:focus,button.bg-dark:hover{background-color:#1d2124!important}.bg-white{background-color:#fff!important}.bg-transparent{background-color:transparent!important}.border{border:1px solid #dee2e6!important}.border-top{border-top:1px solid #dee2e6!important}.border-right{border-right:1px solid #dee2e6!important}.border-bottom{border-bottom:1px solid #dee2e6!important}.border-left{border-left:1px solid #dee2e6!important}.border-0{border:0!important}.border-top-0{border-top:0!important}.border-right-0{border-right:0!important}.border-bottom-0{border-bottom:0!important}.border-left-0{border-left:0!important}.border-primary{border-color:#007bff!important}.border-secondary{border-color:#6c757d!important}.border-success{border-color:#28a745!important}.border-info{border-color:#17a2b8!important}.border-warning{border-color:#ffc107!important}.border-danger{border-color:#dc3545!important}.border-light{border-color:#f8f9fa!important}.border-dark{border-color:#343a40!important}.border-white{border-color:#fff!important}.rounded-sm{border-radius:.2rem!important}.rounded{border-radius:.25rem!important}.rounded-top{border-top-left-radius:.25rem!important;border-top-right-radius:.25rem!important}.rounded-right{border-top-right-radius:.25rem!important;border-bottom-right-radius:.25rem!important}.rounded-bottom{border-bottom-right-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-left{border-top-left-radius:.25rem!important;border-bottom-left-radius:.25rem!important}.rounded-lg{border-radius:.3rem!important}.rounded-circle{border-radius:50%!important}.rounded-pill{border-radius:50rem!important}.rounded-0{border-radius:0!important}.clearfix::after{display:block;clear:both;content:""}.d-none{display:none!important}.d-inline{display:inline!important}.d-inline-block{display:inline-block!important}.d-block{display:block!important}.d-table{display:table!important}.d-table-row{display:table-row!important}.d-table-cell{display:table-cell!important}.d-flex{display:-ms-flexbox!important;display:flex!important}.d-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}@media (min-width:576px){.d-sm-none{display:none!important}.d-sm-inline{display:inline!important}.d-sm-inline-block{display:inline-block!important}.d-sm-block{display:block!important}.d-sm-table{display:table!important}.d-sm-table-row{display:table-row!important}.d-sm-table-cell{display:table-cell!important}.d-sm-flex{display:-ms-flexbox!important;display:flex!important}.d-sm-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:768px){.d-md-none{display:none!important}.d-md-inline{display:inline!important}.d-md-inline-block{display:inline-block!important}.d-md-block{display:block!important}.d-md-table{display:table!important}.d-md-table-row{display:table-row!important}.d-md-table-cell{display:table-cell!important}.d-md-flex{display:-ms-flexbox!important;display:flex!important}.d-md-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:992px){.d-lg-none{display:none!important}.d-lg-inline{display:inline!important}.d-lg-inline-block{display:inline-block!important}.d-lg-block{display:block!important}.d-lg-table{display:table!important}.d-lg-table-row{display:table-row!important}.d-lg-table-cell{display:table-cell!important}.d-lg-flex{display:-ms-flexbox!important;display:flex!important}.d-lg-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media (min-width:1200px){.d-xl-none{display:none!important}.d-xl-inline{display:inline!important}.d-xl-inline-block{display:inline-block!important}.d-xl-block{display:block!important}.d-xl-table{display:table!important}.d-xl-table-row{display:table-row!important}.d-xl-table-cell{display:table-cell!important}.d-xl-flex{display:-ms-flexbox!important;display:flex!important}.d-xl-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}@media print{.d-print-none{display:none!important}.d-print-inline{display:inline!important}.d-print-inline-block{display:inline-block!important}.d-print-block{display:block!important}.d-print-table{display:table!important}.d-print-table-row{display:table-row!important}.d-print-table-cell{display:table-cell!important}.d-print-flex{display:-ms-flexbox!important;display:flex!important}.d-print-inline-flex{display:-ms-inline-flexbox!important;display:inline-flex!important}}.embed-responsive{position:relative;display:block;width:100%;padding:0;overflow:hidden}.embed-responsive::before{display:block;content:""}.embed-responsive .embed-responsive-item,.embed-responsive embed,.embed-responsive iframe,.embed-responsive object,.embed-responsive video{position:absolute;top:0;bottom:0;left:0;width:100%;height:100%;border:0}.embed-responsive-21by9::before{padding-top:42.857143%}.embed-responsive-16by9::before{padding-top:56.25%}.embed-responsive-4by3::before{padding-top:75%}.embed-responsive-1by1::before{padding-top:100%}.flex-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-center{-ms-flex-align:center!important;align-items:center!important}.align-items-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}@media (min-width:576px){.flex-sm-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-sm-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-sm-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-sm-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-sm-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-sm-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-sm-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-sm-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-sm-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-sm-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-sm-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-sm-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-sm-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-sm-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-sm-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-sm-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-sm-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-sm-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-sm-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-sm-center{-ms-flex-align:center!important;align-items:center!important}.align-items-sm-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-sm-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-sm-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-sm-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-sm-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-sm-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-sm-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-sm-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-sm-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-sm-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-sm-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-sm-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-sm-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-sm-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:768px){.flex-md-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-md-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-md-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-md-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-md-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-md-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-md-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-md-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-md-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-md-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-md-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-md-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-md-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-md-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-md-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-md-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-md-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-md-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-md-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-md-center{-ms-flex-align:center!important;align-items:center!important}.align-items-md-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-md-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-md-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-md-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-md-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-md-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-md-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-md-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-md-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-md-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-md-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-md-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-md-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-md-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:992px){.flex-lg-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-lg-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-lg-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-lg-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-lg-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-lg-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-lg-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-lg-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-lg-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-lg-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-lg-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-lg-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-lg-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-lg-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-lg-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-lg-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-lg-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-lg-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-lg-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-lg-center{-ms-flex-align:center!important;align-items:center!important}.align-items-lg-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-lg-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-lg-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-lg-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-lg-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-lg-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-lg-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-lg-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-lg-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-lg-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-lg-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-lg-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-lg-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-lg-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}@media (min-width:1200px){.flex-xl-row{-ms-flex-direction:row!important;flex-direction:row!important}.flex-xl-column{-ms-flex-direction:column!important;flex-direction:column!important}.flex-xl-row-reverse{-ms-flex-direction:row-reverse!important;flex-direction:row-reverse!important}.flex-xl-column-reverse{-ms-flex-direction:column-reverse!important;flex-direction:column-reverse!important}.flex-xl-wrap{-ms-flex-wrap:wrap!important;flex-wrap:wrap!important}.flex-xl-nowrap{-ms-flex-wrap:nowrap!important;flex-wrap:nowrap!important}.flex-xl-wrap-reverse{-ms-flex-wrap:wrap-reverse!important;flex-wrap:wrap-reverse!important}.flex-xl-fill{-ms-flex:1 1 auto!important;flex:1 1 auto!important}.flex-xl-grow-0{-ms-flex-positive:0!important;flex-grow:0!important}.flex-xl-grow-1{-ms-flex-positive:1!important;flex-grow:1!important}.flex-xl-shrink-0{-ms-flex-negative:0!important;flex-shrink:0!important}.flex-xl-shrink-1{-ms-flex-negative:1!important;flex-shrink:1!important}.justify-content-xl-start{-ms-flex-pack:start!important;justify-content:flex-start!important}.justify-content-xl-end{-ms-flex-pack:end!important;justify-content:flex-end!important}.justify-content-xl-center{-ms-flex-pack:center!important;justify-content:center!important}.justify-content-xl-between{-ms-flex-pack:justify!important;justify-content:space-between!important}.justify-content-xl-around{-ms-flex-pack:distribute!important;justify-content:space-around!important}.align-items-xl-start{-ms-flex-align:start!important;align-items:flex-start!important}.align-items-xl-end{-ms-flex-align:end!important;align-items:flex-end!important}.align-items-xl-center{-ms-flex-align:center!important;align-items:center!important}.align-items-xl-baseline{-ms-flex-align:baseline!important;align-items:baseline!important}.align-items-xl-stretch{-ms-flex-align:stretch!important;align-items:stretch!important}.align-content-xl-start{-ms-flex-line-pack:start!important;align-content:flex-start!important}.align-content-xl-end{-ms-flex-line-pack:end!important;align-content:flex-end!important}.align-content-xl-center{-ms-flex-line-pack:center!important;align-content:center!important}.align-content-xl-between{-ms-flex-line-pack:justify!important;align-content:space-between!important}.align-content-xl-around{-ms-flex-line-pack:distribute!important;align-content:space-around!important}.align-content-xl-stretch{-ms-flex-line-pack:stretch!important;align-content:stretch!important}.align-self-xl-auto{-ms-flex-item-align:auto!important;align-self:auto!important}.align-self-xl-start{-ms-flex-item-align:start!important;align-self:flex-start!important}.align-self-xl-end{-ms-flex-item-align:end!important;align-self:flex-end!important}.align-self-xl-center{-ms-flex-item-align:center!important;align-self:center!important}.align-self-xl-baseline{-ms-flex-item-align:baseline!important;align-self:baseline!important}.align-self-xl-stretch{-ms-flex-item-align:stretch!important;align-self:stretch!important}}.float-left{float:left!important}.float-right{float:right!important}.float-none{float:none!important}@media (min-width:576px){.float-sm-left{float:left!important}.float-sm-right{float:right!important}.float-sm-none{float:none!important}}@media (min-width:768px){.float-md-left{float:left!important}.float-md-right{float:right!important}.float-md-none{float:none!important}}@media (min-width:992px){.float-lg-left{float:left!important}.float-lg-right{float:right!important}.float-lg-none{float:none!important}}@media (min-width:1200px){.float-xl-left{float:left!important}.float-xl-right{float:right!important}.float-xl-none{float:none!important}}.user-select-all{-webkit-user-select:all!important;-moz-user-select:all!important;-ms-user-select:all!important;user-select:all!important}.user-select-auto{-webkit-user-select:auto!important;-moz-user-select:auto!important;-ms-user-select:auto!important;user-select:auto!important}.user-select-none{-webkit-user-select:none!important;-moz-user-select:none!important;-ms-user-select:none!important;user-select:none!important}.overflow-auto{overflow:auto!important}.overflow-hidden{overflow:hidden!important}.position-static{position:static!important}.position-relative{position:relative!important}.position-absolute{position:absolute!important}.position-fixed{position:fixed!important}.position-sticky{position:-webkit-sticky!important;position:sticky!important}.fixed-top{position:fixed;top:0;right:0;left:0;z-index:1030}.fixed-bottom{position:fixed;right:0;bottom:0;left:0;z-index:1030}@supports ((position:-webkit-sticky) or (position:sticky)){.sticky-top{position:-webkit-sticky;position:sticky;top:0;z-index:1020}}.sr-only{position:absolute;width:1px;height:1px;padding:0;margin:-1px;overflow:hidden;clip:rect(0,0,0,0);white-space:nowrap;border:0}.sr-only-focusable:active,.sr-only-focusable:focus{position:static;width:auto;height:auto;overflow:visible;clip:auto;white-space:normal}.shadow-sm{box-shadow:0 .125rem .25rem rgba(0,0,0,.075)!important}.shadow{box-shadow:0 .5rem 1rem rgba(0,0,0,.15)!important}.shadow-lg{box-shadow:0 1rem 3rem rgba(0,0,0,.175)!important}.shadow-none{box-shadow:none!important}.w-25{width:25%!important}.w-50{width:50%!important}.w-75{width:75%!important}.w-100{width:100%!important}.w-auto{width:auto!important}.h-25{height:25%!important}.h-50{height:50%!important}.h-75{height:75%!important}.h-100{height:100%!important}.h-auto{height:auto!important}.mw-100{max-width:100%!important}.mh-100{max-height:100%!important}.min-vw-100{min-width:100vw!important}.min-vh-100{min-height:100vh!important}.vw-100{width:100vw!important}.vh-100{height:100vh!important}.m-0{margin:0!important}.mt-0,.my-0{margin-top:0!important}.mr-0,.mx-0{margin-right:0!important}.mb-0,.my-0{margin-bottom:0!important}.ml-0,.mx-0{margin-left:0!important}.m-1{margin:.25rem!important}.mt-1,.my-1{margin-top:.25rem!important}.mr-1,.mx-1{margin-right:.25rem!important}.mb-1,.my-1{margin-bottom:.25rem!important}.ml-1,.mx-1{margin-left:.25rem!important}.m-2{margin:.5rem!important}.mt-2,.my-2{margin-top:.5rem!important}.mr-2,.mx-2{margin-right:.5rem!important}.mb-2,.my-2{margin-bottom:.5rem!important}.ml-2,.mx-2{margin-left:.5rem!important}.m-3{margin:1rem!important}.mt-3,.my-3{margin-top:1rem!important}.mr-3,.mx-3{margin-right:1rem!important}.mb-3,.my-3{margin-bottom:1rem!important}.ml-3,.mx-3{margin-left:1rem!important}.m-4{margin:1.5rem!important}.mt-4,.my-4{margin-top:1.5rem!important}.mr-4,.mx-4{margin-right:1.5rem!important}.mb-4,.my-4{margin-bottom:1.5rem!important}.ml-4,.mx-4{margin-left:1.5rem!important}.m-5{margin:3rem!important}.mt-5,.my-5{margin-top:3rem!important}.mr-5,.mx-5{margin-right:3rem!important}.mb-5,.my-5{margin-bottom:3rem!important}.ml-5,.mx-5{margin-left:3rem!important}.p-0{padding:0!important}.pt-0,.py-0{padding-top:0!important}.pr-0,.px-0{padding-right:0!important}.pb-0,.py-0{padding-bottom:0!important}.pl-0,.px-0{padding-left:0!important}.p-1{padding:.25rem!important}.pt-1,.py-1{padding-top:.25rem!important}.pr-1,.px-1{padding-right:.25rem!important}.pb-1,.py-1{padding-bottom:.25rem!important}.pl-1,.px-1{padding-left:.25rem!important}.p-2{padding:.5rem!important}.pt-2,.py-2{padding-top:.5rem!important}.pr-2,.px-2{padding-right:.5rem!important}.pb-2,.py-2{padding-bottom:.5rem!important}.pl-2,.px-2{padding-left:.5rem!important}.p-3{padding:1rem!important}.pt-3,.py-3{padding-top:1rem!important}.pr-3,.px-3{padding-right:1rem!important}.pb-3,.py-3{padding-bottom:1rem!important}.pl-3,.px-3{padding-left:1rem!important}.p-4{padding:1.5rem!important}.pt-4,.py-4{padding-top:1.5rem!important}.pr-4,.px-4{padding-right:1.5rem!important}.pb-4,.py-4{padding-bottom:1.5rem!important}.pl-4,.px-4{padding-left:1.5rem!important}.p-5{padding:3rem!important}.pt-5,.py-5{padding-top:3rem!important}.pr-5,.px-5{padding-right:3rem!important}.pb-5,.py-5{padding-bottom:3rem!important}.pl-5,.px-5{padding-left:3rem!important}.m-n1{margin:-.25rem!important}.mt-n1,.my-n1{margin-top:-.25rem!important}.mr-n1,.mx-n1{margin-right:-.25rem!important}.mb-n1,.my-n1{margin-bottom:-.25rem!important}.ml-n1,.mx-n1{margin-left:-.25rem!important}.m-n2{margin:-.5rem!important}.mt-n2,.my-n2{margin-top:-.5rem!important}.mr-n2,.mx-n2{margin-right:-.5rem!important}.mb-n2,.my-n2{margin-bottom:-.5rem!important}.ml-n2,.mx-n2{margin-left:-.5rem!important}.m-n3{margin:-1rem!important}.mt-n3,.my-n3{margin-top:-1rem!important}.mr-n3,.mx-n3{margin-right:-1rem!important}.mb-n3,.my-n3{margin-bottom:-1rem!important}.ml-n3,.mx-n3{margin-left:-1rem!important}.m-n4{margin:-1.5rem!important}.mt-n4,.my-n4{margin-top:-1.5rem!important}.mr-n4,.mx-n4{margin-right:-1.5rem!important}.mb-n4,.my-n4{margin-bottom:-1.5rem!important}.ml-n4,.mx-n4{margin-left:-1.5rem!important}.m-n5{margin:-3rem!important}.mt-n5,.my-n5{margin-top:-3rem!important}.mr-n5,.mx-n5{margin-right:-3rem!important}.mb-n5,.my-n5{margin-bottom:-3rem!important}.ml-n5,.mx-n5{margin-left:-3rem!important}.m-auto{margin:auto!important}.mt-auto,.my-auto{margin-top:auto!important}.mr-auto,.mx-auto{margin-right:auto!important}.mb-auto,.my-auto{margin-bottom:auto!important}.ml-auto,.mx-auto{margin-left:auto!important}@media (min-width:576px){.m-sm-0{margin:0!important}.mt-sm-0,.my-sm-0{margin-top:0!important}.mr-sm-0,.mx-sm-0{margin-right:0!important}.mb-sm-0,.my-sm-0{margin-bottom:0!important}.ml-sm-0,.mx-sm-0{margin-left:0!important}.m-sm-1{margin:.25rem!important}.mt-sm-1,.my-sm-1{margin-top:.25rem!important}.mr-sm-1,.mx-sm-1{margin-right:.25rem!important}.mb-sm-1,.my-sm-1{margin-bottom:.25rem!important}.ml-sm-1,.mx-sm-1{margin-left:.25rem!important}.m-sm-2{margin:.5rem!important}.mt-sm-2,.my-sm-2{margin-top:.5rem!important}.mr-sm-2,.mx-sm-2{margin-right:.5rem!important}.mb-sm-2,.my-sm-2{margin-bottom:.5rem!important}.ml-sm-2,.mx-sm-2{margin-left:.5rem!important}.m-sm-3{margin:1rem!important}.mt-sm-3,.my-sm-3{margin-top:1rem!important}.mr-sm-3,.mx-sm-3{margin-right:1rem!important}.mb-sm-3,.my-sm-3{margin-bottom:1rem!important}.ml-sm-3,.mx-sm-3{margin-left:1rem!important}.m-sm-4{margin:1.5rem!important}.mt-sm-4,.my-sm-4{margin-top:1.5rem!important}.mr-sm-4,.mx-sm-4{margin-right:1.5rem!important}.mb-sm-4,.my-sm-4{margin-bottom:1.5rem!important}.ml-sm-4,.mx-sm-4{margin-left:1.5rem!important}.m-sm-5{margin:3rem!important}.mt-sm-5,.my-sm-5{margin-top:3rem!important}.mr-sm-5,.mx-sm-5{margin-right:3rem!important}.mb-sm-5,.my-sm-5{margin-bottom:3rem!important}.ml-sm-5,.mx-sm-5{margin-left:3rem!important}.p-sm-0{padding:0!important}.pt-sm-0,.py-sm-0{padding-top:0!important}.pr-sm-0,.px-sm-0{padding-right:0!important}.pb-sm-0,.py-sm-0{padding-bottom:0!important}.pl-sm-0,.px-sm-0{padding-left:0!important}.p-sm-1{padding:.25rem!important}.pt-sm-1,.py-sm-1{padding-top:.25rem!important}.pr-sm-1,.px-sm-1{padding-right:.25rem!important}.pb-sm-1,.py-sm-1{padding-bottom:.25rem!important}.pl-sm-1,.px-sm-1{padding-left:.25rem!important}.p-sm-2{padding:.5rem!important}.pt-sm-2,.py-sm-2{padding-top:.5rem!important}.pr-sm-2,.px-sm-2{padding-right:.5rem!important}.pb-sm-2,.py-sm-2{padding-bottom:.5rem!important}.pl-sm-2,.px-sm-2{padding-left:.5rem!important}.p-sm-3{padding:1rem!important}.pt-sm-3,.py-sm-3{padding-top:1rem!important}.pr-sm-3,.px-sm-3{padding-right:1rem!important}.pb-sm-3,.py-sm-3{padding-bottom:1rem!important}.pl-sm-3,.px-sm-3{padding-left:1rem!important}.p-sm-4{padding:1.5rem!important}.pt-sm-4,.py-sm-4{padding-top:1.5rem!important}.pr-sm-4,.px-sm-4{padding-right:1.5rem!important}.pb-sm-4,.py-sm-4{padding-bottom:1.5rem!important}.pl-sm-4,.px-sm-4{padding-left:1.5rem!important}.p-sm-5{padding:3rem!important}.pt-sm-5,.py-sm-5{padding-top:3rem!important}.pr-sm-5,.px-sm-5{padding-right:3rem!important}.pb-sm-5,.py-sm-5{padding-bottom:3rem!important}.pl-sm-5,.px-sm-5{padding-left:3rem!important}.m-sm-n1{margin:-.25rem!important}.mt-sm-n1,.my-sm-n1{margin-top:-.25rem!important}.mr-sm-n1,.mx-sm-n1{margin-right:-.25rem!important}.mb-sm-n1,.my-sm-n1{margin-bottom:-.25rem!important}.ml-sm-n1,.mx-sm-n1{margin-left:-.25rem!important}.m-sm-n2{margin:-.5rem!important}.mt-sm-n2,.my-sm-n2{margin-top:-.5rem!important}.mr-sm-n2,.mx-sm-n2{margin-right:-.5rem!important}.mb-sm-n2,.my-sm-n2{margin-bottom:-.5rem!important}.ml-sm-n2,.mx-sm-n2{margin-left:-.5rem!important}.m-sm-n3{margin:-1rem!important}.mt-sm-n3,.my-sm-n3{margin-top:-1rem!important}.mr-sm-n3,.mx-sm-n3{margin-right:-1rem!important}.mb-sm-n3,.my-sm-n3{margin-bottom:-1rem!important}.ml-sm-n3,.mx-sm-n3{margin-left:-1rem!important}.m-sm-n4{margin:-1.5rem!important}.mt-sm-n4,.my-sm-n4{margin-top:-1.5rem!important}.mr-sm-n4,.mx-sm-n4{margin-right:-1.5rem!important}.mb-sm-n4,.my-sm-n4{margin-bottom:-1.5rem!important}.ml-sm-n4,.mx-sm-n4{margin-left:-1.5rem!important}.m-sm-n5{margin:-3rem!important}.mt-sm-n5,.my-sm-n5{margin-top:-3rem!important}.mr-sm-n5,.mx-sm-n5{margin-right:-3rem!important}.mb-sm-n5,.my-sm-n5{margin-bottom:-3rem!important}.ml-sm-n5,.mx-sm-n5{margin-left:-3rem!important}.m-sm-auto{margin:auto!important}.mt-sm-auto,.my-sm-auto{margin-top:auto!important}.mr-sm-auto,.mx-sm-auto{margin-right:auto!important}.mb-sm-auto,.my-sm-auto{margin-bottom:auto!important}.ml-sm-auto,.mx-sm-auto{margin-left:auto!important}}@media (min-width:768px){.m-md-0{margin:0!important}.mt-md-0,.my-md-0{margin-top:0!important}.mr-md-0,.mx-md-0{margin-right:0!important}.mb-md-0,.my-md-0{margin-bottom:0!important}.ml-md-0,.mx-md-0{margin-left:0!important}.m-md-1{margin:.25rem!important}.mt-md-1,.my-md-1{margin-top:.25rem!important}.mr-md-1,.mx-md-1{margin-right:.25rem!important}.mb-md-1,.my-md-1{margin-bottom:.25rem!important}.ml-md-1,.mx-md-1{margin-left:.25rem!important}.m-md-2{margin:.5rem!important}.mt-md-2,.my-md-2{margin-top:.5rem!important}.mr-md-2,.mx-md-2{margin-right:.5rem!important}.mb-md-2,.my-md-2{margin-bottom:.5rem!important}.ml-md-2,.mx-md-2{margin-left:.5rem!important}.m-md-3{margin:1rem!important}.mt-md-3,.my-md-3{margin-top:1rem!important}.mr-md-3,.mx-md-3{margin-right:1rem!important}.mb-md-3,.my-md-3{margin-bottom:1rem!important}.ml-md-3,.mx-md-3{margin-left:1rem!important}.m-md-4{margin:1.5rem!important}.mt-md-4,.my-md-4{margin-top:1.5rem!important}.mr-md-4,.mx-md-4{margin-right:1.5rem!important}.mb-md-4,.my-md-4{margin-bottom:1.5rem!important}.ml-md-4,.mx-md-4{margin-left:1.5rem!important}.m-md-5{margin:3rem!important}.mt-md-5,.my-md-5{margin-top:3rem!important}.mr-md-5,.mx-md-5{margin-right:3rem!important}.mb-md-5,.my-md-5{margin-bottom:3rem!important}.ml-md-5,.mx-md-5{margin-left:3rem!important}.p-md-0{padding:0!important}.pt-md-0,.py-md-0{padding-top:0!important}.pr-md-0,.px-md-0{padding-right:0!important}.pb-md-0,.py-md-0{padding-bottom:0!important}.pl-md-0,.px-md-0{padding-left:0!important}.p-md-1{padding:.25rem!important}.pt-md-1,.py-md-1{padding-top:.25rem!important}.pr-md-1,.px-md-1{padding-right:.25rem!important}.pb-md-1,.py-md-1{padding-bottom:.25rem!important}.pl-md-1,.px-md-1{padding-left:.25rem!important}.p-md-2{padding:.5rem!important}.pt-md-2,.py-md-2{padding-top:.5rem!important}.pr-md-2,.px-md-2{padding-right:.5rem!important}.pb-md-2,.py-md-2{padding-bottom:.5rem!important}.pl-md-2,.px-md-2{padding-left:.5rem!important}.p-md-3{padding:1rem!important}.pt-md-3,.py-md-3{padding-top:1rem!important}.pr-md-3,.px-md-3{padding-right:1rem!important}.pb-md-3,.py-md-3{padding-bottom:1rem!important}.pl-md-3,.px-md-3{padding-left:1rem!important}.p-md-4{padding:1.5rem!important}.pt-md-4,.py-md-4{padding-top:1.5rem!important}.pr-md-4,.px-md-4{padding-right:1.5rem!important}.pb-md-4,.py-md-4{padding-bottom:1.5rem!important}.pl-md-4,.px-md-4{padding-left:1.5rem!important}.p-md-5{padding:3rem!important}.pt-md-5,.py-md-5{padding-top:3rem!important}.pr-md-5,.px-md-5{padding-right:3rem!important}.pb-md-5,.py-md-5{padding-bottom:3rem!important}.pl-md-5,.px-md-5{padding-left:3rem!important}.m-md-n1{margin:-.25rem!important}.mt-md-n1,.my-md-n1{margin-top:-.25rem!important}.mr-md-n1,.mx-md-n1{margin-right:-.25rem!important}.mb-md-n1,.my-md-n1{margin-bottom:-.25rem!important}.ml-md-n1,.mx-md-n1{margin-left:-.25rem!important}.m-md-n2{margin:-.5rem!important}.mt-md-n2,.my-md-n2{margin-top:-.5rem!important}.mr-md-n2,.mx-md-n2{margin-right:-.5rem!important}.mb-md-n2,.my-md-n2{margin-bottom:-.5rem!important}.ml-md-n2,.mx-md-n2{margin-left:-.5rem!important}.m-md-n3{margin:-1rem!important}.mt-md-n3,.my-md-n3{margin-top:-1rem!important}.mr-md-n3,.mx-md-n3{margin-right:-1rem!important}.mb-md-n3,.my-md-n3{margin-bottom:-1rem!important}.ml-md-n3,.mx-md-n3{margin-left:-1rem!important}.m-md-n4{margin:-1.5rem!important}.mt-md-n4,.my-md-n4{margin-top:-1.5rem!important}.mr-md-n4,.mx-md-n4{margin-right:-1.5rem!important}.mb-md-n4,.my-md-n4{margin-bottom:-1.5rem!important}.ml-md-n4,.mx-md-n4{margin-left:-1.5rem!important}.m-md-n5{margin:-3rem!important}.mt-md-n5,.my-md-n5{margin-top:-3rem!important}.mr-md-n5,.mx-md-n5{margin-right:-3rem!important}.mb-md-n5,.my-md-n5{margin-bottom:-3rem!important}.ml-md-n5,.mx-md-n5{margin-left:-3rem!important}.m-md-auto{margin:auto!important}.mt-md-auto,.my-md-auto{margin-top:auto!important}.mr-md-auto,.mx-md-auto{margin-right:auto!important}.mb-md-auto,.my-md-auto{margin-bottom:auto!important}.ml-md-auto,.mx-md-auto{margin-left:auto!important}}@media (min-width:992px){.m-lg-0{margin:0!important}.mt-lg-0,.my-lg-0{margin-top:0!important}.mr-lg-0,.mx-lg-0{margin-right:0!important}.mb-lg-0,.my-lg-0{margin-bottom:0!important}.ml-lg-0,.mx-lg-0{margin-left:0!important}.m-lg-1{margin:.25rem!important}.mt-lg-1,.my-lg-1{margin-top:.25rem!important}.mr-lg-1,.mx-lg-1{margin-right:.25rem!important}.mb-lg-1,.my-lg-1{margin-bottom:.25rem!important}.ml-lg-1,.mx-lg-1{margin-left:.25rem!important}.m-lg-2{margin:.5rem!important}.mt-lg-2,.my-lg-2{margin-top:.5rem!important}.mr-lg-2,.mx-lg-2{margin-right:.5rem!important}.mb-lg-2,.my-lg-2{margin-bottom:.5rem!important}.ml-lg-2,.mx-lg-2{margin-left:.5rem!important}.m-lg-3{margin:1rem!important}.mt-lg-3,.my-lg-3{margin-top:1rem!important}.mr-lg-3,.mx-lg-3{margin-right:1rem!important}.mb-lg-3,.my-lg-3{margin-bottom:1rem!important}.ml-lg-3,.mx-lg-3{margin-left:1rem!important}.m-lg-4{margin:1.5rem!important}.mt-lg-4,.my-lg-4{margin-top:1.5rem!important}.mr-lg-4,.mx-lg-4{margin-right:1.5rem!important}.mb-lg-4,.my-lg-4{margin-bottom:1.5rem!important}.ml-lg-4,.mx-lg-4{margin-left:1.5rem!important}.m-lg-5{margin:3rem!important}.mt-lg-5,.my-lg-5{margin-top:3rem!important}.mr-lg-5,.mx-lg-5{margin-right:3rem!important}.mb-lg-5,.my-lg-5{margin-bottom:3rem!important}.ml-lg-5,.mx-lg-5{margin-left:3rem!important}.p-lg-0{padding:0!important}.pt-lg-0,.py-lg-0{padding-top:0!important}.pr-lg-0,.px-lg-0{padding-right:0!important}.pb-lg-0,.py-lg-0{padding-bottom:0!important}.pl-lg-0,.px-lg-0{padding-left:0!important}.p-lg-1{padding:.25rem!important}.pt-lg-1,.py-lg-1{padding-top:.25rem!important}.pr-lg-1,.px-lg-1{padding-right:.25rem!important}.pb-lg-1,.py-lg-1{padding-bottom:.25rem!important}.pl-lg-1,.px-lg-1{padding-left:.25rem!important}.p-lg-2{padding:.5rem!important}.pt-lg-2,.py-lg-2{padding-top:.5rem!important}.pr-lg-2,.px-lg-2{padding-right:.5rem!important}.pb-lg-2,.py-lg-2{padding-bottom:.5rem!important}.pl-lg-2,.px-lg-2{padding-left:.5rem!important}.p-lg-3{padding:1rem!important}.pt-lg-3,.py-lg-3{padding-top:1rem!important}.pr-lg-3,.px-lg-3{padding-right:1rem!important}.pb-lg-3,.py-lg-3{padding-bottom:1rem!important}.pl-lg-3,.px-lg-3{padding-left:1rem!important}.p-lg-4{padding:1.5rem!important}.pt-lg-4,.py-lg-4{padding-top:1.5rem!important}.pr-lg-4,.px-lg-4{padding-right:1.5rem!important}.pb-lg-4,.py-lg-4{padding-bottom:1.5rem!important}.pl-lg-4,.px-lg-4{padding-left:1.5rem!important}.p-lg-5{padding:3rem!important}.pt-lg-5,.py-lg-5{padding-top:3rem!important}.pr-lg-5,.px-lg-5{padding-right:3rem!important}.pb-lg-5,.py-lg-5{padding-bottom:3rem!important}.pl-lg-5,.px-lg-5{padding-left:3rem!important}.m-lg-n1{margin:-.25rem!important}.mt-lg-n1,.my-lg-n1{margin-top:-.25rem!important}.mr-lg-n1,.mx-lg-n1{margin-right:-.25rem!important}.mb-lg-n1,.my-lg-n1{margin-bottom:-.25rem!important}.ml-lg-n1,.mx-lg-n1{margin-left:-.25rem!important}.m-lg-n2{margin:-.5rem!important}.mt-lg-n2,.my-lg-n2{margin-top:-.5rem!important}.mr-lg-n2,.mx-lg-n2{margin-right:-.5rem!important}.mb-lg-n2,.my-lg-n2{margin-bottom:-.5rem!important}.ml-lg-n2,.mx-lg-n2{margin-left:-.5rem!important}.m-lg-n3{margin:-1rem!important}.mt-lg-n3,.my-lg-n3{margin-top:-1rem!important}.mr-lg-n3,.mx-lg-n3{margin-right:-1rem!important}.mb-lg-n3,.my-lg-n3{margin-bottom:-1rem!important}.ml-lg-n3,.mx-lg-n3{margin-left:-1rem!important}.m-lg-n4{margin:-1.5rem!important}.mt-lg-n4,.my-lg-n4{margin-top:-1.5rem!important}.mr-lg-n4,.mx-lg-n4{margin-right:-1.5rem!important}.mb-lg-n4,.my-lg-n4{margin-bottom:-1.5rem!important}.ml-lg-n4,.mx-lg-n4{margin-left:-1.5rem!important}.m-lg-n5{margin:-3rem!important}.mt-lg-n5,.my-lg-n5{margin-top:-3rem!important}.mr-lg-n5,.mx-lg-n5{margin-right:-3rem!important}.mb-lg-n5,.my-lg-n5{margin-bottom:-3rem!important}.ml-lg-n5,.mx-lg-n5{margin-left:-3rem!important}.m-lg-auto{margin:auto!important}.mt-lg-auto,.my-lg-auto{margin-top:auto!important}.mr-lg-auto,.mx-lg-auto{margin-right:auto!important}.mb-lg-auto,.my-lg-auto{margin-bottom:auto!important}.ml-lg-auto,.mx-lg-auto{margin-left:auto!important}}@media (min-width:1200px){.m-xl-0{margin:0!important}.mt-xl-0,.my-xl-0{margin-top:0!important}.mr-xl-0,.mx-xl-0{margin-right:0!important}.mb-xl-0,.my-xl-0{margin-bottom:0!important}.ml-xl-0,.mx-xl-0{margin-left:0!important}.m-xl-1{margin:.25rem!important}.mt-xl-1,.my-xl-1{margin-top:.25rem!important}.mr-xl-1,.mx-xl-1{margin-right:.25rem!important}.mb-xl-1,.my-xl-1{margin-bottom:.25rem!important}.ml-xl-1,.mx-xl-1{margin-left:.25rem!important}.m-xl-2{margin:.5rem!important}.mt-xl-2,.my-xl-2{margin-top:.5rem!important}.mr-xl-2,.mx-xl-2{margin-right:.5rem!important}.mb-xl-2,.my-xl-2{margin-bottom:.5rem!important}.ml-xl-2,.mx-xl-2{margin-left:.5rem!important}.m-xl-3{margin:1rem!important}.mt-xl-3,.my-xl-3{margin-top:1rem!important}.mr-xl-3,.mx-xl-3{margin-right:1rem!important}.mb-xl-3,.my-xl-3{margin-bottom:1rem!important}.ml-xl-3,.mx-xl-3{margin-left:1rem!important}.m-xl-4{margin:1.5rem!important}.mt-xl-4,.my-xl-4{margin-top:1.5rem!important}.mr-xl-4,.mx-xl-4{margin-right:1.5rem!important}.mb-xl-4,.my-xl-4{margin-bottom:1.5rem!important}.ml-xl-4,.mx-xl-4{margin-left:1.5rem!important}.m-xl-5{margin:3rem!important}.mt-xl-5,.my-xl-5{margin-top:3rem!important}.mr-xl-5,.mx-xl-5{margin-right:3rem!important}.mb-xl-5,.my-xl-5{margin-bottom:3rem!important}.ml-xl-5,.mx-xl-5{margin-left:3rem!important}.p-xl-0{padding:0!important}.pt-xl-0,.py-xl-0{padding-top:0!important}.pr-xl-0,.px-xl-0{padding-right:0!important}.pb-xl-0,.py-xl-0{padding-bottom:0!important}.pl-xl-0,.px-xl-0{padding-left:0!important}.p-xl-1{padding:.25rem!important}.pt-xl-1,.py-xl-1{padding-top:.25rem!important}.pr-xl-1,.px-xl-1{padding-right:.25rem!important}.pb-xl-1,.py-xl-1{padding-bottom:.25rem!important}.pl-xl-1,.px-xl-1{padding-left:.25rem!important}.p-xl-2{padding:.5rem!important}.pt-xl-2,.py-xl-2{padding-top:.5rem!important}.pr-xl-2,.px-xl-2{padding-right:.5rem!important}.pb-xl-2,.py-xl-2{padding-bottom:.5rem!important}.pl-xl-2,.px-xl-2{padding-left:.5rem!important}.p-xl-3{padding:1rem!important}.pt-xl-3,.py-xl-3{padding-top:1rem!important}.pr-xl-3,.px-xl-3{padding-right:1rem!important}.pb-xl-3,.py-xl-3{padding-bottom:1rem!important}.pl-xl-3,.px-xl-3{padding-left:1rem!important}.p-xl-4{padding:1.5rem!important}.pt-xl-4,.py-xl-4{padding-top:1.5rem!important}.pr-xl-4,.px-xl-4{padding-right:1.5rem!important}.pb-xl-4,.py-xl-4{padding-bottom:1.5rem!important}.pl-xl-4,.px-xl-4{padding-left:1.5rem!important}.p-xl-5{padding:3rem!important}.pt-xl-5,.py-xl-5{padding-top:3rem!important}.pr-xl-5,.px-xl-5{padding-right:3rem!important}.pb-xl-5,.py-xl-5{padding-bottom:3rem!important}.pl-xl-5,.px-xl-5{padding-left:3rem!important}.m-xl-n1{margin:-.25rem!important}.mt-xl-n1,.my-xl-n1{margin-top:-.25rem!important}.mr-xl-n1,.mx-xl-n1{margin-right:-.25rem!important}.mb-xl-n1,.my-xl-n1{margin-bottom:-.25rem!important}.ml-xl-n1,.mx-xl-n1{margin-left:-.25rem!important}.m-xl-n2{margin:-.5rem!important}.mt-xl-n2,.my-xl-n2{margin-top:-.5rem!important}.mr-xl-n2,.mx-xl-n2{margin-right:-.5rem!important}.mb-xl-n2,.my-xl-n2{margin-bottom:-.5rem!important}.ml-xl-n2,.mx-xl-n2{margin-left:-.5rem!important}.m-xl-n3{margin:-1rem!important}.mt-xl-n3,.my-xl-n3{margin-top:-1rem!important}.mr-xl-n3,.mx-xl-n3{margin-right:-1rem!important}.mb-xl-n3,.my-xl-n3{margin-bottom:-1rem!important}.ml-xl-n3,.mx-xl-n3{margin-left:-1rem!important}.m-xl-n4{margin:-1.5rem!important}.mt-xl-n4,.my-xl-n4{margin-top:-1.5rem!important}.mr-xl-n4,.mx-xl-n4{margin-right:-1.5rem!important}.mb-xl-n4,.my-xl-n4{margin-bottom:-1.5rem!important}.ml-xl-n4,.mx-xl-n4{margin-left:-1.5rem!important}.m-xl-n5{margin:-3rem!important}.mt-xl-n5,.my-xl-n5{margin-top:-3rem!important}.mr-xl-n5,.mx-xl-n5{margin-right:-3rem!important}.mb-xl-n5,.my-xl-n5{margin-bottom:-3rem!important}.ml-xl-n5,.mx-xl-n5{margin-left:-3rem!important}.m-xl-auto{margin:auto!important}.mt-xl-auto,.my-xl-auto{margin-top:auto!important}.mr-xl-auto,.mx-xl-auto{margin-right:auto!important}.mb-xl-auto,.my-xl-auto{margin-bottom:auto!important}.ml-xl-auto,.mx-xl-auto{margin-left:auto!important}}.stretched-link::after{position:absolute;top:0;right:0;bottom:0;left:0;z-index:1;pointer-events:auto;content:"";background-color:rgba(0,0,0,0)}.text-monospace{font-family:SFMono-Regular,Menlo,Monaco,Consolas,"Liberation Mono","Courier New",monospace!important}.text-justify{text-align:justify!important}.text-wrap{white-space:normal!important}.text-nowrap{white-space:nowrap!important}.text-truncate{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.text-left{text-align:left!important}.text-right{text-align:right!important}.text-center{text-align:center!important}@media (min-width:576px){.text-sm-left{text-align:left!important}.text-sm-right{text-align:right!important}.text-sm-center{text-align:center!important}}@media (min-width:768px){.text-md-left{text-align:left!important}.text-md-right{text-align:right!important}.text-md-center{text-align:center!important}}@media (min-width:992px){.text-lg-left{text-align:left!important}.text-lg-right{text-align:right!important}.text-lg-center{text-align:center!important}}@media (min-width:1200px){.text-xl-left{text-align:left!important}.text-xl-right{text-align:right!important}.text-xl-center{text-align:center!important}}.text-lowercase{text-transform:lowercase!important}.text-uppercase{text-transform:uppercase!important}.text-capitalize{text-transform:capitalize!important}.font-weight-light{font-weight:300!important}.font-weight-lighter{font-weight:lighter!important}.font-weight-normal{font-weight:400!important}.font-weight-bold{font-weight:700!important}.font-weight-bolder{font-weight:bolder!important}.font-italic{font-style:italic!important}.text-white{color:#fff!important}.text-primary{color:#007bff!important}a.text-primary:focus,a.text-primary:hover{color:#0056b3!important}.text-secondary{color:#6c757d!important}a.text-secondary:focus,a.text-secondary:hover{color:#494f54!important}.text-success{color:#28a745!important}a.text-success:focus,a.text-success:hover{color:#19692c!important}.text-info{color:#17a2b8!important}a.text-info:focus,a.text-info:hover{color:#0f6674!important}.text-warning{color:#ffc107!important}a.text-warning:focus,a.text-warning:hover{color:#ba8b00!important}.text-danger{color:#dc3545!important}a.text-danger:focus,a.text-danger:hover{color:#a71d2a!important}.text-light{color:#f8f9fa!important}a.text-light:focus,a.text-light:hover{color:#cbd3da!important}.text-dark{color:#343a40!important}a.text-dark:focus,a.text-dark:hover{color:#121416!important}.text-body{color:#212529!important}.text-muted{color:#6c757d!important}.text-black-50{color:rgba(0,0,0,.5)!important}.text-white-50{color:rgba(255,255,255,.5)!important}.text-hide{font:0/0 a;color:transparent;text-shadow:none;background-color:transparent;border:0}.text-decoration-none{text-decoration:none!important}.text-break{word-break:break-word!important;word-wrap:break-word!important}.text-reset{color:inherit!important}.visible{visibility:visible!important}.invisible{visibility:hidden!important}@media print{*,::after,::before{text-shadow:none!important;box-shadow:none!important}a:not(.btn){text-decoration:underline}abbr[title]::after{content:" (" attr(title) ")"}pre{white-space:pre-wrap!important}blockquote,pre{border:1px solid #adb5bd;page-break-inside:avoid}thead{display:table-header-group}img,tr{page-break-inside:avoid}h2,h3,p{orphans:3;widows:3}h2,h3{page-break-after:avoid}@page{size:a3}body{min-width:992px!important}.container{min-width:992px!important}.navbar{display:none}.badge{border:1px solid #000}.table{border-collapse:collapse!important}.table td,.table th{background-color:#fff!important}.table-bordered td,.table-bordered th{border:1px solid #dee2e6!important}.table-dark{color:inherit}.table-dark tbody+tbody,.table-dark td,.table-dark th,.table-dark thead th{border-color:#dee2e6}.table .thead-dark th{color:inherit;border-color:#dee2e6}} -/*# sourceMappingURL=bootstrap.min.css.map */ \ No newline at end of file +/*# sourceMappingURL=bootstrap.min.css.map */ diff --git a/src/documents/static/signin.css b/src/documents/static/signin.css index 7441e0394..981200995 100644 --- a/src/documents/static/signin.css +++ b/src/documents/static/signin.css @@ -42,3 +42,58 @@ body { border-top-left-radius: 0; border-top-right-radius: 0; } + +@media (prefers-color-scheme: dark) { + /* + From theme_dark.scss + $primary-dark-mode: #45973a; + $danger-dark-mode: #b71631; + $bg-dark-mode: #161618; + $bg-dark-mode-accent: #21262d; + $bg-light-dark-mode: #1c1c1f; + $text-color-dark-mode: #abb2bf; + $border-color-dark-mode: #47494f; + */ + body { + background-color: #161618 !important; + color: #abb2bf; + } + + svg.logo .text { + fill: #abb2bf!important; + } + + .form-control:not(.is-invalid):not(.btn) { + border-color: #47494f; + } + + .form-control:not(.btn) { + background-color: #161618; + color: #abb2bf; + } + + .form-control:not(.btn)::placeholder { + color: #abb2bf; + } + + .form-control:not(.btn):focus { + background-color: #1c1c1f !important; + color: #8e97a9 !important; + } + + .btn-primary { + color: #fff; + background-color: #17541f; + border-color: #17541f; + } + + .btn-primary:hover, .btn-primary:focus { + background-color: #0f3614; + border-color: #0c2c10; + } + + .btn-primary:not(:disabled):not(.disabled):active { + background-color: #0c2c10; + border-color: #09220d; + } +} diff --git a/src/documents/tasks.py b/src/documents/tasks.py index 8c9b00dd6..f24be562f 100644 --- a/src/documents/tasks.py +++ b/src/documents/tasks.py @@ -2,14 +2,16 @@ import logging import tqdm from django.conf import settings +from django.db.models.signals import post_save from whoosh.writing import AsyncWriter from documents import index, sanity_checker -from documents.classifier import DocumentClassifier, \ - IncompatibleClassifierVersionError +from documents.classifier import DocumentClassifier, load_classifier from documents.consumer import Consumer, ConsumerError -from documents.models import Document -from documents.sanity_checker import SanityFailedError +from documents.models import Document, Tag, DocumentType, Correspondent +from documents.sanity_checker import SanityCheckFailedException + +logger = logging.getLogger("paperless.tasks") def index_optimize(): @@ -18,40 +20,45 @@ def index_optimize(): writer.commit(optimize=True) -def index_reindex(): +def index_reindex(progress_bar_disable=False): documents = Document.objects.all() ix = index.open_index(recreate=True) with AsyncWriter(ix) as writer: - for document in tqdm.tqdm(documents): + for document in tqdm.tqdm(documents, disable=progress_bar_disable): index.update_document(writer, document) def train_classifier(): - classifier = DocumentClassifier() + if (not Tag.objects.filter( + matching_algorithm=Tag.MATCH_AUTO).exists() and + not DocumentType.objects.filter( + matching_algorithm=Tag.MATCH_AUTO).exists() and + not Correspondent.objects.filter( + matching_algorithm=Tag.MATCH_AUTO).exists()): - try: - # load the classifier, since we might not have to train it again. - classifier.reload() - except (FileNotFoundError, IncompatibleClassifierVersionError): - # This is what we're going to fix here. - pass + return + + classifier = load_classifier() + + if not classifier: + classifier = DocumentClassifier() try: if classifier.train(): - logging.getLogger(__name__).info( + logger.info( "Saving updated classifier model to {}...".format( settings.MODEL_FILE) ) - classifier.save_classifier() + classifier.save() else: - logging.getLogger(__name__).debug( + logger.debug( "Training data unchanged." ) except Exception as e: - logging.getLogger(__name__).error( + logger.warning( "Classifier error: " + str(e) ) @@ -61,7 +68,8 @@ def consume_file(path, override_title=None, override_correspondent_id=None, override_document_type_id=None, - override_tag_ids=None): + override_tag_ids=None, + task_id=None): document = Consumer().try_consume_file( path, @@ -69,7 +77,9 @@ def consume_file(path, override_title=override_title, override_correspondent_id=override_correspondent_id, override_document_type_id=override_document_type_id, - override_tag_ids=override_tag_ids) + override_tag_ids=override_tag_ids, + task_id=task_id + ) if document: return "Success. New document id {} created".format( @@ -83,7 +93,27 @@ def consume_file(path, def sanity_check(): messages = sanity_checker.check_sanity() - if len(messages) > 0: - raise SanityFailedError(messages) + messages.log_messages() + + if messages.has_error(): + raise SanityCheckFailedException( + "Sanity check failed with errors. See log.") + elif messages.has_warning(): + return "Sanity check exited with warnings. See log." + elif len(messages) > 0: + return "Sanity check exited with infos. See log." else: return "No issues detected." + + +def bulk_update_documents(document_ids): + documents = Document.objects.filter(id__in=document_ids) + + ix = index.open_index() + + for doc in documents: + post_save.send(Document, instance=doc, created=False) + + with AsyncWriter(ix) as writer: + for doc in documents: + index.update_document(writer, doc) diff --git a/src/documents/templates/index.html b/src/documents/templates/index.html index 728f3a0e7..b8f60a9a5 100644 --- a/src/documents/templates/index.html +++ b/src/documents/templates/index.html @@ -1,19 +1,27 @@ <!doctype html> {% load static %} +{% load i18n %} <html lang="en"> <head> <meta charset="utf-8"> - <title>PaperlessUi - + Paperless-ng + + + + + - + + + + - Loading... - - - + {% translate "Paperless-ng is loading..." %} + + + diff --git a/src/documents/templates/registration/logged_out.html b/src/documents/templates/registration/logged_out.html index 4a3cb60bb..b6438a34a 100644 --- a/src/documents/templates/registration/logged_out.html +++ b/src/documents/templates/registration/logged_out.html @@ -1,6 +1,7 @@ {% load static %} +{% load i18n %} @@ -9,7 +10,8 @@ - Paperless Sign In + + {% translate "Paperless-ng signed out" %} @@ -36,9 +38,12 @@ diff --git a/src/documents/templates/registration/login.html b/src/documents/templates/registration/login.html index 25a707bed..0b117bd5c 100644 --- a/src/documents/templates/registration/login.html +++ b/src/documents/templates/registration/login.html @@ -1,6 +1,7 @@ {% load static %} +{% load i18n %} @@ -9,7 +10,9 @@ - Paperless Sign In + + + {% translate "Paperless-ng sign in" %} @@ -37,18 +40,23 @@ diff --git a/src/documents/tests/data/model.pickle b/src/documents/tests/data/model.pickle new file mode 100644 index 000000000..db303ec80 Binary files /dev/null and b/src/documents/tests/data/model.pickle differ diff --git a/src/documents/tests/samples/documents/originals/0000002.pdf b/src/documents/tests/samples/documents/originals/0000002.pdf new file mode 100644 index 000000000..5e75266ca Binary files /dev/null and b/src/documents/tests/samples/documents/originals/0000002.pdf differ diff --git a/src/documents/tests/samples/documents/originals/0000002.pdf.gpg b/src/documents/tests/samples/documents/originals/0000002.pdf.gpg deleted file mode 100644 index 0322a8039..000000000 Binary files a/src/documents/tests/samples/documents/originals/0000002.pdf.gpg and /dev/null differ diff --git a/src/documents/tests/samples/documents/originals/0000003.pdf b/src/documents/tests/samples/documents/originals/0000003.pdf new file mode 100644 index 000000000..afbeef5c8 Binary files /dev/null and b/src/documents/tests/samples/documents/originals/0000003.pdf differ diff --git a/src/documents/tests/samples/documents/originals/0000004.pdf.gpg b/src/documents/tests/samples/documents/originals/0000004.pdf.gpg new file mode 100644 index 000000000..754efcbf6 Binary files /dev/null and b/src/documents/tests/samples/documents/originals/0000004.pdf.gpg differ diff --git a/src/documents/tests/samples/documents/thumbnails/0000002.png b/src/documents/tests/samples/documents/thumbnails/0000002.png new file mode 100644 index 000000000..a3a768401 Binary files /dev/null and b/src/documents/tests/samples/documents/thumbnails/0000002.png differ diff --git a/src/documents/tests/samples/documents/thumbnails/0000003.png b/src/documents/tests/samples/documents/thumbnails/0000003.png new file mode 100644 index 000000000..a3a768401 Binary files /dev/null and b/src/documents/tests/samples/documents/thumbnails/0000003.png differ diff --git a/src/documents/tests/samples/documents/thumbnails/0000002.png.gpg b/src/documents/tests/samples/documents/thumbnails/0000004.png.gpg similarity index 100% rename from src/documents/tests/samples/documents/thumbnails/0000002.png.gpg rename to src/documents/tests/samples/documents/thumbnails/0000004.png.gpg diff --git a/src/documents/tests/samples/simple-noalpha.png b/src/documents/tests/samples/simple-noalpha.png new file mode 100644 index 000000000..9a76e76a0 Binary files /dev/null and b/src/documents/tests/samples/simple-noalpha.png differ diff --git a/src/documents/tests/samples/simple.jpg b/src/documents/tests/samples/simple.jpg new file mode 100644 index 000000000..a8c58af0d Binary files /dev/null and b/src/documents/tests/samples/simple.jpg differ diff --git a/src/documents/tests/samples/simple.png b/src/documents/tests/samples/simple.png new file mode 100644 index 000000000..a3a768401 Binary files /dev/null and b/src/documents/tests/samples/simple.png differ diff --git a/src/documents/tests/samples/simple.txt b/src/documents/tests/samples/simple.txt new file mode 100644 index 000000000..6de7b8c69 --- /dev/null +++ b/src/documents/tests/samples/simple.txt @@ -0,0 +1 @@ +This is a test file. diff --git a/src/documents/tests/samples/test_with_bom.pdf b/src/documents/tests/samples/test_with_bom.pdf new file mode 100644 index 000000000..c4a46701c Binary files /dev/null and b/src/documents/tests/samples/test_with_bom.pdf differ diff --git a/src/documents/tests/test_admin.py b/src/documents/tests/test_admin.py new file mode 100644 index 000000000..fc1d7ffaf --- /dev/null +++ b/src/documents/tests/test_admin.py @@ -0,0 +1,63 @@ +from unittest import mock + +from django.contrib.admin.sites import AdminSite +from django.test import TestCase +from django.utils import timezone + +from documents import index +from documents.admin import DocumentAdmin +from documents.models import Document +from documents.tests.utils import DirectoriesMixin + + +class TestDocumentAdmin(DirectoriesMixin, TestCase): + + def get_document_from_index(self, doc): + ix = index.open_index() + with ix.searcher() as searcher: + return searcher.document(id=doc.id) + + def setUp(self) -> None: + super(TestDocumentAdmin, self).setUp() + self.doc_admin = DocumentAdmin(model=Document, admin_site=AdminSite()) + + def test_save_model(self): + doc = Document.objects.create(title="test") + + doc.title = "new title" + self.doc_admin.save_model(None, doc, None, None) + self.assertEqual(Document.objects.get(id=doc.id).title, "new title") + self.assertEqual(self.get_document_from_index(doc)['id'], doc.id) + + def test_delete_model(self): + doc = Document.objects.create(title="test") + index.add_or_update_document(doc) + self.assertIsNotNone(self.get_document_from_index(doc)) + + self.doc_admin.delete_model(None, doc) + + self.assertRaises(Document.DoesNotExist, Document.objects.get, id=doc.id) + self.assertIsNone(self.get_document_from_index(doc)) + + def test_delete_queryset(self): + docs = [] + for i in range(42): + doc = Document.objects.create(title="Many documents with the same title", checksum=f"{i:02}") + docs.append(doc) + index.add_or_update_document(doc) + + self.assertEqual(Document.objects.count(), 42) + + for doc in docs: + self.assertIsNotNone(self.get_document_from_index(doc)) + + self.doc_admin.delete_queryset(None, Document.objects.all()) + + self.assertEqual(Document.objects.count(), 0) + + for doc in docs: + self.assertIsNone(self.get_document_from_index(doc)) + + def test_created(self): + doc = Document.objects.create(title="test", created=timezone.datetime(2020, 4, 12)) + self.assertEqual(self.doc_admin.created_(doc), "2020-04-12") diff --git a/src/documents/tests/test_api.py b/src/documents/tests/test_api.py index ab1716366..2f8dc18da 100644 --- a/src/documents/tests/test_api.py +++ b/src/documents/tests/test_api.py @@ -1,15 +1,21 @@ +import datetime +import io +import json import os import shutil import tempfile +import zipfile from unittest import mock +import pytest +from django.conf import settings from django.contrib.auth.models import User -from pathvalidate import ValidationError +from django.test import override_settings from rest_framework.test import APITestCase from whoosh.writing import AsyncWriter -from documents import index -from documents.models import Document, Correspondent, DocumentType, Tag +from documents import index, bulk_edit +from documents.models import Document, Correspondent, DocumentType, Tag, SavedView, MatchingModel from documents.tests.utils import DirectoriesMixin @@ -18,8 +24,8 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): def setUp(self): super(TestDocumentApi, self).setUp() - user = User.objects.create_superuser(username="temp_admin") - self.client.force_login(user=user) + self.user = User.objects.create_superuser(username="temp_admin") + self.client.force_login(user=self.user) def testDocuments(self): @@ -64,6 +70,56 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(len(Document.objects.all()), 0) + def test_document_fields(self): + c = Correspondent.objects.create(name="c", pk=41) + dt = DocumentType.objects.create(name="dt", pk=63) + tag = Tag.objects.create(name="t", pk=85) + doc = Document.objects.create(title="WOW", content="the content", correspondent=c, document_type=dt, checksum="123", mime_type="application/pdf") + + response = self.client.get("/api/documents/", format='json') + self.assertEqual(response.status_code, 200) + results_full = response.data['results'] + self.assertTrue("content" in results_full[0]) + self.assertTrue("id" in results_full[0]) + + response = self.client.get("/api/documents/?fields=id", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertFalse("content" in results[0]) + self.assertTrue("id" in results[0]) + self.assertEqual(len(results[0]), 1) + + response = self.client.get("/api/documents/?fields=content", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertTrue("content" in results[0]) + self.assertFalse("id" in results[0]) + self.assertEqual(len(results[0]), 1) + + response = self.client.get("/api/documents/?fields=id,content", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertTrue("content" in results[0]) + self.assertTrue("id" in results[0]) + self.assertEqual(len(results[0]), 2) + + response = self.client.get("/api/documents/?fields=id,conteasdnt", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertFalse("content" in results[0]) + self.assertTrue("id" in results[0]) + self.assertEqual(len(results[0]), 1) + + response = self.client.get("/api/documents/?fields=", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertEqual(results_full, results) + + response = self.client.get("/api/documents/?fields=dgfhs", format='json') + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertEqual(len(results[0]), 0) + def test_document_actions(self): _, filename = tempfile.mkstemp(dir=self.dirs.originals_dir) @@ -94,21 +150,19 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, 200) self.assertEqual(response.content, content_thumbnail) + @override_settings(PAPERLESS_FILENAME_FORMAT="") def test_download_with_archive(self): - _, filename = tempfile.mkstemp(dir=self.dirs.originals_dir) - content = b"This is a test" content_archive = b"This is the same test but archived" - with open(filename, "wb") as f: - f.write(content) - - filename = os.path.basename(filename) - - doc = Document.objects.create(title="none", filename=filename, + doc = Document.objects.create(title="none", filename="my_document.pdf", + archive_filename="archived.pdf", mime_type="application/pdf") + with open(doc.source_path, "wb") as f: + f.write(content) + with open(doc.archive_path, "wb") as f: f.write(content_archive) @@ -170,15 +224,19 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, 200) results = response.data['results'] self.assertEqual(len(results), 2) - self.assertEqual(results[0]['id'], doc2.id) - self.assertEqual(results[1]['id'], doc3.id) + self.assertCountEqual([results[0]['id'], results[1]['id']], [doc2.id, doc3.id]) response = self.client.get("/api/documents/?tags__id__in={},{}".format(tag_inbox.id, tag_3.id)) self.assertEqual(response.status_code, 200) results = response.data['results'] self.assertEqual(len(results), 2) - self.assertEqual(results[0]['id'], doc1.id) - self.assertEqual(results[1]['id'], doc3.id) + self.assertCountEqual([results[0]['id'], results[1]['id']], [doc1.id, doc3.id]) + + response = self.client.get("/api/documents/?tags__id__in={},{}".format(tag_2.id, tag_3.id)) + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertEqual(len(results), 2) + self.assertCountEqual([results[0]['id'], results[1]['id']], [doc2.id, doc3.id]) response = self.client.get("/api/documents/?tags__id__all={},{}".format(tag_2.id, tag_3.id)) self.assertEqual(response.status_code, 200) @@ -200,8 +258,7 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, 200) results = response.data['results'] self.assertEqual(len(results), 2) - self.assertEqual(results[0]['id'], doc1.id) - self.assertEqual(results[1]['id'], doc2.id) + self.assertCountEqual([results[0]['id'], results[1]['id']], [doc1.id, doc2.id]) response = self.client.get("/api/documents/?tags__id__none={},{}".format(tag_3.id, tag_2.id)) self.assertEqual(response.status_code, 200) @@ -214,10 +271,28 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): results = response.data['results'] self.assertEqual(len(results), 0) - def test_search_no_query(self): - response = self.client.get("/api/search/") - results = response.data['results'] + def test_documents_title_content_filter(self): + doc1 = Document.objects.create(title="title A", content="content A", checksum="A", mime_type="application/pdf") + doc2 = Document.objects.create(title="title B", content="content A", checksum="B", mime_type="application/pdf") + doc3 = Document.objects.create(title="title A", content="content B", checksum="C", mime_type="application/pdf") + doc4 = Document.objects.create(title="title B", content="content B", checksum="D", mime_type="application/pdf") + + response = self.client.get("/api/documents/?title_content=A") + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertEqual(len(results), 3) + self.assertCountEqual([results[0]['id'], results[1]['id'], results[2]['id']], [doc1.id, doc2.id, doc3.id]) + + response = self.client.get("/api/documents/?title_content=B") + self.assertEqual(response.status_code, 200) + results = response.data['results'] + self.assertEqual(len(results), 3) + self.assertCountEqual([results[0]['id'], results[1]['id'], results[2]['id']], [doc2.id, doc3.id, doc4.id]) + + response = self.client.get("/api/documents/?title_content=X") + self.assertEqual(response.status_code, 200) + results = response.data['results'] self.assertEqual(len(results), 0) def test_search(self): @@ -231,32 +306,24 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): index.update_document(writer, d1) index.update_document(writer, d2) index.update_document(writer, d3) - response = self.client.get("/api/search/?query=bank") + response = self.client.get("/api/documents/?query=bank") results = response.data['results'] self.assertEqual(response.data['count'], 3) - self.assertEqual(response.data['page'], 1) - self.assertEqual(response.data['page_count'], 1) self.assertEqual(len(results), 3) - response = self.client.get("/api/search/?query=september") + response = self.client.get("/api/documents/?query=september") results = response.data['results'] self.assertEqual(response.data['count'], 1) - self.assertEqual(response.data['page'], 1) - self.assertEqual(response.data['page_count'], 1) self.assertEqual(len(results), 1) - response = self.client.get("/api/search/?query=statement") + response = self.client.get("/api/documents/?query=statement") results = response.data['results'] self.assertEqual(response.data['count'], 2) - self.assertEqual(response.data['page'], 1) - self.assertEqual(response.data['page_count'], 1) self.assertEqual(len(results), 2) - response = self.client.get("/api/search/?query=sfegdfg") + response = self.client.get("/api/documents/?query=sfegdfg") results = response.data['results'] self.assertEqual(response.data['count'], 0) - self.assertEqual(response.data['page'], 0) - self.assertEqual(response.data['page_count'], 0) self.assertEqual(len(results), 0) def test_search_multi_page(self): @@ -269,53 +336,34 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): seen_ids = [] for i in range(1, 6): - response = self.client.get(f"/api/search/?query=content&page={i}") + response = self.client.get(f"/api/documents/?query=content&page={i}&page_size=10") results = response.data['results'] self.assertEqual(response.data['count'], 55) - self.assertEqual(response.data['page'], i) - self.assertEqual(response.data['page_count'], 6) self.assertEqual(len(results), 10) for result in results: self.assertNotIn(result['id'], seen_ids) seen_ids.append(result['id']) - response = self.client.get(f"/api/search/?query=content&page=6") + response = self.client.get(f"/api/documents/?query=content&page=6&page_size=10") results = response.data['results'] self.assertEqual(response.data['count'], 55) - self.assertEqual(response.data['page'], 6) - self.assertEqual(response.data['page_count'], 6) self.assertEqual(len(results), 5) for result in results: self.assertNotIn(result['id'], seen_ids) seen_ids.append(result['id']) - response = self.client.get(f"/api/search/?query=content&page=7") - results = response.data['results'] - self.assertEqual(response.data['count'], 55) - self.assertEqual(response.data['page'], 6) - self.assertEqual(response.data['page_count'], 6) - self.assertEqual(len(results), 5) - def test_search_invalid_page(self): with AsyncWriter(index.open_index()) as writer: for i in range(15): doc = Document.objects.create(checksum=str(i), pk=i+1, title=f"Document {i+1}", content="content") index.update_document(writer, doc) - first_page = self.client.get(f"/api/search/?query=content&page=1").data - second_page = self.client.get(f"/api/search/?query=content&page=2").data - should_be_first_page_1 = self.client.get(f"/api/search/?query=content&page=0").data - should_be_first_page_2 = self.client.get(f"/api/search/?query=content&page=dgfd").data - should_be_first_page_3 = self.client.get(f"/api/search/?query=content&page=").data - should_be_first_page_4 = self.client.get(f"/api/search/?query=content&page=-7868").data - - self.assertDictEqual(first_page, should_be_first_page_1) - self.assertDictEqual(first_page, should_be_first_page_2) - self.assertDictEqual(first_page, should_be_first_page_3) - self.assertDictEqual(first_page, should_be_first_page_4) - self.assertNotEqual(len(first_page['results']), len(second_page['results'])) + response = self.client.get(f"/api/documents/?query=content&page=0&page_size=10") + self.assertEqual(response.status_code, 404) + response = self.client.get(f"/api/documents/?query=content&page=3&page_size=10") + self.assertEqual(response.status_code, 404) @mock.patch("documents.index.autocomplete") def test_search_autocomplete(self, m): @@ -339,6 +387,7 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(response.status_code, 200) self.assertEqual(len(response.data), 10) + @pytest.mark.skip(reason="Not implemented yet") def test_search_spelling_correction(self): with AsyncWriter(index.open_index()) as writer: for i in range(55): @@ -355,6 +404,98 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(correction, None) + def test_search_more_like(self): + d1=Document.objects.create(title="invoice", content="the thing i bought at a shop and paid with bank account", checksum="A", pk=1) + d2=Document.objects.create(title="bank statement 1", content="things i paid for in august", pk=2, checksum="B") + d3=Document.objects.create(title="bank statement 3", content="things i paid for in september", pk=3, checksum="C") + with AsyncWriter(index.open_index()) as writer: + index.update_document(writer, d1) + index.update_document(writer, d2) + index.update_document(writer, d3) + + response = self.client.get(f"/api/documents/?more_like_id={d2.id}") + + self.assertEqual(response.status_code, 200) + + results = response.data['results'] + + self.assertEqual(len(results), 2) + self.assertEqual(results[0]['id'], d3.id) + self.assertEqual(results[1]['id'], d1.id) + + def test_search_filtering(self): + t = Tag.objects.create(name="tag") + t2 = Tag.objects.create(name="tag2") + c = Correspondent.objects.create(name="correspondent") + dt = DocumentType.objects.create(name="type") + + d1 = Document.objects.create(checksum="1", correspondent=c, content="test") + d2 = Document.objects.create(checksum="2", document_type=dt, content="test") + d3 = Document.objects.create(checksum="3", content="test") + d3.tags.add(t) + d3.tags.add(t2) + d4 = Document.objects.create(checksum="4", created=datetime.datetime(2020, 7, 13), content="test") + d4.tags.add(t2) + d5 = Document.objects.create(checksum="5", added=datetime.datetime(2020, 7, 13), content="test") + d6 = Document.objects.create(checksum="6", content="test2") + + with AsyncWriter(index.open_index()) as writer: + for doc in Document.objects.all(): + index.update_document(writer, doc) + + def search_query(q): + r = self.client.get("/api/documents/?query=test" + q) + self.assertEqual(r.status_code, 200) + return [hit['id'] for hit in r.data['results']] + + self.assertCountEqual(search_query(""), [d1.id, d2.id, d3.id, d4.id, d5.id]) + self.assertCountEqual(search_query("&is_tagged=true"), [d3.id, d4.id]) + self.assertCountEqual(search_query("&is_tagged=false"), [d1.id, d2.id, d5.id]) + self.assertCountEqual(search_query("&correspondent__id=" + str(c.id)), [d1.id]) + self.assertCountEqual(search_query("&document_type__id=" + str(dt.id)), [d2.id]) + self.assertCountEqual(search_query("&correspondent__isnull"), [d2.id, d3.id, d4.id, d5.id]) + self.assertCountEqual(search_query("&document_type__isnull"), [d1.id, d3.id, d4.id, d5.id]) + self.assertCountEqual(search_query("&tags__id__all=" + str(t.id) + "," + str(t2.id)), [d3.id]) + self.assertCountEqual(search_query("&tags__id__all=" + str(t.id)), [d3.id]) + self.assertCountEqual(search_query("&tags__id__all=" + str(t2.id)), [d3.id, d4.id]) + + self.assertIn(d4.id, search_query("&created__date__lt=" + datetime.datetime(2020, 9, 2).strftime("%Y-%m-%d"))) + self.assertNotIn(d4.id, search_query("&created__date__gt=" + datetime.datetime(2020, 9, 2).strftime("%Y-%m-%d"))) + + self.assertNotIn(d4.id, search_query("&created__date__lt=" + datetime.datetime(2020, 1, 2).strftime("%Y-%m-%d"))) + self.assertIn(d4.id, search_query("&created__date__gt=" + datetime.datetime(2020, 1, 2).strftime("%Y-%m-%d"))) + + self.assertIn(d5.id, search_query("&added__date__lt=" + datetime.datetime(2020, 9, 2).strftime("%Y-%m-%d"))) + self.assertNotIn(d5.id, search_query("&added__date__gt=" + datetime.datetime(2020, 9, 2).strftime("%Y-%m-%d"))) + + self.assertNotIn(d5.id, search_query("&added__date__lt=" + datetime.datetime(2020, 1, 2).strftime("%Y-%m-%d"))) + self.assertIn(d5.id, search_query("&added__date__gt=" + datetime.datetime(2020, 1, 2).strftime("%Y-%m-%d"))) + + def test_search_sorting(self): + c1 = Correspondent.objects.create(name="corres Ax") + c2 = Correspondent.objects.create(name="corres Cx") + c3 = Correspondent.objects.create(name="corres Bx") + d1 = Document.objects.create(checksum="1", correspondent=c1, content="test", archive_serial_number=2, title="3") + d2 = Document.objects.create(checksum="2", correspondent=c2, content="test", archive_serial_number=3, title="2") + d3 = Document.objects.create(checksum="3", correspondent=c3, content="test", archive_serial_number=1, title="1") + + with AsyncWriter(index.open_index()) as writer: + for doc in Document.objects.all(): + index.update_document(writer, doc) + + def search_query(q): + r = self.client.get("/api/documents/?query=test" + q) + self.assertEqual(r.status_code, 200) + return [hit['id'] for hit in r.data['results']] + + self.assertListEqual(search_query("&ordering=archive_serial_number"), [d3.id, d1.id, d2.id]) + self.assertListEqual(search_query("&ordering=-archive_serial_number"), [d2.id, d1.id, d3.id]) + self.assertListEqual(search_query("&ordering=title"), [d3.id, d2.id, d1.id]) + self.assertListEqual(search_query("&ordering=-title"), [d1.id, d2.id, d3.id]) + self.assertListEqual(search_query("&ordering=correspondent__name"), [d1.id, d3.id, d2.id]) + self.assertListEqual(search_query("&ordering=-correspondent__name"), [d2.id, d3.id, d1.id]) + + def test_statistics(self): doc1 = Document.objects.create(title="none1", checksum="A") @@ -370,6 +511,13 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertEqual(response.data['documents_total'], 3) self.assertEqual(response.data['documents_inbox'], 1) + def test_statistics_no_inbox_tag(self): + Document.objects.create(title="none1", checksum="A") + + response = self.client.get("/api/statistics/") + self.assertEqual(response.status_code, 200) + self.assertEqual(response.data['documents_inbox'], None) + @mock.patch("documents.views.async_task") def test_upload(self, m): @@ -387,6 +535,23 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertIsNone(kwargs['override_document_type_id']) self.assertIsNone(kwargs['override_tag_ids']) + @mock.patch("documents.views.async_task") + def test_upload_empty_metadata(self, m): + + with open(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), "rb") as f: + response = self.client.post("/api/documents/post_document/", {"document": f, "title": "", "correspondent": "", "document_type": ""}) + + self.assertEqual(response.status_code, 200) + + m.assert_called_once() + + args, kwargs = m.call_args + self.assertEqual(kwargs['override_filename'], "simple.pdf") + self.assertIsNone(kwargs['override_title']) + self.assertIsNone(kwargs['override_correspondent_id']) + self.assertIsNone(kwargs['override_document_type_id']) + self.assertIsNone(kwargs['override_tag_ids']) + @mock.patch("documents.views.async_task") def test_upload_invalid_form(self, m): @@ -486,10 +651,13 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): async_task.assert_not_called() def test_get_metadata(self): - doc = Document.objects.create(title="test", filename="file.pdf", mime_type="image/png", archive_checksum="A") + doc = Document.objects.create(title="test", filename="file.pdf", mime_type="image/png", archive_checksum="A", archive_filename="archive.pdf") - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png"), doc.source_path) - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), doc.archive_path) + source_file = os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png") + archive_file = os.path.join(os.path.dirname(__file__), "samples", "simple.pdf") + + shutil.copy(source_file, doc.source_path) + shutil.copy(archive_file, doc.archive_path) response = self.client.get(f"/api/documents/{doc.pk}/metadata/") self.assertEqual(response.status_code, 200) @@ -500,6 +668,14 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertTrue(meta['has_archive_version']) self.assertEqual(len(meta['original_metadata']), 0) self.assertGreater(len(meta['archive_metadata']), 0) + self.assertEqual(meta['media_filename'], "file.pdf") + self.assertEqual(meta['archive_media_filename'], "archive.pdf") + self.assertEqual(meta['original_size'], os.stat(source_file).st_size) + self.assertEqual(meta['archive_size'], os.stat(archive_file).st_size) + + def test_get_metadata_invalid_doc(self): + response = self.client.get(f"/api/documents/34576/metadata/") + self.assertEqual(response.status_code, 404) def test_get_metadata_no_archive(self): doc = Document.objects.create(title="test", filename="file.pdf", mime_type="application/pdf") @@ -515,3 +691,746 @@ class TestDocumentApi(DirectoriesMixin, APITestCase): self.assertFalse(meta['has_archive_version']) self.assertGreater(len(meta['original_metadata']), 0) self.assertIsNone(meta['archive_metadata']) + self.assertIsNone(meta['archive_media_filename']) + + def test_get_metadata_missing_files(self): + doc = Document.objects.create(title="test", filename="file.pdf", mime_type="application/pdf", archive_filename="file.pdf", archive_checksum="B", checksum="A") + + response = self.client.get(f"/api/documents/{doc.pk}/metadata/") + self.assertEqual(response.status_code, 200) + + meta = response.data + + self.assertTrue(meta['has_archive_version']) + self.assertIsNone(meta['original_metadata']) + self.assertIsNone(meta['original_size']) + self.assertIsNone(meta['archive_metadata']) + self.assertIsNone(meta['archive_size']) + + + def test_get_empty_suggestions(self): + doc = Document.objects.create(title="test", mime_type="application/pdf") + + response = self.client.get(f"/api/documents/{doc.pk}/suggestions/") + + self.assertEqual(response.status_code, 200) + self.assertEqual(response.data, {'correspondents': [], 'tags': [], 'document_types': []}) + + def test_get_suggestions_invalid_doc(self): + response = self.client.get(f"/api/documents/34676/suggestions/") + self.assertEqual(response.status_code, 404) + + @mock.patch("documents.views.match_correspondents") + @mock.patch("documents.views.match_tags") + @mock.patch("documents.views.match_document_types") + def test_get_suggestions(self, match_document_types, match_tags, match_correspondents): + doc = Document.objects.create(title="test", mime_type="application/pdf", content="this is an invoice!") + match_tags.return_value = [Tag(id=56), Tag(id=123)] + match_document_types.return_value = [DocumentType(id=23)] + match_correspondents.return_value = [Correspondent(id=88), Correspondent(id=2)] + + response = self.client.get(f"/api/documents/{doc.pk}/suggestions/") + self.assertEqual(response.data, {'correspondents': [88,2], 'tags': [56,123], 'document_types': [23]}) + + def test_saved_views(self): + u1 = User.objects.create_user("user1") + u2 = User.objects.create_user("user2") + + v1 = SavedView.objects.create(user=u1, name="test1", sort_field="", show_on_dashboard=False, show_in_sidebar=False) + v2 = SavedView.objects.create(user=u2, name="test2", sort_field="", show_on_dashboard=False, show_in_sidebar=False) + v3 = SavedView.objects.create(user=u2, name="test3", sort_field="", show_on_dashboard=False, show_in_sidebar=False) + + response = self.client.get("/api/saved_views/") + self.assertEqual(response.status_code, 200) + self.assertEqual(response.data['count'], 0) + + self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 404) + + self.client.force_login(user=u1) + + response = self.client.get("/api/saved_views/") + self.assertEqual(response.status_code, 200) + self.assertEqual(response.data['count'], 1) + + self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 200) + + self.client.force_login(user=u2) + + response = self.client.get("/api/saved_views/") + self.assertEqual(response.status_code, 200) + self.assertEqual(response.data['count'], 2) + + self.assertEqual(self.client.get(f"/api/saved_views/{v1.id}/").status_code, 404) + + def test_create_update_patch(self): + + u1 = User.objects.create_user("user1") + + view = { + "name": "test", + "show_on_dashboard": True, + "show_in_sidebar": True, + "sort_field": "created2", + "filter_rules": [ + { + "rule_type": 4, + "value": "test" + } + ] + } + + response = self.client.post("/api/saved_views/", view, format='json') + self.assertEqual(response.status_code, 201) + + v1 = SavedView.objects.get(name="test") + self.assertEqual(v1.sort_field, "created2") + self.assertEqual(v1.filter_rules.count(), 1) + self.assertEqual(v1.user, self.user) + + response = self.client.patch(f"/api/saved_views/{v1.id}/", { + "show_in_sidebar": False + }, format='json') + + v1 = SavedView.objects.get(id=v1.id) + self.assertEqual(response.status_code, 200) + self.assertFalse(v1.show_in_sidebar) + self.assertEqual(v1.filter_rules.count(), 1) + + view['filter_rules'] = [{ + "rule_type": 12, + "value": "secret" + }] + + response = self.client.put(f"/api/saved_views/{v1.id}/", view, format='json') + self.assertEqual(response.status_code, 200) + + v1 = SavedView.objects.get(id=v1.id) + self.assertEqual(v1.filter_rules.count(), 1) + self.assertEqual(v1.filter_rules.first().value, "secret") + + view['filter_rules'] = [] + + response = self.client.put(f"/api/saved_views/{v1.id}/", view, format='json') + self.assertEqual(response.status_code, 200) + + v1 = SavedView.objects.get(id=v1.id) + self.assertEqual(v1.filter_rules.count(), 0) + + def test_get_logs(self): + response = self.client.get("/api/logs/") + self.assertEqual(response.status_code, 200) + self.assertCountEqual(response.data, ["mail", "paperless"]) + + def test_get_invalid_log(self): + response = self.client.get("/api/logs/bogus_log/") + self.assertEqual(response.status_code, 404) + + @override_settings(LOGGING_DIR="bogus_dir") + def test_get_nonexistent_log(self): + response = self.client.get("/api/logs/paperless/") + self.assertEqual(response.status_code, 404) + + def test_get_log(self): + log_data = "test\ntest2\n" + with open(os.path.join(settings.LOGGING_DIR, "paperless.log"), "w") as f: + f.write(log_data) + response = self.client.get("/api/logs/paperless/") + self.assertEqual(response.status_code, 200) + self.assertListEqual(response.data, ["test", "test2"]) + + def test_invalid_regex_other_algorithm(self): + for endpoint in ['correspondents', 'tags', 'document_types']: + response = self.client.post(f"/api/{endpoint}/", { + "name": "test", + "matching_algorithm": MatchingModel.MATCH_ANY, + "match": "[" + }, format='json') + self.assertEqual(response.status_code, 201, endpoint) + + def test_invalid_regex(self): + for endpoint in ['correspondents', 'tags', 'document_types']: + response = self.client.post(f"/api/{endpoint}/", { + "name": "test", + "matching_algorithm": MatchingModel.MATCH_REGEX, + "match": "[" + }, format='json') + self.assertEqual(response.status_code, 400, endpoint) + + def test_valid_regex(self): + for endpoint in ['correspondents', 'tags', 'document_types']: + response = self.client.post(f"/api/{endpoint}/", { + "name": "test", + "matching_algorithm": MatchingModel.MATCH_REGEX, + "match": "[0-9]" + }, format='json') + self.assertEqual(response.status_code, 201, endpoint) + + def test_regex_no_algorithm(self): + for endpoint in ['correspondents', 'tags', 'document_types']: + response = self.client.post(f"/api/{endpoint}/", { + "name": "test", + "match": "[0-9]" + }, format='json') + self.assertEqual(response.status_code, 201, endpoint) + + def test_tag_color_default(self): + response = self.client.post("/api/tags/", { + "name": "tag" + }, format="json") + self.assertEqual(response.status_code, 201) + self.assertEqual(Tag.objects.get(id=response.data['id']).color, "#a6cee3") + self.assertEqual(self.client.get(f"/api/tags/{response.data['id']}/", format="json").data['colour'], 1) + + def test_tag_color(self): + response = self.client.post("/api/tags/", { + "name": "tag", + "colour": 3 + }, format="json") + self.assertEqual(response.status_code, 201) + self.assertEqual(Tag.objects.get(id=response.data['id']).color, "#b2df8a") + self.assertEqual(self.client.get(f"/api/tags/{response.data['id']}/", format="json").data['colour'], 3) + + def test_tag_color_invalid(self): + response = self.client.post("/api/tags/", { + "name": "tag", + "colour": 34 + }, format="json") + self.assertEqual(response.status_code, 400) + + def test_tag_color_custom(self): + tag = Tag.objects.create(name="test", color="#abcdef") + self.assertEqual(self.client.get(f"/api/tags/{tag.id}/", format="json").data['colour'], 1) + + +class TestDocumentApiV2(DirectoriesMixin, APITestCase): + + def setUp(self): + super(TestDocumentApiV2, self).setUp() + + self.user = User.objects.create_superuser(username="temp_admin") + + self.client.force_login(user=self.user) + self.client.defaults['HTTP_ACCEPT'] = 'application/json; version=2' + + def test_tag_validate_color(self): + self.assertEqual(self.client.post("/api/tags/", {"name": "test", "color": "#12fFaA"}, format="json").status_code, 201) + + self.assertEqual(self.client.post("/api/tags/", {"name": "test1", "color": "abcdef"}, format="json").status_code, 400) + self.assertEqual(self.client.post("/api/tags/", {"name": "test2", "color": "#abcdfg"}, format="json").status_code, 400) + self.assertEqual(self.client.post("/api/tags/", {"name": "test3", "color": "#asd"}, format="json").status_code, 400) + self.assertEqual(self.client.post("/api/tags/", {"name": "test4", "color": "#12121212"}, format="json").status_code, 400) + + def test_tag_text_color(self): + t = Tag.objects.create(name="tag1", color="#000000") + self.assertEqual(self.client.get(f"/api/tags/{t.id}/", format="json").data['text_color'], "#ffffff") + + t.color = "#ffffff" + t.save() + self.assertEqual(self.client.get(f"/api/tags/{t.id}/", format="json").data['text_color'], "#000000") + + t.color = "asdf" + t.save() + self.assertEqual(self.client.get(f"/api/tags/{t.id}/", format="json").data['text_color'], "#000000") + + t.color = "123" + t.save() + self.assertEqual(self.client.get(f"/api/tags/{t.id}/", format="json").data['text_color'], "#000000") + + +class TestBulkEdit(DirectoriesMixin, APITestCase): + + def setUp(self): + super(TestBulkEdit, self).setUp() + + user = User.objects.create_superuser(username="temp_admin") + self.client.force_login(user=user) + + patcher = mock.patch('documents.bulk_edit.async_task') + self.async_task = patcher.start() + self.addCleanup(patcher.stop) + self.c1 = Correspondent.objects.create(name="c1") + self.c2 = Correspondent.objects.create(name="c2") + self.dt1 = DocumentType.objects.create(name="dt1") + self.dt2 = DocumentType.objects.create(name="dt2") + self.t1 = Tag.objects.create(name="t1") + self.t2 = Tag.objects.create(name="t2") + self.doc1 = Document.objects.create(checksum="A", title="A") + self.doc2 = Document.objects.create(checksum="B", title="B", correspondent=self.c1, document_type=self.dt1) + self.doc3 = Document.objects.create(checksum="C", title="C", correspondent=self.c2, document_type=self.dt2) + self.doc4 = Document.objects.create(checksum="D", title="D") + self.doc5 = Document.objects.create(checksum="E", title="E") + self.doc2.tags.add(self.t1) + self.doc3.tags.add(self.t2) + self.doc4.tags.add(self.t1, self.t2) + + def test_set_correspondent(self): + self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 1) + bulk_edit.set_correspondent([self.doc1.id, self.doc2.id, self.doc3.id], self.c2.id) + self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 3) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc1.id, self.doc2.id]) + + def test_unset_correspondent(self): + self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 1) + bulk_edit.set_correspondent([self.doc1.id, self.doc2.id, self.doc3.id], None) + self.assertEqual(Document.objects.filter(correspondent=self.c2).count(), 0) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc2.id, self.doc3.id]) + + def test_set_document_type(self): + self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 1) + bulk_edit.set_document_type([self.doc1.id, self.doc2.id, self.doc3.id], self.dt2.id) + self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 3) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc1.id, self.doc2.id]) + + def test_unset_document_type(self): + self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 1) + bulk_edit.set_document_type([self.doc1.id, self.doc2.id, self.doc3.id], None) + self.assertEqual(Document.objects.filter(document_type=self.dt2).count(), 0) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc2.id, self.doc3.id]) + + def test_add_tag(self): + self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 2) + bulk_edit.add_tag([self.doc1.id, self.doc2.id, self.doc3.id, self.doc4.id], self.t1.id) + self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 4) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc1.id, self.doc3.id]) + + def test_remove_tag(self): + self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 2) + bulk_edit.remove_tag([self.doc1.id, self.doc3.id, self.doc4.id], self.t1.id) + self.assertEqual(Document.objects.filter(tags__id=self.t1.id).count(), 1) + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + self.assertCountEqual(kwargs['document_ids'], [self.doc4.id]) + + def test_modify_tags(self): + tag_unrelated = Tag.objects.create(name="unrelated") + self.doc2.tags.add(tag_unrelated) + self.doc3.tags.add(tag_unrelated) + bulk_edit.modify_tags([self.doc2.id, self.doc3.id], add_tags=[self.t2.id], remove_tags=[self.t1.id]) + + self.assertCountEqual(list(self.doc2.tags.all()), [self.t2, tag_unrelated]) + self.assertCountEqual(list(self.doc3.tags.all()), [self.t2, tag_unrelated]) + + self.async_task.assert_called_once() + args, kwargs = self.async_task.call_args + # TODO: doc3 should not be affected, but the query for that is rather complicated + self.assertCountEqual(kwargs['document_ids'], [self.doc2.id, self.doc3.id]) + + def test_delete(self): + self.assertEqual(Document.objects.count(), 5) + bulk_edit.delete([self.doc1.id, self.doc2.id]) + self.assertEqual(Document.objects.count(), 3) + self.assertCountEqual([doc.id for doc in Document.objects.all()], [self.doc3.id, self.doc4.id, self.doc5.id]) + + @mock.patch("documents.serialisers.bulk_edit.set_correspondent") + def test_api_set_correspondent(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "set_correspondent", + "parameters": {"correspondent": self.c1.id} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertEqual(kwargs['correspondent'], self.c1.id) + + @mock.patch("documents.serialisers.bulk_edit.set_correspondent") + def test_api_unset_correspondent(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "set_correspondent", + "parameters": {"correspondent": None} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertIsNone(kwargs['correspondent']) + + @mock.patch("documents.serialisers.bulk_edit.set_document_type") + def test_api_set_type(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "set_document_type", + "parameters": {"document_type": self.dt1.id} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertEqual(kwargs['document_type'], self.dt1.id) + + @mock.patch("documents.serialisers.bulk_edit.set_document_type") + def test_api_unset_type(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "set_document_type", + "parameters": {"document_type": None} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertIsNone(kwargs['document_type']) + + @mock.patch("documents.serialisers.bulk_edit.add_tag") + def test_api_add_tag(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "add_tag", + "parameters": {"tag": self.t1.id} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertEqual(kwargs['tag'], self.t1.id) + + @mock.patch("documents.serialisers.bulk_edit.remove_tag") + def test_api_remove_tag(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "remove_tag", + "parameters": {"tag": self.t1.id} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertEqual(kwargs['tag'], self.t1.id) + + @mock.patch("documents.serialisers.bulk_edit.modify_tags") + def test_api_modify_tags(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id, self.doc3.id], + "method": "modify_tags", + "parameters": {"add_tags": [self.t1.id], "remove_tags": [self.t2.id]} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertListEqual(args[0], [self.doc1.id, self.doc3.id]) + self.assertEqual(kwargs['add_tags'], [self.t1.id]) + self.assertEqual(kwargs['remove_tags'], [self.t2.id]) + + @mock.patch("documents.serialisers.bulk_edit.delete") + def test_api_delete(self, m): + m.return_value = "OK" + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc1.id], + "method": "delete", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + m.assert_called_once() + args, kwargs = m.call_args + self.assertEqual(args[0], [self.doc1.id]) + self.assertEqual(len(kwargs), 0) + + def test_api_invalid_doc(self): + self.assertEqual(Document.objects.count(), 5) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [-235], + "method": "delete", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + self.assertEqual(Document.objects.count(), 5) + + def test_api_invalid_method(self): + self.assertEqual(Document.objects.count(), 5) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "exterminate", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + self.assertEqual(Document.objects.count(), 5) + + def test_api_invalid_correspondent(self): + self.assertEqual(self.doc2.correspondent, self.c1) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "set_correspondent", + "parameters": {'correspondent': 345657} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + doc2 = Document.objects.get(id=self.doc2.id) + self.assertEqual(doc2.correspondent, self.c1) + + def test_api_no_correspondent(self): + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "set_correspondent", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_invalid_document_type(self): + self.assertEqual(self.doc2.document_type, self.dt1) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "set_document_type", + "parameters": {'document_type': 345657} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + doc2 = Document.objects.get(id=self.doc2.id) + self.assertEqual(doc2.document_type, self.dt1) + + def test_api_no_document_type(self): + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "set_document_type", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_add_invalid_tag(self): + self.assertEqual(list(self.doc2.tags.all()), [self.t1]) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "add_tag", + "parameters": {'tag': 345657} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + self.assertEqual(list(self.doc2.tags.all()), [self.t1]) + + def test_api_add_tag_no_tag(self): + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "add_tag", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_delete_invalid_tag(self): + self.assertEqual(list(self.doc2.tags.all()), [self.t1]) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "remove_tag", + "parameters": {'tag': 345657} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + self.assertEqual(list(self.doc2.tags.all()), [self.t1]) + + def test_api_delete_tag_no_tag(self): + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "remove_tag", + "parameters": {} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_modify_invalid_tags(self): + self.assertEqual(list(self.doc2.tags.all()), [self.t1]) + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "modify_tags", + "parameters": {'add_tags': [self.t2.id, 1657], "remove_tags": [1123123]} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_modify_tags_no_tags(self): + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "modify_tags", + "parameters": {"remove_tags": [1123123]} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + response = self.client.post("/api/documents/bulk_edit/", json.dumps({ + "documents": [self.doc2.id], + "method": "modify_tags", + "parameters": {'add_tags': [self.t2.id, 1657]} + }), content_type='application/json') + self.assertEqual(response.status_code, 400) + + def test_api_selection_data_empty(self): + response = self.client.post("/api/documents/selection_data/", json.dumps({ + "documents": [] + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + for field, Entity in [('selected_correspondents', Correspondent), ('selected_tags', Tag), ('selected_document_types', DocumentType)]: + self.assertEqual(len(response.data[field]), Entity.objects.count()) + for correspondent in response.data[field]: + self.assertEqual(correspondent['document_count'], 0) + self.assertCountEqual( + map(lambda c: c['id'], response.data[field]), + map(lambda c: c['id'], Entity.objects.values('id'))) + + def test_api_selection_data(self): + response = self.client.post("/api/documents/selection_data/", json.dumps({ + "documents": [self.doc1.id, self.doc2.id, self.doc4.id, self.doc5.id] + }), content_type='application/json') + self.assertEqual(response.status_code, 200) + + self.assertCountEqual(response.data['selected_correspondents'], [{"id": self.c1.id, "document_count": 1}, {"id": self.c2.id, "document_count": 0}]) + self.assertCountEqual(response.data['selected_tags'], [{"id": self.t1.id, "document_count": 2}, {"id": self.t2.id, "document_count": 1}]) + self.assertCountEqual(response.data['selected_document_types'], [{"id": self.c1.id, "document_count": 1}, {"id": self.c2.id, "document_count": 0}]) + + +class TestBulkDownload(DirectoriesMixin, APITestCase): + + def setUp(self): + super(TestBulkDownload, self).setUp() + + user = User.objects.create_superuser(username="temp_admin") + self.client.force_login(user=user) + + self.doc1 = Document.objects.create(title="unrelated", checksum="A") + self.doc2 = Document.objects.create(title="document A", filename="docA.pdf", mime_type="application/pdf", checksum="B", created=datetime.datetime(2021, 1, 1)) + self.doc2b = Document.objects.create(title="document A", filename="docA2.pdf", mime_type="application/pdf", checksum="D", created=datetime.datetime(2021, 1, 1)) + self.doc3 = Document.objects.create(title="document B", filename="docB.jpg", mime_type="image/jpeg", checksum="C", created=datetime.datetime(2020, 3, 21), archive_filename="docB.pdf", archive_checksum="D") + + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), self.doc2.source_path) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.png"), self.doc2b.source_path) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.jpg"), self.doc3.source_path) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "test_with_bom.pdf"), self.doc3.archive_path) + + def test_download_originals(self): + response = self.client.post("/api/documents/bulk_download/", json.dumps({ + "documents": [self.doc2.id, self.doc3.id], + "content": "originals" + }), content_type='application/json') + + self.assertEqual(response.status_code, 200) + self.assertEqual(response['Content-Type'], 'application/zip') + + with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: + self.assertEqual(len(zipf.filelist), 2) + self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) + self.assertIn("2020-03-21 document B.jpg", zipf.namelist()) + + with self.doc2.source_file as f: + self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) + + with self.doc3.source_file as f: + self.assertEqual(f.read(), zipf.read("2020-03-21 document B.jpg")) + + def test_download_default(self): + response = self.client.post("/api/documents/bulk_download/", json.dumps({ + "documents": [self.doc2.id, self.doc3.id] + }), content_type='application/json') + + self.assertEqual(response.status_code, 200) + self.assertEqual(response['Content-Type'], 'application/zip') + + with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: + self.assertEqual(len(zipf.filelist), 2) + self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) + self.assertIn("2020-03-21 document B.pdf", zipf.namelist()) + + with self.doc2.source_file as f: + self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) + + with self.doc3.archive_file as f: + self.assertEqual(f.read(), zipf.read("2020-03-21 document B.pdf")) + + def test_download_both(self): + response = self.client.post("/api/documents/bulk_download/", json.dumps({ + "documents": [self.doc2.id, self.doc3.id], + "content": "both" + }), content_type='application/json') + + self.assertEqual(response.status_code, 200) + self.assertEqual(response['Content-Type'], 'application/zip') + + with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: + self.assertEqual(len(zipf.filelist), 3) + self.assertIn("originals/2021-01-01 document A.pdf", zipf.namelist()) + self.assertIn("archive/2020-03-21 document B.pdf", zipf.namelist()) + self.assertIn("originals/2020-03-21 document B.jpg", zipf.namelist()) + + with self.doc2.source_file as f: + self.assertEqual(f.read(), zipf.read("originals/2021-01-01 document A.pdf")) + + with self.doc3.archive_file as f: + self.assertEqual(f.read(), zipf.read("archive/2020-03-21 document B.pdf")) + + with self.doc3.source_file as f: + self.assertEqual(f.read(), zipf.read("originals/2020-03-21 document B.jpg")) + + def test_filename_clashes(self): + response = self.client.post("/api/documents/bulk_download/", json.dumps({ + "documents": [self.doc2.id, self.doc2b.id] + }), content_type='application/json') + + self.assertEqual(response.status_code, 200) + self.assertEqual(response['Content-Type'], 'application/zip') + + with zipfile.ZipFile(io.BytesIO(response.content)) as zipf: + self.assertEqual(len(zipf.filelist), 2) + + self.assertIn("2021-01-01 document A.pdf", zipf.namelist()) + self.assertIn("2021-01-01 document A_01.pdf", zipf.namelist()) + + with self.doc2.source_file as f: + self.assertEqual(f.read(), zipf.read("2021-01-01 document A.pdf")) + + with self.doc2b.source_file as f: + self.assertEqual(f.read(), zipf.read("2021-01-01 document A_01.pdf")) + + def test_compression(self): + response = self.client.post("/api/documents/bulk_download/", json.dumps({ + "documents": [self.doc2.id, self.doc2b.id], + "compression": "lzma" + }), content_type='application/json') + +class TestApiAuth(APITestCase): + + def test_auth_required(self): + + d = Document.objects.create(title="Test") + + self.assertEqual(self.client.get("/api/documents/").status_code, 401) + + self.assertEqual(self.client.get(f"/api/documents/{d.id}/").status_code, 401) + self.assertEqual(self.client.get(f"/api/documents/{d.id}/download/").status_code, 401) + self.assertEqual(self.client.get(f"/api/documents/{d.id}/preview/").status_code, 401) + self.assertEqual(self.client.get(f"/api/documents/{d.id}/thumb/").status_code, 401) + + self.assertEqual(self.client.get("/api/tags/").status_code, 401) + self.assertEqual(self.client.get("/api/correspondents/").status_code, 401) + self.assertEqual(self.client.get("/api/document_types/").status_code, 401) + + self.assertEqual(self.client.get("/api/logs/").status_code, 401) + self.assertEqual(self.client.get("/api/saved_views/").status_code, 401) + + self.assertEqual(self.client.get("/api/search/autocomplete/").status_code, 401) + self.assertEqual(self.client.get("/api/documents/bulk_edit/").status_code, 401) + self.assertEqual(self.client.get("/api/documents/bulk_download/").status_code, 401) + self.assertEqual(self.client.get("/api/documents/selection_data/").status_code, 401) + + def test_api_version_no_auth(self): + + response = self.client.get("/api/") + self.assertNotIn("X-Api-Version", response) + self.assertNotIn("X-Version", response) + + def test_api_version_with_auth(self): + user = User.objects.create_superuser(username="test") + self.client.force_login(user) + response = self.client.get("/api/") + self.assertIn("X-Api-Version", response) + self.assertIn("X-Version", response) diff --git a/src/documents/tests/test_checks.py b/src/documents/tests/test_checks.py index 1027c11a0..ee4fbe8d1 100644 --- a/src/documents/tests/test_checks.py +++ b/src/documents/tests/test_checks.py @@ -1,9 +1,12 @@ import unittest +from unittest import mock +from django.core.checks import Error from django.test import TestCase from .factories import DocumentFactory -from ..checks import changed_password_check +from .. import document_consumer_declaration +from ..checks import changed_password_check, parser_check from ..models import Document @@ -15,3 +18,13 @@ class ChecksTestCase(TestCase): def test_changed_password_check_no_encryption(self): DocumentFactory.create(storage_type=Document.STORAGE_TYPE_UNENCRYPTED) self.assertEqual(changed_password_check(None), []) + + def test_parser_check(self): + + self.assertEqual(parser_check(None), []) + + with mock.patch('documents.checks.document_consumer_declaration.send') as m: + m.return_value = [] + + self.assertEqual(parser_check(None), [Error("No parsers found. This is a bug. The consumer won't be " + "able to consume any documents without parsers.")]) diff --git a/src/documents/tests/test_classifier.py b/src/documents/tests/test_classifier.py index 9e999794d..fcc08f842 100644 --- a/src/documents/tests/test_classifier.py +++ b/src/documents/tests/test_classifier.py @@ -1,10 +1,13 @@ +import os import tempfile -from time import sleep +from pathlib import Path from unittest import mock +import pytest +from django.conf import settings from django.test import TestCase, override_settings -from documents.classifier import DocumentClassifier, IncompatibleClassifierVersionError +from documents.classifier import DocumentClassifier, IncompatibleClassifierVersionError, load_classifier from documents.models import Correspondent, Document, Tag, DocumentType from documents.tests.utils import DirectoriesMixin @@ -82,37 +85,19 @@ class TestClassifier(DirectoriesMixin, TestCase): self.assertTrue(self.classifier.train()) self.assertFalse(self.classifier.train()) - self.classifier.save_classifier() + self.classifier.save() classifier2 = DocumentClassifier() current_ver = DocumentClassifier.FORMAT_VERSION with mock.patch("documents.classifier.DocumentClassifier.FORMAT_VERSION", current_ver+1): # assure that we won't load old classifiers. - self.assertRaises(IncompatibleClassifierVersionError, classifier2.reload) + self.assertRaises(IncompatibleClassifierVersionError, classifier2.load) - self.classifier.save_classifier() + self.classifier.save() # assure that we can load the classifier after saving it. - classifier2.reload() - - def testReload(self): - - self.generate_test_data() - self.assertTrue(self.classifier.train()) - self.classifier.save_classifier() - - classifier2 = DocumentClassifier() - classifier2.reload() - v1 = classifier2.classifier_version - - # change the classifier after some time. - sleep(1) - self.classifier.save_classifier() - - classifier2.reload() - v2 = classifier2.classifier_version - self.assertNotEqual(v1, v2) + classifier2.load() @override_settings(DATA_DIR=tempfile.mkdtemp()) def testSaveClassifier(self): @@ -121,12 +106,21 @@ class TestClassifier(DirectoriesMixin, TestCase): self.classifier.train() - self.classifier.save_classifier() + self.classifier.save() new_classifier = DocumentClassifier() - new_classifier.reload() + new_classifier.load() self.assertFalse(new_classifier.train()) + @override_settings(MODEL_FILE=os.path.join(os.path.dirname(__file__), "data", "model.pickle")) + def test_load_and_classify(self): + self.generate_test_data() + + new_classifier = DocumentClassifier() + new_classifier.load() + + self.assertCountEqual(new_classifier.predict_tags(self.doc2.content), [45, 12]) + def test_one_correspondent_predict(self): c1 = Correspondent.objects.create(name="c1", matching_algorithm=Correspondent.MATCH_AUTO) doc1 = Document.objects.create(title="doc1", content="this is a document from c1", correspondent=c1, checksum="A") @@ -235,3 +229,42 @@ class TestClassifier(DirectoriesMixin, TestCase): self.classifier.train() self.assertListEqual(self.classifier.predict_tags(doc1.content), [t1.pk]) self.assertListEqual(self.classifier.predict_tags(doc2.content), []) + + def test_load_classifier_not_exists(self): + self.assertFalse(os.path.exists(settings.MODEL_FILE)) + self.assertIsNone(load_classifier()) + + @mock.patch("documents.classifier.DocumentClassifier.load") + def test_load_classifier(self, load): + Path(settings.MODEL_FILE).touch() + self.assertIsNotNone(load_classifier()) + load.assert_called_once() + + @override_settings(CACHES={'default': {'BACKEND': 'django.core.cache.backends.locmem.LocMemCache'}}) + @override_settings(MODEL_FILE=os.path.join(os.path.dirname(__file__), "data", "model.pickle")) + @pytest.mark.skip(reason="Disabled caching due to high memory usage - need to investigate.") + def test_load_classifier_cached(self): + classifier = load_classifier() + self.assertIsNotNone(classifier) + + with mock.patch("documents.classifier.DocumentClassifier.load") as load: + classifier2 = load_classifier() + load.assert_not_called() + + @mock.patch("documents.classifier.DocumentClassifier.load") + def test_load_classifier_incompatible_version(self, load): + Path(settings.MODEL_FILE).touch() + self.assertTrue(os.path.exists(settings.MODEL_FILE)) + + load.side_effect = IncompatibleClassifierVersionError() + self.assertIsNone(load_classifier()) + self.assertFalse(os.path.exists(settings.MODEL_FILE)) + + @mock.patch("documents.classifier.DocumentClassifier.load") + def test_load_classifier_os_error(self, load): + Path(settings.MODEL_FILE).touch() + self.assertTrue(os.path.exists(settings.MODEL_FILE)) + + load.side_effect = OSError() + self.assertIsNone(load_classifier()) + self.assertTrue(os.path.exists(settings.MODEL_FILE)) diff --git a/src/documents/tests/test_consumer.py b/src/documents/tests/test_consumer.py index b4b19be4c..514c3ca19 100644 --- a/src/documents/tests/test_consumer.py +++ b/src/documents/tests/test_consumer.py @@ -5,12 +5,14 @@ import tempfile from unittest import mock from unittest.mock import MagicMock +from django.conf import settings from django.test import TestCase, override_settings from .utils import DirectoriesMixin from ..consumer import Consumer, ConsumerError from ..models import FileInfo, Tag, Correspondent, DocumentType, Document from ..parsers import DocumentParser, ParseError +from ..tasks import sanity_check class TestAttributes(TestCase): @@ -29,81 +31,6 @@ class TestAttributes(TestCase): self.assertEqual(tuple([t.name for t in file_info.tags]), tags, filename) - def test_guess_attributes_from_name0(self): - self._test_guess_attributes_from_name( - "Sender - Title.pdf", "Sender", "Title", ()) - - def test_guess_attributes_from_name1(self): - self._test_guess_attributes_from_name( - "Spaced Sender - Title.pdf", "Spaced Sender", "Title", ()) - - def test_guess_attributes_from_name2(self): - self._test_guess_attributes_from_name( - "Sender - Spaced Title.pdf", "Sender", "Spaced Title", ()) - - def test_guess_attributes_from_name3(self): - self._test_guess_attributes_from_name( - "Dashed-Sender - Title.pdf", "Dashed-Sender", "Title", ()) - - def test_guess_attributes_from_name4(self): - self._test_guess_attributes_from_name( - "Sender - Dashed-Title.pdf", "Sender", "Dashed-Title", ()) - - def test_guess_attributes_from_name5(self): - self._test_guess_attributes_from_name( - "Sender - Title - tag1,tag2,tag3.pdf", - "Sender", - "Title", - self.TAGS - ) - - def test_guess_attributes_from_name6(self): - self._test_guess_attributes_from_name( - "Spaced Sender - Title - tag1,tag2,tag3.pdf", - "Spaced Sender", - "Title", - self.TAGS - ) - - def test_guess_attributes_from_name7(self): - self._test_guess_attributes_from_name( - "Sender - Spaced Title - tag1,tag2,tag3.pdf", - "Sender", - "Spaced Title", - self.TAGS - ) - - def test_guess_attributes_from_name8(self): - self._test_guess_attributes_from_name( - "Dashed-Sender - Title - tag1,tag2,tag3.pdf", - "Dashed-Sender", - "Title", - self.TAGS - ) - - def test_guess_attributes_from_name9(self): - self._test_guess_attributes_from_name( - "Sender - Dashed-Title - tag1,tag2,tag3.pdf", - "Sender", - "Dashed-Title", - self.TAGS - ) - - def test_guess_attributes_from_name10(self): - self._test_guess_attributes_from_name( - "Σενδερ - Τιτλε - tag1,tag2,tag3.pdf", - "Σενδερ", - "Τιτλε", - self.TAGS - ) - - def test_guess_attributes_from_name_when_correspondent_empty(self): - self._test_guess_attributes_from_name( - ' - weird empty correspondent but should not break.pdf', - None, - 'weird empty correspondent but should not break', - () - ) def test_guess_attributes_from_name_when_title_starts_with_dash(self): self._test_guess_attributes_from_name( @@ -121,28 +48,6 @@ class TestAttributes(TestCase): () ) - def test_guess_attributes_from_name_when_title_is_empty(self): - self._test_guess_attributes_from_name( - 'weird correspondent but should not break - .pdf', - 'weird correspondent but should not break', - '', - () - ) - - def test_case_insensitive_tag_creation(self): - """ - Tags should be detected and created as lower case. - :return: - """ - - filename = "Title - Correspondent - tAg1,TAG2.pdf" - self.assertEqual(len(FileInfo.from_filename(filename).tags), 2) - - path = "Title - Correspondent - tag1,tag2.pdf" - self.assertEqual(len(FileInfo.from_filename(filename).tags), 2) - - self.assertEqual(Tag.objects.all().count(), 2) - class TestFieldPermutations(TestCase): @@ -199,69 +104,7 @@ class TestFieldPermutations(TestCase): filename = template.format(**spec) self._test_guessed_attributes(filename, **spec) - def test_title_and_correspondent(self): - template = '{correspondent} - {title}.pdf' - for correspondent in self.valid_correspondents: - for title in self.valid_titles: - spec = dict(correspondent=correspondent, title=title) - filename = template.format(**spec) - self._test_guessed_attributes(filename, **spec) - - def test_title_and_correspondent_and_tags(self): - template = '{correspondent} - {title} - {tags}.pdf' - for correspondent in self.valid_correspondents: - for title in self.valid_titles: - for tags in self.valid_tags: - spec = dict(correspondent=correspondent, title=title, - tags=tags) - filename = template.format(**spec) - self._test_guessed_attributes(filename, **spec) - - def test_created_and_correspondent_and_title_and_tags(self): - - template = ( - "{created} - " - "{correspondent} - " - "{title} - " - "{tags}.pdf" - ) - - for created in self.valid_dates: - for correspondent in self.valid_correspondents: - for title in self.valid_titles: - for tags in self.valid_tags: - spec = { - "created": created, - "correspondent": correspondent, - "title": title, - "tags": tags, - } - self._test_guessed_attributes( - template.format(**spec), **spec) - - def test_created_and_correspondent_and_title(self): - - template = "{created} - {correspondent} - {title}.pdf" - - for created in self.valid_dates: - for correspondent in self.valid_correspondents: - for title in self.valid_titles: - - # Skip cases where title looks like a tag as we can't - # accommodate such cases. - if title.lower() == title: - continue - - spec = { - "created": created, - "correspondent": correspondent, - "title": title - } - self._test_guessed_attributes( - template.format(**spec), **spec) - def test_created_and_title(self): - template = "{created} - {title}.pdf" for created in self.valid_dates: @@ -273,21 +116,6 @@ class TestFieldPermutations(TestCase): self._test_guessed_attributes( template.format(**spec), **spec) - def test_created_and_title_and_tags(self): - - template = "{created} - {title} - {tags}.pdf" - - for created in self.valid_dates: - for title in self.valid_titles: - for tags in self.valid_tags: - spec = { - "created": created, - "title": title, - "tags": tags - } - self._test_guessed_attributes( - template.format(**spec), **spec) - def test_invalid_date_format(self): info = FileInfo.from_filename("06112017Z - title.pdf") self.assertEqual(info.title, "title") @@ -336,54 +164,46 @@ class TestFieldPermutations(TestCase): info = FileInfo.from_filename(filename) self.assertEqual(info.title, "anotherall") - # Complex transformation without date in replacement string - with self.settings( - FILENAME_PARSE_TRANSFORMS=[(exact_patt, repl1)]): - info = FileInfo.from_filename(filename) - self.assertEqual(info.title, "0001") - self.assertEqual(len(info.tags), 2) - self.assertEqual(info.tags[0].name, "tag1") - self.assertEqual(info.tags[1].name, "tag2") - self.assertIsNone(info.created) - - # Complex transformation with date in replacement string - with self.settings( - FILENAME_PARSE_TRANSFORMS=[ - (none_patt, "none.gif"), - (exact_patt, repl2), # <-- matches - (exact_patt, repl1), - (all_patt, "all.gif")]): - info = FileInfo.from_filename(filename) - self.assertEqual(info.title, "0001") - self.assertEqual(len(info.tags), 2) - self.assertEqual(info.tags[0].name, "tag1") - self.assertEqual(info.tags[1].name, "tag2") - self.assertEqual(info.created.year, 2019) - self.assertEqual(info.created.month, 9) - self.assertEqual(info.created.day, 8) - class DummyParser(DocumentParser): - def get_thumbnail(self, document_path, mime_type): + def get_thumbnail(self, document_path, mime_type, file_name=None): # not important during tests raise NotImplementedError() def __init__(self, logging_group, scratch_dir, archive_path): - super(DummyParser, self).__init__(logging_group) + super(DummyParser, self).__init__(logging_group, None) _, self.fake_thumb = tempfile.mkstemp(suffix=".png", dir=scratch_dir) self.archive_path = archive_path - def get_optimised_thumbnail(self, document_path, mime_type): + def get_optimised_thumbnail(self, document_path, mime_type, file_name=None): return self.fake_thumb - def parse(self, document_path, mime_type): + def parse(self, document_path, mime_type, file_name=None): self.text = "The Text" +class CopyParser(DocumentParser): + + def get_thumbnail(self, document_path, mime_type, file_name=None): + return self.fake_thumb + + def get_optimised_thumbnail(self, document_path, mime_type, file_name=None): + return self.fake_thumb + + def __init__(self, logging_group, progress_callback=None): + super(CopyParser, self).__init__(logging_group, progress_callback) + _, self.fake_thumb = tempfile.mkstemp(suffix=".png", dir=self.tempdir) + + def parse(self, document_path, mime_type, file_name=None): + self.text = "The text" + self.archive_path = os.path.join(self.tempdir, "archive.pdf") + shutil.copy(document_path, self.archive_path) + + class FaultyParser(DocumentParser): - def get_thumbnail(self, document_path, mime_type): + def get_thumbnail(self, document_path, mime_type, file_name=None): # not important during tests raise NotImplementedError() @@ -391,10 +211,10 @@ class FaultyParser(DocumentParser): super(FaultyParser, self).__init__(logging_group) _, self.fake_thumb = tempfile.mkstemp(suffix=".png", dir=scratch_dir) - def get_optimised_thumbnail(self, document_path, mime_type): + def get_optimised_thumbnail(self, document_path, mime_type, file_name=None): return self.fake_thumb - def parse(self, document_path, mime_type): + def parse(self, document_path, mime_type, file_name=None): raise ParseError("Does not compute.") @@ -403,6 +223,8 @@ def fake_magic_from_file(file, mime=False): if mime: if os.path.splitext(file)[1] == ".pdf": return "application/pdf" + elif os.path.splitext(file)[1] == ".png": + return "image/png" else: return "unknown" else: @@ -412,10 +234,24 @@ def fake_magic_from_file(file, mime=False): @mock.patch("documents.consumer.magic.from_file", fake_magic_from_file) class TestConsumer(DirectoriesMixin, TestCase): - def make_dummy_parser(self, logging_group): + def _assert_first_last_send_progress(self, first_status="STARTING", last_status="SUCCESS", first_progress=0, first_progress_max=100, last_progress=100, last_progress_max=100): + + self._send_progress.assert_called() + + args, kwargs = self._send_progress.call_args_list[0] + self.assertEqual(args[0], first_progress) + self.assertEqual(args[1], first_progress_max) + self.assertEqual(args[2], first_status) + + args, kwargs = self._send_progress.call_args_list[len(self._send_progress.call_args_list) - 1] + self.assertEqual(args[0], last_progress) + self.assertEqual(args[1], last_progress_max) + self.assertEqual(args[2], last_status) + + def make_dummy_parser(self, logging_group, progress_callback=None): return DummyParser(logging_group, self.dirs.scratch_dir, self.get_test_archive_file()) - def make_faulty_parser(self, logging_group): + def make_faulty_parser(self, logging_group, progress_callback=None): return FaultyParser(logging_group, self.dirs.scratch_dir) def setUp(self): @@ -428,7 +264,11 @@ class TestConsumer(DirectoriesMixin, TestCase): "mime_types": {"application/pdf": ".pdf"}, "weight": 0 })] + self.addCleanup(patcher.stop) + # this prevents websocket message reports during testing. + patcher = mock.patch("documents.consumer.Consumer._send_progress") + self._send_progress = patcher.start() self.addCleanup(patcher.stop) self.consumer = Consumer() @@ -456,6 +296,7 @@ class TestConsumer(DirectoriesMixin, TestCase): self.assertIsNone(document.correspondent) self.assertIsNone(document.document_type) self.assertEqual(document.filename, "0000001.pdf") + self.assertEqual(document.archive_filename, "0000001.pdf") self.assertTrue(os.path.isfile( document.source_path @@ -474,31 +315,57 @@ class TestConsumer(DirectoriesMixin, TestCase): self.assertFalse(os.path.isfile(filename)) + self._assert_first_last_send_progress() + + @override_settings(PAPERLESS_FILENAME_FORMAT=None) + def testDeleteMacFiles(self): + # https://github.com/jonaswinkler/paperless-ng/discussions/1037 + + filename = self.get_test_file() + shadow_file = os.path.join(self.dirs.scratch_dir, "._sample.pdf") + + shutil.copy(filename, shadow_file) + + self.assertTrue(os.path.isfile(shadow_file)) + + document = self.consumer.try_consume_file(filename) + + self.assertTrue(os.path.isfile( + document.source_path + )) + + self.assertFalse(os.path.isfile(shadow_file)) + self.assertFalse(os.path.isfile(filename)) + + def testOverrideFilename(self): filename = self.get_test_file() - override_filename = "My Bank - Statement for November.pdf" + override_filename = "Statement for November.pdf" document = self.consumer.try_consume_file(filename, override_filename=override_filename) - self.assertEqual(document.correspondent.name, "My Bank") self.assertEqual(document.title, "Statement for November") - def testOverrideTitle(self): + self._assert_first_last_send_progress() + def testOverrideTitle(self): document = self.consumer.try_consume_file(self.get_test_file(), override_title="Override Title") self.assertEqual(document.title, "Override Title") + self._assert_first_last_send_progress() def testOverrideCorrespondent(self): c = Correspondent.objects.create(name="test") document = self.consumer.try_consume_file(self.get_test_file(), override_correspondent_id=c.pk) self.assertEqual(document.correspondent.id, c.id) + self._assert_first_last_send_progress() def testOverrideDocumentType(self): dt = DocumentType.objects.create(name="test") document = self.consumer.try_consume_file(self.get_test_file(), override_document_type_id=dt.pk) self.assertEqual(document.document_type.id, dt.id) + self._assert_first_last_send_progress() def testOverrideTags(self): t1 = Tag.objects.create(name="t1") @@ -509,37 +376,42 @@ class TestConsumer(DirectoriesMixin, TestCase): self.assertIn(t1, document.tags.all()) self.assertNotIn(t2, document.tags.all()) self.assertIn(t3, document.tags.all()) + self._assert_first_last_send_progress() def testNotAFile(self): - try: - self.consumer.try_consume_file("non-existing-file") - except ConsumerError as e: - self.assertTrue(str(e).endswith('It is not a file')) - return - self.fail("Should throw exception") + self.assertRaisesMessage( + ConsumerError, + "File not found", + self.consumer.try_consume_file, + "non-existing-file" + ) + + self._assert_first_last_send_progress(last_status="FAILED") def testDuplicates1(self): self.consumer.try_consume_file(self.get_test_file()) - try: - self.consumer.try_consume_file(self.get_test_file()) - except ConsumerError as e: - self.assertTrue(str(e).endswith("It is a duplicate.")) - return + self.assertRaisesMessage( + ConsumerError, + "It is a duplicate", + self.consumer.try_consume_file, + self.get_test_file() + ) - self.fail("Should throw exception") + self._assert_first_last_send_progress(last_status="FAILED") def testDuplicates2(self): self.consumer.try_consume_file(self.get_test_file()) - try: - self.consumer.try_consume_file(self.get_test_archive_file()) - except ConsumerError as e: - self.assertTrue(str(e).endswith("It is a duplicate.")) - return + self.assertRaisesMessage( + ConsumerError, + "It is a duplicate", + self.consumer.try_consume_file, + self.get_test_archive_file() + ) - self.fail("Should throw exception") + self._assert_first_last_send_progress(last_status="FAILED") def testDuplicates3(self): self.consumer.try_consume_file(self.get_test_archive_file()) @@ -549,13 +421,15 @@ class TestConsumer(DirectoriesMixin, TestCase): def testNoParsers(self, m): m.return_value = [] - try: - self.consumer.try_consume_file(self.get_test_file()) - except ConsumerError as e: - self.assertTrue("No parsers abvailable for" in str(e)) - return + self.assertRaisesMessage( + ConsumerError, + "sample.pdf: Unsupported mime type application/pdf", + self.consumer.try_consume_file, + self.get_test_file() + ) + + self._assert_first_last_send_progress(last_status="FAILED") - self.fail("Should throw exception") @mock.patch("documents.parsers.document_consumer_declaration.send") def testFaultyParser(self, m): @@ -565,24 +439,28 @@ class TestConsumer(DirectoriesMixin, TestCase): "weight": 0 })] - try: - self.consumer.try_consume_file(self.get_test_file()) - except ConsumerError as e: - self.assertEqual(str(e), "Does not compute.") - return + self.assertRaisesMessage( + ConsumerError, + "sample.pdf: Error while consuming document sample.pdf: Does not compute.", + self.consumer.try_consume_file, + self.get_test_file() + ) - self.fail("Should throw exception.") + self._assert_first_last_send_progress(last_status="FAILED") @mock.patch("documents.consumer.Consumer._write") def testPostSaveError(self, m): filename = self.get_test_file() m.side_effect = OSError("NO.") - try: - self.consumer.try_consume_file(filename) - except ConsumerError as e: - self.assertEqual(str(e), "NO.") - else: - self.fail("Should raise exception") + + self.assertRaisesMessage( + ConsumerError, + "sample.pdf: The following error occured while consuming sample.pdf: NO.", + self.consumer.try_consume_file, + filename + ) + + self._assert_first_last_send_progress(last_status="FAILED") # file not deleted self.assertTrue(os.path.isfile(filename)) @@ -594,11 +472,13 @@ class TestConsumer(DirectoriesMixin, TestCase): def testFilenameHandling(self): filename = self.get_test_file() - document = self.consumer.try_consume_file(filename, override_filename="Bank - Test.pdf", override_title="new docs") + document = self.consumer.try_consume_file(filename, override_title="new docs") self.assertEqual(document.title, "new docs") - self.assertEqual(document.correspondent.name, "Bank") - self.assertEqual(document.filename, "Bank/new docs.pdf") + self.assertEqual(document.filename, "none/new docs.pdf") + self.assertEqual(document.archive_filename, "none/new docs.pdf") + + self._assert_first_last_send_progress() @override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}") @mock.patch("documents.signals.handlers.generate_unique_filename") @@ -611,20 +491,22 @@ class TestConsumer(DirectoriesMixin, TestCase): filenames.insert(0, f) return f - m.side_effect = lambda f, root: get_filename() + m.side_effect = lambda f, archive_filename = False: get_filename() filename = self.get_test_file() Tag.objects.create(name="test", is_inbox_tag=True) - document = self.consumer.try_consume_file(filename, override_filename="Bank - Test.pdf", override_title="new docs") + document = self.consumer.try_consume_file(filename, override_title="new docs") self.assertEqual(document.title, "new docs") - self.assertEqual(document.correspondent.name, "Bank") self.assertIsNotNone(os.path.isfile(document.title)) self.assertTrue(os.path.isfile(document.source_path)) + self.assertTrue(os.path.isfile(document.archive_path)) - @mock.patch("documents.consumer.DocumentClassifier") + self._assert_first_last_send_progress() + + @mock.patch("documents.consumer.load_classifier") def testClassifyDocument(self, m): correspondent = Correspondent.objects.create(name="test") dtype = DocumentType.objects.create(name="test") @@ -642,3 +524,161 @@ class TestConsumer(DirectoriesMixin, TestCase): self.assertEqual(document.document_type, dtype) self.assertIn(t1, document.tags.all()) self.assertNotIn(t2, document.tags.all()) + + self._assert_first_last_send_progress() + + @override_settings(CONSUMER_DELETE_DUPLICATES=True) + def test_delete_duplicate(self): + dst = self.get_test_file() + self.assertTrue(os.path.isfile(dst)) + doc = self.consumer.try_consume_file(dst) + + self._assert_first_last_send_progress() + + self.assertFalse(os.path.isfile(dst)) + self.assertIsNotNone(doc) + + self._send_progress.reset_mock() + + dst = self.get_test_file() + self.assertTrue(os.path.isfile(dst)) + self.assertRaises(ConsumerError, self.consumer.try_consume_file, dst) + self.assertFalse(os.path.isfile(dst)) + self._assert_first_last_send_progress(last_status="FAILED") + + @override_settings(CONSUMER_DELETE_DUPLICATES=False) + def test_no_delete_duplicate(self): + dst = self.get_test_file() + self.assertTrue(os.path.isfile(dst)) + doc = self.consumer.try_consume_file(dst) + + self.assertFalse(os.path.isfile(dst)) + self.assertIsNotNone(doc) + + dst = self.get_test_file() + self.assertTrue(os.path.isfile(dst)) + self.assertRaises(ConsumerError, self.consumer.try_consume_file, dst) + self.assertTrue(os.path.isfile(dst)) + + self._assert_first_last_send_progress(last_status="FAILED") + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + @mock.patch("documents.parsers.document_consumer_declaration.send") + def test_similar_filenames(self, m): + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), os.path.join(settings.CONSUMPTION_DIR, "simple.pdf")) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.png"), os.path.join(settings.CONSUMPTION_DIR, "simple.png")) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple-noalpha.png"), os.path.join(settings.CONSUMPTION_DIR, "simple.png.pdf")) + m.return_value = [(None, { + "parser": CopyParser, + "mime_types": {"application/pdf": ".pdf", "image/png": ".png"}, + "weight": 0 + })] + doc1 = self.consumer.try_consume_file(os.path.join(settings.CONSUMPTION_DIR, "simple.png")) + doc2 = self.consumer.try_consume_file(os.path.join(settings.CONSUMPTION_DIR, "simple.pdf")) + doc3 = self.consumer.try_consume_file(os.path.join(settings.CONSUMPTION_DIR, "simple.png.pdf")) + + self.assertEqual(doc1.filename, "simple.png") + self.assertEqual(doc1.archive_filename, "simple.pdf") + self.assertEqual(doc2.filename, "simple.pdf") + self.assertEqual(doc2.archive_filename, "simple_01.pdf") + self.assertEqual(doc3.filename, "simple.png.pdf") + self.assertEqual(doc3.archive_filename, "simple.png.pdf") + + sanity_check() + + +class PreConsumeTestCase(TestCase): + + @mock.patch("documents.consumer.Popen") + @override_settings(PRE_CONSUME_SCRIPT=None) + def test_no_pre_consume_script(self, m): + c = Consumer() + c.path = "path-to-file" + c.run_pre_consume_script() + m.assert_not_called() + + @mock.patch("documents.consumer.Popen") + @mock.patch("documents.consumer.Consumer._send_progress") + @override_settings(PRE_CONSUME_SCRIPT="does-not-exist") + def test_pre_consume_script_not_found(self, m, m2): + c = Consumer() + c.filename = "somefile.pdf" + c.path = "path-to-file" + self.assertRaises(ConsumerError, c.run_pre_consume_script) + + @mock.patch("documents.consumer.Popen") + def test_pre_consume_script(self, m): + with tempfile.NamedTemporaryFile() as script: + with override_settings(PRE_CONSUME_SCRIPT=script.name): + c = Consumer() + c.path = "path-to-file" + c.run_pre_consume_script() + + m.assert_called_once() + + args, kwargs = m.call_args + + command = args[0] + + self.assertEqual(command[0], script.name) + self.assertEqual(command[1], "path-to-file") + + +class PostConsumeTestCase(TestCase): + + @mock.patch("documents.consumer.Popen") + @override_settings(POST_CONSUME_SCRIPT=None) + def test_no_post_consume_script(self, m): + doc = Document.objects.create(title="Test", mime_type="application/pdf") + tag1 = Tag.objects.create(name="a") + tag2 = Tag.objects.create(name="b") + doc.tags.add(tag1) + doc.tags.add(tag2) + + Consumer().run_post_consume_script(doc) + + m.assert_not_called() + + @override_settings(POST_CONSUME_SCRIPT="does-not-exist") + @mock.patch("documents.consumer.Consumer._send_progress") + def test_post_consume_script_not_found(self, m): + doc = Document.objects.create(title="Test", mime_type="application/pdf") + c = Consumer() + c.filename = "somefile.pdf" + self.assertRaises(ConsumerError, c.run_post_consume_script, doc) + + @mock.patch("documents.consumer.Popen") + def test_post_consume_script_simple(self, m): + with tempfile.NamedTemporaryFile() as script: + with override_settings(POST_CONSUME_SCRIPT=script.name): + doc = Document.objects.create(title="Test", mime_type="application/pdf") + + Consumer().run_post_consume_script(doc) + + m.assert_called_once() + + @mock.patch("documents.consumer.Popen") + def test_post_consume_script_with_correspondent(self, m): + with tempfile.NamedTemporaryFile() as script: + with override_settings(POST_CONSUME_SCRIPT=script.name): + c = Correspondent.objects.create(name="my_bank") + doc = Document.objects.create(title="Test", mime_type="application/pdf", correspondent=c) + tag1 = Tag.objects.create(name="a") + tag2 = Tag.objects.create(name="b") + doc.tags.add(tag1) + doc.tags.add(tag2) + + Consumer().run_post_consume_script(doc) + + m.assert_called_once() + + args, kwargs = m.call_args + + command = args[0] + + self.assertEqual(command[0], script.name) + self.assertEqual(command[1], str(doc.pk)) + self.assertEqual(command[5], f"/api/documents/{doc.pk}/download/") + self.assertEqual(command[6], f"/api/documents/{doc.pk}/thumb/") + self.assertEqual(command[7], "my_bank") + self.assertCountEqual(command[8].split(","), ["a", "b"]) diff --git a/src/documents/tests/test_date_parsing.py b/src/documents/tests/test_date_parsing.py index 357b0937e..50543ee75 100644 --- a/src/documents/tests/test_date_parsing.py +++ b/src/documents/tests/test_date_parsing.py @@ -1,7 +1,6 @@ import datetime import os import shutil -from unittest import mock from uuid import uuid4 from dateutil import tz @@ -9,7 +8,6 @@ from django.conf import settings from django.test import TestCase, override_settings from documents.parsers import parse_date -from paperless_tesseract.parsers import RasterisedDocumentParser class TestDate(TestCase): @@ -138,3 +136,18 @@ class TestDate(TestCase): @override_settings(FILENAME_DATE_ORDER="YMD") def test_filename_date_parse_invalid(self, *args): self.assertIsNone(parse_date("/tmp/20 408000l 2475 - test.pdf", "No date in here")) + + @override_settings(IGNORE_DATES=(datetime.date(2019, 11, 3), datetime.date(2020, 1, 17))) + def test_ignored_dates(self, *args): + text = ( + "lorem ipsum 110319, 20200117 and lorem 13.02.2018 lorem " + "ipsum" + ) + date = parse_date("", text) + self.assertEqual( + date, + datetime.datetime( + 2018, 2, 13, 0, 0, + tzinfo=tz.gettz(settings.TIME_ZONE) + ) + ) diff --git a/src/documents/tests/test_document_model.py b/src/documents/tests/test_document_model.py index 74bd9a2a7..87261ecd9 100644 --- a/src/documents/tests/test_document_model.py +++ b/src/documents/tests/test_document_model.py @@ -1,10 +1,10 @@ import shutil import tempfile -from datetime import datetime from pathlib import Path from unittest import mock from django.test import TestCase, override_settings +from django.utils import timezone from ..models import Document, Correspondent @@ -47,20 +47,20 @@ class TestDocument(TestCase): def test_file_name(self): - doc = Document(mime_type="application/pdf", title="test", created=datetime(2020, 12, 25)) + doc = Document(mime_type="application/pdf", title="test", created=timezone.datetime(2020, 12, 25)) self.assertEqual(doc.get_public_filename(), "2020-12-25 test.pdf") def test_file_name_jpg(self): - doc = Document(mime_type="image/jpeg", title="test", created=datetime(2020, 12, 25)) + doc = Document(mime_type="image/jpeg", title="test", created=timezone.datetime(2020, 12, 25)) self.assertEqual(doc.get_public_filename(), "2020-12-25 test.jpg") def test_file_name_unknown(self): - doc = Document(mime_type="application/zip", title="test", created=datetime(2020, 12, 25)) + doc = Document(mime_type="application/zip", title="test", created=timezone.datetime(2020, 12, 25)) self.assertEqual(doc.get_public_filename(), "2020-12-25 test.zip") def test_file_name_invalid_type(self): - doc = Document(mime_type="image/jpegasd", title="test", created=datetime(2020, 12, 25)) + doc = Document(mime_type="image/jpegasd", title="test", created=timezone.datetime(2020, 12, 25)) self.assertEqual(doc.get_public_filename(), "2020-12-25 test") diff --git a/src/documents/tests/test_file_handling.py b/src/documents/tests/test_file_handling.py index 719b0078a..59af7e317 100644 --- a/src/documents/tests/test_file_handling.py +++ b/src/documents/tests/test_file_handling.py @@ -9,11 +9,12 @@ from unittest import mock from django.conf import settings from django.db import DatabaseError from django.test import TestCase, override_settings +from django.utils import timezone from .utils import DirectoriesMixin from ..file_handling import generate_filename, create_source_path_directory, delete_empty_directories, \ generate_unique_filename -from ..models import Document, Correspondent +from ..models import Document, Correspondent, Tag, DocumentType class TestFileHandling(DirectoriesMixin, TestCase): @@ -189,6 +190,24 @@ class TestFileHandling(DirectoriesMixin, TestCase): self.assertEqual(os.path.isdir(settings.ORIGINALS_DIR + "/none"), True) self.assertTrue(os.path.isfile(important_file)) + @override_settings(PAPERLESS_FILENAME_FORMAT="{document_type} - {title}") + def test_document_type(self): + dt = DocumentType.objects.create(name="my_doc_type") + d = Document.objects.create(title="the_doc", mime_type="application/pdf") + + self.assertEqual(generate_filename(d), "none - the_doc.pdf") + + d.document_type = dt + + self.assertEqual(generate_filename(d), "my_doc_type - the_doc.pdf") + + @override_settings(PAPERLESS_FILENAME_FORMAT="{asn} - {title}") + def test_asn(self): + d1 = Document.objects.create(title="the_doc", mime_type="application/pdf", archive_serial_number=652, checksum="A") + d2 = Document.objects.create(title="the_doc", mime_type="application/pdf", archive_serial_number=None, checksum="B") + self.assertEqual(generate_filename(d1), "652 - the_doc.pdf") + self.assertEqual(generate_filename(d2), "none - the_doc.pdf") + @override_settings(PAPERLESS_FILENAME_FORMAT="{tags[type]}") def test_tags_with_underscore(self): document = Document() @@ -267,6 +286,57 @@ class TestFileHandling(DirectoriesMixin, TestCase): self.assertEqual(generate_filename(document), "none.pdf") + @override_settings(PAPERLESS_FILENAME_FORMAT="{tags}") + def test_tags_without_args(self): + document = Document() + document.mime_type = "application/pdf" + document.storage_type = Document.STORAGE_TYPE_UNENCRYPTED + document.save() + + self.assertEqual(generate_filename(document), f"{document.pk:07}.pdf") + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title} {tag_list}") + def test_tag_list(self): + doc = Document.objects.create(title="doc1", mime_type="application/pdf") + doc.tags.create(name="tag2") + doc.tags.create(name="tag1") + + self.assertEqual(generate_filename(doc), "doc1 tag1,tag2.pdf") + + doc = Document.objects.create(title="doc2", checksum="B", mime_type="application/pdf") + + self.assertEqual(generate_filename(doc), "doc2.pdf") + + @override_settings(PAPERLESS_FILENAME_FORMAT="//etc/something/{title}") + def test_filename_relative(self): + doc = Document.objects.create(title="doc1", mime_type="application/pdf") + doc.filename = generate_filename(doc) + doc.save() + + self.assertEqual(doc.source_path, os.path.join(settings.ORIGINALS_DIR, "etc", "something", "doc1.pdf")) + + @override_settings(PAPERLESS_FILENAME_FORMAT="{created_year}-{created_month}-{created_day}") + def test_created_year_month_day(self): + d1 = timezone.make_aware(datetime.datetime(2020, 3, 6, 1, 1, 1)) + doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", created=d1) + + self.assertEqual(generate_filename(doc1), "2020-03-06.pdf") + + doc1.created = timezone.make_aware(datetime.datetime(2020, 11, 16, 1, 1, 1)) + + self.assertEqual(generate_filename(doc1), "2020-11-16.pdf") + + @override_settings(PAPERLESS_FILENAME_FORMAT="{added_year}-{added_month}-{added_day}") + def test_added_year_month_day(self): + d1 = timezone.make_aware(datetime.datetime(232, 1, 9, 1, 1, 1)) + doc1 = Document.objects.create(title="doc1", mime_type="application/pdf", added=d1) + + self.assertEqual(generate_filename(doc1), "232-01-09.pdf") + + doc1.added = timezone.make_aware(datetime.datetime(2020, 11, 16, 1, 1, 1)) + + self.assertEqual(generate_filename(doc1), "2020-11-16.pdf") + @override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{correspondent}/{correspondent}") def test_nested_directory_cleanup(self): document = Document() @@ -376,6 +446,18 @@ class TestFileHandling(DirectoriesMixin, TestCase): self.assertEqual(document2.filename, "qwe.pdf") + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + @mock.patch("documents.signals.handlers.Document.objects.filter") + def test_no_update_without_change(self, m): + doc = Document.objects.create(title="document", filename="document.pdf", archive_filename="document.pdf", checksum="A", archive_checksum="B", mime_type="application/pdf") + Path(doc.source_path).touch() + Path(doc.archive_path).touch() + + doc.save() + + m.assert_not_called() + + class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): @@ -385,7 +467,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", filename="0000001.pdf", checksum="A", archive_filename="0000001.pdf", archive_checksum="B") self.assertTrue(os.path.isfile(original)) self.assertTrue(os.path.isfile(archive)) @@ -398,7 +480,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") self.assertFalse(os.path.isfile(original)) self.assertFalse(os.path.isfile(archive)) @@ -412,7 +494,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): original = os.path.join(settings.ORIGINALS_DIR, "0000001.pdf") archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") self.assertTrue(os.path.isfile(original)) self.assertFalse(os.path.isfile(archive)) @@ -423,14 +505,49 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): def test_move_archive_exists(self): original = os.path.join(settings.ORIGINALS_DIR, "0000001.pdf") archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") + existing_archive_file = os.path.join(settings.ARCHIVE_DIR, "none", "my_doc.pdf") Path(original).touch() Path(archive).touch() os.makedirs(os.path.join(settings.ARCHIVE_DIR, "none")) - Path(os.path.join(settings.ARCHIVE_DIR, "none", "my_doc.pdf")).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + Path(existing_archive_file).touch() + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") + + self.assertFalse(os.path.isfile(original)) + self.assertFalse(os.path.isfile(archive)) + self.assertTrue(os.path.isfile(doc.source_path)) + self.assertTrue(os.path.isfile(doc.archive_path)) + self.assertTrue(os.path.isfile(existing_archive_file)) + self.assertEqual(doc.archive_filename, "none/my_doc_01.pdf") + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + def test_move_original_only(self): + original = os.path.join(settings.ORIGINALS_DIR, "document_01.pdf") + archive = os.path.join(settings.ARCHIVE_DIR, "document.pdf") + Path(original).touch() + Path(archive).touch() + + doc = Document.objects.create(mime_type="application/pdf", title="document", filename="document_01.pdf", checksum="A", + archive_checksum="B", archive_filename="document.pdf") + + self.assertEqual(doc.filename, "document.pdf") + self.assertEqual(doc.archive_filename, "document.pdf") + + self.assertTrue(os.path.isfile(doc.source_path)) + self.assertTrue(os.path.isfile(doc.archive_path)) + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + def test_move_archive_only(self): + original = os.path.join(settings.ORIGINALS_DIR, "document.pdf") + archive = os.path.join(settings.ARCHIVE_DIR, "document_01.pdf") + Path(original).touch() + Path(archive).touch() + + doc = Document.objects.create(mime_type="application/pdf", title="document", filename="document.pdf", checksum="A", + archive_checksum="B", archive_filename="document_01.pdf") + + self.assertEqual(doc.filename, "document.pdf") + self.assertEqual(doc.archive_filename, "document.pdf") - self.assertTrue(os.path.isfile(original)) - self.assertTrue(os.path.isfile(archive)) self.assertTrue(os.path.isfile(doc.source_path)) self.assertTrue(os.path.isfile(doc.archive_path)) @@ -451,8 +568,9 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") + m.assert_called() self.assertTrue(os.path.isfile(original)) self.assertTrue(os.path.isfile(archive)) self.assertTrue(os.path.isfile(doc.source_path)) @@ -464,7 +582,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") #Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", archive_filename="0000001.pdf", checksum="A", archive_checksum="B") self.assertFalse(os.path.isfile(original)) self.assertTrue(os.path.isfile(archive)) @@ -488,19 +606,21 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", archive_filename="0000001.pdf", checksum="A", archive_checksum="B") + m.assert_called() self.assertTrue(os.path.isfile(original)) self.assertTrue(os.path.isfile(archive)) self.assertTrue(os.path.isfile(doc.source_path)) self.assertTrue(os.path.isfile(doc.archive_path)) + @override_settings(PAPERLESS_FILENAME_FORMAT="") def test_archive_deleted(self): original = os.path.join(settings.ORIGINALS_DIR, "0000001.pdf") archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document.objects.create(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") self.assertTrue(os.path.isfile(original)) self.assertTrue(os.path.isfile(archive)) @@ -514,6 +634,28 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): self.assertFalse(os.path.isfile(doc.source_path)) self.assertFalse(os.path.isfile(doc.archive_path)) + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + def test_archive_deleted2(self): + original = os.path.join(settings.ORIGINALS_DIR, "document.png") + original2 = os.path.join(settings.ORIGINALS_DIR, "0000001.pdf") + archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") + Path(original).touch() + Path(original2).touch() + Path(archive).touch() + + doc1 = Document.objects.create(mime_type="image/png", title="document", filename="document.png", checksum="A", archive_checksum="B", archive_filename="0000001.pdf") + doc2 = Document.objects.create(mime_type="application/pdf", title="0000001", filename="0000001.pdf", checksum="C") + + self.assertTrue(os.path.isfile(doc1.source_path)) + self.assertTrue(os.path.isfile(doc1.archive_path)) + self.assertTrue(os.path.isfile(doc2.source_path)) + + doc2.delete() + + self.assertTrue(os.path.isfile(doc1.source_path)) + self.assertTrue(os.path.isfile(doc1.archive_path)) + self.assertFalse(os.path.isfile(doc2.source_path)) + @override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}") def test_database_error(self): @@ -521,7 +663,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): archive = os.path.join(settings.ARCHIVE_DIR, "0000001.pdf") Path(original).touch() Path(archive).touch() - doc = Document(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_checksum="B") + doc = Document(mime_type="application/pdf", title="my_doc", filename="0000001.pdf", checksum="A", archive_filename="0000001.pdf", archive_checksum="B") with mock.patch("documents.signals.handlers.Document.objects.filter") as m: m.side_effect = DatabaseError() doc.save() @@ -531,6 +673,7 @@ class TestFileHandlingWithArchive(DirectoriesMixin, TestCase): self.assertTrue(os.path.isfile(doc.source_path)) self.assertTrue(os.path.isfile(doc.archive_path)) + class TestFilenameGeneration(TestCase): @override_settings( @@ -548,13 +691,13 @@ class TestFilenameGeneration(TestCase): PAPERLESS_FILENAME_FORMAT="{created}" ) def test_date(self): - doc = Document.objects.create(title="does not matter", created=datetime.datetime(2020,5,21, 7,36,51, 153), mime_type="application/pdf", pk=2, checksum="2") + doc = Document.objects.create(title="does not matter", created=timezone.make_aware(datetime.datetime(2020,5,21, 7,36,51, 153)), mime_type="application/pdf", pk=2, checksum="2") self.assertEqual(generate_filename(doc), "2020-05-21.pdf") def run(): doc = Document.objects.create(checksum=str(uuid.uuid4()), title=str(uuid.uuid4()), content="wow") - doc.filename = generate_unique_filename(doc, settings.ORIGINALS_DIR) + doc.filename = generate_unique_filename(doc) Path(doc.thumbnail_path).touch() with open(doc.source_path, "w") as f: f.write(str(uuid.uuid4())) diff --git a/src/documents/tests/test_index.py b/src/documents/tests/test_index.py index 830fca0e0..14304ab28 100644 --- a/src/documents/tests/test_index.py +++ b/src/documents/tests/test_index.py @@ -1,14 +1,25 @@ from django.test import TestCase -from documents.index import JsonFormatter +from documents import index +from documents.models import Document +from documents.tests.utils import DirectoriesMixin -class JsonFormatterTest(TestCase): +class TestAutoComplete(DirectoriesMixin, TestCase): - def setUp(self) -> None: - self.formatter = JsonFormatter() + def test_auto_complete(self): - def test_empty_fragments(self): - self.assertListEqual(self.formatter.format([]), []) + doc1 = Document.objects.create(title="doc1", checksum="A", content="test test2 test3") + doc2 = Document.objects.create(title="doc2", checksum="B", content="test test2") + doc3 = Document.objects.create(title="doc3", checksum="C", content="test2") + index.add_or_update_document(doc1) + index.add_or_update_document(doc2) + index.add_or_update_document(doc3) + ix = index.open_index() + + self.assertListEqual(index.autocomplete(ix, "tes"), [b"test3", b"test", b"test2"]) + self.assertListEqual(index.autocomplete(ix, "tes", limit=3), [b"test3", b"test", b"test2"]) + self.assertListEqual(index.autocomplete(ix, "tes", limit=1), [b"test3"]) + self.assertListEqual(index.autocomplete(ix, "tes", limit=0), []) diff --git a/src/documents/tests/test_logger.py b/src/documents/tests/test_logger.py deleted file mode 100644 index bbc9c2b5d..000000000 --- a/src/documents/tests/test_logger.py +++ /dev/null @@ -1,66 +0,0 @@ -import logging -import uuid -from unittest import mock - -from django.test import TestCase, override_settings - -from ..models import Log - - -class TestPaperlessLog(TestCase): - - def __init__(self, *args, **kwargs): - TestCase.__init__(self, *args, **kwargs) - self.logger = logging.getLogger( - "documents.management.commands.document_consumer") - - @override_settings(DISABLE_DBHANDLER=False) - def test_that_it_saves_at_all(self): - - kw = {"group": uuid.uuid4()} - - self.assertEqual(Log.objects.all().count(), 0) - - with mock.patch("logging.StreamHandler.emit") as __: - - # Debug messages are ignored by default - self.logger.debug("This is a debugging message", extra=kw) - self.assertEqual(Log.objects.all().count(), 1) - - self.logger.info("This is an informational message", extra=kw) - self.assertEqual(Log.objects.all().count(), 2) - - self.logger.warning("This is an warning message", extra=kw) - self.assertEqual(Log.objects.all().count(), 3) - - self.logger.error("This is an error message", extra=kw) - self.assertEqual(Log.objects.all().count(), 4) - - self.logger.critical("This is a critical message", extra=kw) - self.assertEqual(Log.objects.all().count(), 5) - - @override_settings(DISABLE_DBHANDLER=False) - def test_groups(self): - - kw1 = {"group": uuid.uuid4()} - kw2 = {"group": uuid.uuid4()} - - self.assertEqual(Log.objects.all().count(), 0) - - with mock.patch("logging.StreamHandler.emit") as __: - - self.logger.info("This is an informational message", extra=kw2) - self.assertEqual(Log.objects.all().count(), 1) - self.assertEqual(Log.objects.filter(group=kw2["group"]).count(), 1) - - self.logger.warning("This is an warning message", extra=kw1) - self.assertEqual(Log.objects.all().count(), 2) - self.assertEqual(Log.objects.filter(group=kw1["group"]).count(), 1) - - self.logger.error("This is an error message", extra=kw2) - self.assertEqual(Log.objects.all().count(), 3) - self.assertEqual(Log.objects.filter(group=kw2["group"]).count(), 2) - - self.logger.critical("This is a critical message", extra=kw1) - self.assertEqual(Log.objects.all().count(), 4) - self.assertEqual(Log.objects.filter(group=kw1["group"]).count(), 2) diff --git a/src/documents/tests/test_management.py b/src/documents/tests/test_management.py new file mode 100644 index 000000000..f7beb8907 --- /dev/null +++ b/src/documents/tests/test_management.py @@ -0,0 +1,193 @@ +import hashlib +import tempfile +import filecmp +import os +import shutil +from pathlib import Path +from unittest import mock + +from django.test import TestCase, override_settings + + +from django.core.management import call_command + +from documents.file_handling import generate_filename +from documents.management.commands.document_archiver import handle_document +from documents.models import Document +from documents.tests.utils import DirectoriesMixin + + +sample_file = os.path.join(os.path.dirname(__file__), "samples", "simple.pdf") + + +@override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}") +class TestArchiver(DirectoriesMixin, TestCase): + + def make_models(self): + return Document.objects.create(checksum="A", title="A", content="first document", mime_type="application/pdf") + + def test_archiver(self): + + doc = self.make_models() + shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"{doc.id:07}.pdf")) + + call_command('document_archiver') + + def test_handle_document(self): + + doc = self.make_models() + shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"{doc.id:07}.pdf")) + + handle_document(doc.pk) + + doc = Document.objects.get(id=doc.id) + + self.assertIsNotNone(doc.checksum) + self.assertIsNotNone(doc.archive_checksum) + self.assertTrue(os.path.isfile(doc.archive_path)) + self.assertTrue(os.path.isfile(doc.source_path)) + self.assertTrue(filecmp.cmp(sample_file, doc.source_path)) + self.assertEqual(doc.archive_filename, "none/A.pdf") + + def test_unknown_mime_type(self): + doc = self.make_models() + doc.mime_type = "sdgfh" + doc.save() + shutil.copy(sample_file, doc.source_path) + + handle_document(doc.pk) + + doc = Document.objects.get(id=doc.id) + + self.assertIsNotNone(doc.checksum) + self.assertIsNone(doc.archive_checksum) + self.assertIsNone(doc.archive_filename) + self.assertTrue(os.path.isfile(doc.source_path)) + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}") + def test_naming_priorities(self): + doc1 = Document.objects.create(checksum="A", title="document", content="first document", mime_type="application/pdf", filename="document.pdf") + doc2 = Document.objects.create(checksum="B", title="document", content="second document", mime_type="application/pdf", filename="document_01.pdf") + shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"document.pdf")) + shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"document_01.pdf")) + + handle_document(doc2.pk) + handle_document(doc1.pk) + + doc1 = Document.objects.get(id=doc1.id) + doc2 = Document.objects.get(id=doc2.id) + + self.assertEqual(doc1.archive_filename, "document.pdf") + self.assertEqual(doc2.archive_filename, "document_01.pdf") + + +class TestDecryptDocuments(TestCase): + + @override_settings( + ORIGINALS_DIR=os.path.join(os.path.dirname(__file__), "samples", "originals"), + THUMBNAIL_DIR=os.path.join(os.path.dirname(__file__), "samples", "thumb"), + PASSPHRASE="test", + PAPERLESS_FILENAME_FORMAT=None + ) + @mock.patch("documents.management.commands.decrypt_documents.input") + def test_decrypt(self, m): + + media_dir = tempfile.mkdtemp() + originals_dir = os.path.join(media_dir, "documents", "originals") + thumb_dir = os.path.join(media_dir, "documents", "thumbnails") + os.makedirs(originals_dir, exist_ok=True) + os.makedirs(thumb_dir, exist_ok=True) + + override_settings( + ORIGINALS_DIR=originals_dir, + THUMBNAIL_DIR=thumb_dir, + PASSPHRASE="test" + ).enable() + + doc = Document.objects.create(checksum="82186aaa94f0b98697d704b90fd1c072", title="wow", filename="0000004.pdf.gpg", mime_type="application/pdf", storage_type=Document.STORAGE_TYPE_GPG) + + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000004.pdf.gpg"), os.path.join(originals_dir, "0000004.pdf.gpg")) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", f"0000004.png.gpg"), os.path.join(thumb_dir, f"{doc.id:07}.png.gpg")) + + call_command('decrypt_documents') + + doc.refresh_from_db() + + self.assertEqual(doc.storage_type, Document.STORAGE_TYPE_UNENCRYPTED) + self.assertEqual(doc.filename, "0000004.pdf") + self.assertTrue(os.path.isfile(os.path.join(originals_dir, "0000004.pdf"))) + self.assertTrue(os.path.isfile(doc.source_path)) + self.assertTrue(os.path.isfile(os.path.join(thumb_dir, f"{doc.id:07}.png"))) + self.assertTrue(os.path.isfile(doc.thumbnail_path)) + + with doc.source_file as f: + checksum = hashlib.md5(f.read()).hexdigest() + self.assertEqual(checksum, doc.checksum) + + +class TestMakeIndex(TestCase): + + @mock.patch("documents.management.commands.document_index.index_reindex") + def test_reindex(self, m): + call_command("document_index", "reindex") + m.assert_called_once() + + @mock.patch("documents.management.commands.document_index.index_optimize") + def test_optimize(self, m): + call_command("document_index", "optimize") + m.assert_called_once() + + +class TestRenamer(DirectoriesMixin, TestCase): + + @override_settings(PAPERLESS_FILENAME_FORMAT="") + def test_rename(self): + doc = Document.objects.create(title="test", mime_type="image/jpeg") + doc.filename = generate_filename(doc) + doc.archive_filename = generate_filename(doc, archive_filename=True) + doc.save() + + Path(doc.source_path).touch() + Path(doc.archive_path).touch() + + with override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}"): + call_command("document_renamer") + + doc2 = Document.objects.get(id=doc.id) + + self.assertEqual(doc2.filename, "none/test.jpg") + self.assertEqual(doc2.archive_filename, "none/test.pdf") + self.assertFalse(os.path.isfile(doc.source_path)) + self.assertFalse(os.path.isfile(doc.archive_path)) + self.assertTrue(os.path.isfile(doc2.source_path)) + self.assertTrue(os.path.isfile(doc2.archive_path)) + + +class TestCreateClassifier(TestCase): + + @mock.patch("documents.management.commands.document_create_classifier.train_classifier") + def test_create_classifier(self, m): + call_command("document_create_classifier") + + m.assert_called_once() + + +class TestSanityChecker(DirectoriesMixin, TestCase): + + def test_no_issues(self): + with self.assertLogs() as capture: + call_command("document_sanity_checker") + + self.assertEqual(len(capture.output), 1) + self.assertIn("Sanity checker detected no issues.", capture.output[0]) + + def test_errors(self): + doc = Document.objects.create(title="test", content="test", filename="test.pdf", checksum="abc") + Path(doc.source_path).touch() + Path(doc.thumbnail_path).touch() + + with self.assertLogs() as capture: + call_command("document_sanity_checker") + + self.assertEqual(len(capture.output), 1) + self.assertIn("Checksum mismatch of document", capture.output[0]) diff --git a/src/documents/tests/test_management_archiver.py b/src/documents/tests/test_management_archiver.py deleted file mode 100644 index 0828f05ff..000000000 --- a/src/documents/tests/test_management_archiver.py +++ /dev/null @@ -1,40 +0,0 @@ -import filecmp -import os -import shutil - -from django.core.management import call_command -from django.test import TestCase - -from documents.management.commands.document_archiver import handle_document -from documents.models import Document -from documents.tests.utils import DirectoriesMixin - - -sample_file = os.path.join(os.path.dirname(__file__), "samples", "simple.pdf") - - -class TestArchiver(DirectoriesMixin, TestCase): - - def make_models(self): - return Document.objects.create(checksum="A", title="A", content="first document", mime_type="application/pdf") - - def test_archiver(self): - - doc = self.make_models() - shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"{doc.id:07}.pdf")) - - call_command('document_archiver') - - def test_handle_document(self): - - doc = self.make_models() - shutil.copy(sample_file, os.path.join(self.dirs.originals_dir, f"{doc.id:07}.pdf")) - - handle_document(doc.pk) - - doc = Document.objects.get(id=doc.id) - - self.assertIsNotNone(doc.checksum) - self.assertTrue(os.path.isfile(doc.archive_path)) - self.assertTrue(os.path.isfile(doc.source_path)) - self.assertTrue(filecmp.cmp(sample_file, doc.source_path)) diff --git a/src/documents/tests/test_management_consumer.py b/src/documents/tests/test_management_consumer.py index b6a61a167..377e8fc54 100644 --- a/src/documents/tests/test_management_consumer.py +++ b/src/documents/tests/test_management_consumer.py @@ -60,10 +60,10 @@ class ConsumerMixin: super(ConsumerMixin, self).tearDown() - def wait_for_task_mock_call(self): + def wait_for_task_mock_call(self, excpeted_call_count=1): n = 0 while n < 100: - if self.task_mock.call_count > 0: + if self.task_mock.call_count >= excpeted_call_count: # give task_mock some time to finish and raise errors sleep(1) return @@ -202,8 +202,44 @@ class TestConsumer(DirectoriesMixin, ConsumerMixin, TransactionTestCase): self.assertRaises(CommandError, call_command, 'document_consumer', '--oneshot') + def test_mac_write(self): + self.task_mock.side_effect = self.bogus_task -@override_settings(CONSUMER_POLLING=1) + self.t_start() + + shutil.copy(self.sample_file, os.path.join(self.dirs.consumption_dir, ".DS_STORE")) + shutil.copy(self.sample_file, os.path.join(self.dirs.consumption_dir, "my_file.pdf")) + shutil.copy(self.sample_file, os.path.join(self.dirs.consumption_dir, "._my_file.pdf")) + shutil.copy(self.sample_file, os.path.join(self.dirs.consumption_dir, "my_second_file.pdf")) + shutil.copy(self.sample_file, os.path.join(self.dirs.consumption_dir, "._my_second_file.pdf")) + + sleep(5) + + self.wait_for_task_mock_call(excpeted_call_count=2) + + self.assertEqual(2, self.task_mock.call_count) + + fnames = [os.path.basename(args[1]) for args, _ in self.task_mock.call_args_list] + self.assertCountEqual(fnames, ["my_file.pdf", "my_second_file.pdf"]) + + def test_is_ignored(self): + test_paths = [ + (os.path.join(self.dirs.consumption_dir, "foo.pdf"), False), + (os.path.join(self.dirs.consumption_dir, "foo","bar.pdf"), False), + (os.path.join(self.dirs.consumption_dir, ".DS_STORE", "foo.pdf"), True), + (os.path.join(self.dirs.consumption_dir, "foo", ".DS_STORE", "bar.pdf"), True), + (os.path.join(self.dirs.consumption_dir, ".stfolder", "foo.pdf"), True), + (os.path.join(self.dirs.consumption_dir, "._foo.pdf"), True), + (os.path.join(self.dirs.consumption_dir, "._foo", "bar.pdf"), False), + ] + for file_path, expected_ignored in test_paths: + self.assertEqual( + expected_ignored, + document_consumer._is_ignored(file_path), + f'_is_ignored("{file_path}") != {expected_ignored}') + + +@override_settings(CONSUMER_POLLING=1, CONSUMER_POLLING_DELAY=1, CONSUMER_POLLING_RETRY_COUNT=20) class TestConsumerPolling(TestConsumer): # just do all the tests with polling pass @@ -215,8 +251,7 @@ class TestConsumerRecursive(TestConsumer): pass -@override_settings(CONSUMER_RECURSIVE=True) -@override_settings(CONSUMER_POLLING=1) +@override_settings(CONSUMER_RECURSIVE=True, CONSUMER_POLLING=1, CONSUMER_POLLING_DELAY=1, CONSUMER_POLLING_RETRY_COUNT=20) class TestConsumerRecursivePolling(TestConsumer): # just do all the tests with polling and recursive pass @@ -257,6 +292,6 @@ class TestConsumerTags(DirectoriesMixin, ConsumerMixin, TransactionTestCase): # their order. self.assertCountEqual(kwargs["override_tag_ids"], tag_ids) - @override_settings(CONSUMER_POLLING=1) + @override_settings(CONSUMER_POLLING=1, CONSUMER_POLLING_DELAY=1, CONSUMER_POLLING_RETRY_COUNT=20) def test_consume_file_with_path_tags_polling(self): self.test_consume_file_with_path_tags() diff --git a/src/documents/tests/test_management_decrypt.py b/src/documents/tests/test_management_decrypt.py deleted file mode 100644 index 1d64b1105..000000000 --- a/src/documents/tests/test_management_decrypt.py +++ /dev/null @@ -1,57 +0,0 @@ -import hashlib -import json -import os -import shutil -import tempfile -from unittest import mock - -from django.core.management import call_command -from django.test import TestCase, override_settings - -from documents.management.commands import document_exporter -from documents.models import Document, Tag, DocumentType, Correspondent - - -class TestDecryptDocuments(TestCase): - - @override_settings( - ORIGINALS_DIR=os.path.join(os.path.dirname(__file__), "samples", "originals"), - THUMBNAIL_DIR=os.path.join(os.path.dirname(__file__), "samples", "thumb"), - PASSPHRASE="test", - PAPERLESS_FILENAME_FORMAT=None - ) - @mock.patch("documents.management.commands.decrypt_documents.input") - def test_decrypt(self, m): - - media_dir = tempfile.mkdtemp() - originals_dir = os.path.join(media_dir, "documents", "originals") - thumb_dir = os.path.join(media_dir, "documents", "thumbnails") - os.makedirs(originals_dir, exist_ok=True) - os.makedirs(thumb_dir, exist_ok=True) - - override_settings( - ORIGINALS_DIR=originals_dir, - THUMBNAIL_DIR=thumb_dir, - PASSPHRASE="test" - ).enable() - - doc = Document.objects.create(checksum="9c9691e51741c1f4f41a20896af31770", title="wow", filename="0000002.pdf.gpg", mime_type="application/pdf", storage_type=Document.STORAGE_TYPE_GPG) - - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000002.pdf.gpg"), os.path.join(originals_dir, "0000002.pdf.gpg")) - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", f"0000002.png.gpg"), os.path.join(thumb_dir, f"{doc.id:07}.png.gpg")) - - call_command('decrypt_documents') - - doc.refresh_from_db() - - self.assertEqual(doc.storage_type, Document.STORAGE_TYPE_UNENCRYPTED) - self.assertEqual(doc.filename, "0000002.pdf") - self.assertTrue(os.path.isfile(os.path.join(originals_dir, "0000002.pdf"))) - self.assertTrue(os.path.isfile(doc.source_path)) - self.assertTrue(os.path.isfile(os.path.join(thumb_dir, f"{doc.id:07}.png"))) - self.assertTrue(os.path.isfile(doc.thumbnail_path)) - - with doc.source_file as f: - checksum = hashlib.md5(f.read()).hexdigest() - self.assertEqual(checksum, doc.checksum) - diff --git a/src/documents/tests/test_management_exporter.py b/src/documents/tests/test_management_exporter.py index 22d6fc7f6..9e2dd0804 100644 --- a/src/documents/tests/test_management_exporter.py +++ b/src/documents/tests/test_management_exporter.py @@ -3,6 +3,8 @@ import json import os import shutil import tempfile +from pathlib import Path +from unittest import mock from django.core.management import call_command from django.test import TestCase, override_settings @@ -10,48 +12,87 @@ from django.test import TestCase, override_settings from documents.management.commands import document_exporter from documents.models import Document, Tag, DocumentType, Correspondent from documents.sanity_checker import check_sanity +from documents.settings import EXPORTER_FILE_NAME from documents.tests.utils import DirectoriesMixin, paperless_environment class TestExportImport(DirectoriesMixin, TestCase): + def setUp(self) -> None: + self.target = tempfile.mkdtemp() + self.addCleanup(shutil.rmtree, self.target) + + self.d1 = Document.objects.create(content="Content", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", title="wow1", filename="0000001.pdf", mime_type="application/pdf", archive_filename="0000001.pdf") + self.d2 = Document.objects.create(content="Content", checksum="9c9691e51741c1f4f41a20896af31770", title="wow2", filename="0000002.pdf", mime_type="application/pdf") + self.d3 = Document.objects.create(content="Content", checksum="d38d7ed02e988e072caf924e0f3fcb76", title="wow2", filename="0000003.pdf", mime_type="application/pdf") + self.d4 = Document.objects.create(content="Content", checksum="82186aaa94f0b98697d704b90fd1c072", title="wow_dec", filename="0000004.pdf.gpg", mime_type="application/pdf", storage_type=Document.STORAGE_TYPE_GPG) + + self.t1 = Tag.objects.create(name="t") + self.dt1 = DocumentType.objects.create(name="dt") + self.c1 = Correspondent.objects.create(name="c") + + self.d1.tags.add(self.t1) + self.d1.correspondent = self.c1 + self.d1.document_type = self.dt1 + self.d1.save() + super(TestExportImport, self).setUp() + + def _get_document_from_manifest(self, manifest, id): + f = list(filter(lambda d: d['model'] == "documents.document" and d['pk'] == id, manifest)) + if len(f) == 1: + return f[0] + else: + raise ValueError(f"document with id {id} does not exist in manifest") + @override_settings( PASSPHRASE="test" ) - def test_exporter(self): + def _do_export(self, use_filename_format=False, compare_checksums=False, delete=False): + args = ['document_exporter', self.target] + if use_filename_format: + args += ["--use-filename-format"] + if compare_checksums: + args += ["--compare-checksums"] + if delete: + args += ["--delete"] + + call_command(*args) + + with open(os.path.join(self.target, "manifest.json")) as f: + manifest = json.load(f) + + return manifest + + def test_exporter(self, use_filename_format=False): shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) - file = os.path.join(self.dirs.originals_dir, "0000001.pdf") + manifest = self._do_export(use_filename_format=use_filename_format) - Document.objects.create(content="Content", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", title="wow", filename="0000001.pdf", mime_type="application/pdf") - Document.objects.create(content="Content", checksum="9c9691e51741c1f4f41a20896af31770", title="wow", filename="0000002.pdf.gpg", mime_type="application/pdf", storage_type=Document.STORAGE_TYPE_GPG) - Tag.objects.create(name="t") - DocumentType.objects.create(name="dt") - Correspondent.objects.create(name="c") + self.assertEqual(len(manifest), 8) + self.assertEqual(len(list(filter(lambda e: e['model'] == 'documents.document', manifest))), 4) - target = tempfile.mkdtemp() - self.addCleanup(shutil.rmtree, target) + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) - call_command('document_exporter', target) - - with open(os.path.join(target, "manifest.json")) as f: - manifest = json.load(f) - - self.assertEqual(len(manifest), 5) + self.assertEqual(self._get_document_from_manifest(manifest, self.d1.id)['fields']['title'], "wow1") + self.assertEqual(self._get_document_from_manifest(manifest, self.d2.id)['fields']['title'], "wow2") + self.assertEqual(self._get_document_from_manifest(manifest, self.d3.id)['fields']['title'], "wow2") + self.assertEqual(self._get_document_from_manifest(manifest, self.d4.id)['fields']['title'], "wow_dec") for element in manifest: if element['model'] == 'documents.document': - fname = os.path.join(target, element[document_exporter.EXPORTER_FILE_NAME]) + fname = os.path.join(self.target, element[document_exporter.EXPORTER_FILE_NAME]) self.assertTrue(os.path.exists(fname)) - self.assertTrue(os.path.exists(os.path.join(target, element[document_exporter.EXPORTER_THUMBNAIL_NAME]))) + self.assertTrue(os.path.exists(os.path.join(self.target, element[document_exporter.EXPORTER_THUMBNAIL_NAME]))) with open(fname, "rb") as f: checksum = hashlib.md5(f.read()).hexdigest() self.assertEqual(checksum, element['fields']['checksum']) + self.assertEqual(element['fields']['storage_type'], Document.STORAGE_TYPE_UNENCRYPTED) + if document_exporter.EXPORTER_ARCHIVE_NAME in element: - fname = os.path.join(target, element[document_exporter.EXPORTER_ARCHIVE_NAME]) + fname = os.path.join(self.target, element[document_exporter.EXPORTER_ARCHIVE_NAME]) self.assertTrue(os.path.exists(fname)) with open(fname, "rb") as f: @@ -59,11 +100,124 @@ class TestExportImport(DirectoriesMixin, TestCase): self.assertEqual(checksum, element['fields']['archive_checksum']) with paperless_environment() as dirs: - call_command('document_importer', target) + self.assertEqual(Document.objects.count(), 4) + Document.objects.all().delete() + Correspondent.objects.all().delete() + DocumentType.objects.all().delete() + Tag.objects.all().delete() + self.assertEqual(Document.objects.count(), 0) + + call_command('document_importer', self.target) + self.assertEqual(Document.objects.count(), 4) + self.assertEqual(Tag.objects.count(), 1) + self.assertEqual(Correspondent.objects.count(), 1) + self.assertEqual(DocumentType.objects.count(), 1) + self.assertEqual(Document.objects.get(id=self.d1.id).title, "wow1") + self.assertEqual(Document.objects.get(id=self.d2.id).title, "wow2") + self.assertEqual(Document.objects.get(id=self.d3.id).title, "wow2") + self.assertEqual(Document.objects.get(id=self.d4.id).title, "wow_dec") messages = check_sanity() # everything is alright after the test self.assertEqual(len(messages), 0, str([str(m) for m in messages])) + def test_exporter_with_filename_format(self): + shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) + shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) + + with override_settings(PAPERLESS_FILENAME_FORMAT="{created_year}/{correspondent}/{title}"): + self.test_exporter(use_filename_format=True) + + def test_update_export_changed_time(self): + shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) + shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) + + self._do_export() + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + st_mtime_1 = os.stat(os.path.join(self.target, "manifest.json")).st_mtime + + with mock.patch("documents.management.commands.document_exporter.shutil.copy2") as m: + self._do_export() + m.assert_not_called() + + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + st_mtime_2 = os.stat(os.path.join(self.target, "manifest.json")).st_mtime + + Path(self.d1.source_path).touch() + + with mock.patch("documents.management.commands.document_exporter.shutil.copy2") as m: + self._do_export() + self.assertEqual(m.call_count, 1) + + st_mtime_3 = os.stat(os.path.join(self.target, "manifest.json")).st_mtime + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + self.assertNotEqual(st_mtime_1, st_mtime_2) + self.assertNotEqual(st_mtime_2, st_mtime_3) + + def test_update_export_changed_checksum(self): + shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) + shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) + + self._do_export() + + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + with mock.patch("documents.management.commands.document_exporter.shutil.copy2") as m: + self._do_export() + m.assert_not_called() + + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + self.d2.checksum = "asdfasdgf3" + self.d2.save() + + with mock.patch("documents.management.commands.document_exporter.shutil.copy2") as m: + self._do_export(compare_checksums=True) + self.assertEqual(m.call_count, 1) + + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + def test_update_export_deleted_document(self): + shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) + shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) + + manifest = self._do_export() + + self.assertTrue(len(manifest), 7) + doc_from_manifest = self._get_document_from_manifest(manifest, self.d3.id) + self.assertTrue(os.path.isfile(os.path.join(self.target, doc_from_manifest[EXPORTER_FILE_NAME]))) + self.d3.delete() + + manifest = self._do_export() + self.assertRaises(ValueError, self._get_document_from_manifest, manifest, self.d3.id) + self.assertTrue(os.path.isfile(os.path.join(self.target, doc_from_manifest[EXPORTER_FILE_NAME]))) + + manifest = self._do_export(delete=True) + self.assertFalse(os.path.isfile(os.path.join(self.target, doc_from_manifest[EXPORTER_FILE_NAME]))) + + self.assertTrue(len(manifest), 6) + + @override_settings(PAPERLESS_FILENAME_FORMAT="{title}/{correspondent}") + def test_update_export_changed_location(self): + shutil.rmtree(os.path.join(self.dirs.media_dir, "documents")) + shutil.copytree(os.path.join(os.path.dirname(__file__), "samples", "documents"), os.path.join(self.dirs.media_dir, "documents")) + + m = self._do_export(use_filename_format=True) + self.assertTrue(os.path.isfile(os.path.join(self.target, "wow1", "c.pdf"))) + + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + + self.d1.title = "new_title" + self.d1.save() + self._do_export(use_filename_format=True, delete=True) + self.assertFalse(os.path.isfile(os.path.join(self.target, "wow1", "c.pdf"))) + self.assertFalse(os.path.isdir(os.path.join(self.target, "wow1"))) + self.assertTrue(os.path.isfile(os.path.join(self.target, "new_title", "c.pdf"))) + self.assertTrue(os.path.exists(os.path.join(self.target, "manifest.json"))) + self.assertTrue(os.path.isfile(os.path.join(self.target, "wow2", "none.pdf"))) + self.assertTrue(os.path.isfile(os.path.join(self.target, "wow2", "none_01.pdf"))) + def test_export_missing_files(self): target = tempfile.mkdtemp() diff --git a/src/documents/tests/test_management_retagger.py b/src/documents/tests/test_management_retagger.py index 907a23d09..39e9c80b7 100644 --- a/src/documents/tests/test_management_retagger.py +++ b/src/documents/tests/test_management_retagger.py @@ -11,14 +11,17 @@ class TestRetagger(DirectoriesMixin, TestCase): self.d1 = Document.objects.create(checksum="A", title="A", content="first document") self.d2 = Document.objects.create(checksum="B", title="B", content="second document") self.d3 = Document.objects.create(checksum="C", title="C", content="unrelated document") + self.d4 = Document.objects.create(checksum="D", title="D", content="auto document") self.tag_first = Tag.objects.create(name="tag1", match="first", matching_algorithm=Tag.MATCH_ANY) self.tag_second = Tag.objects.create(name="tag2", match="second", matching_algorithm=Tag.MATCH_ANY) self.tag_inbox = Tag.objects.create(name="test", is_inbox_tag=True) self.tag_no_match = Tag.objects.create(name="test2") + self.tag_auto = Tag.objects.create(name="tagauto", matching_algorithm=Tag.MATCH_AUTO) self.d3.tags.add(self.tag_inbox) self.d3.tags.add(self.tag_no_match) + self.d4.tags.add(self.tag_auto) self.correspondent_first = Correspondent.objects.create( @@ -32,7 +35,8 @@ class TestRetagger(DirectoriesMixin, TestCase): name="dt2", match="second", matching_algorithm=DocumentType.MATCH_ANY) def get_updated_docs(self): - return Document.objects.get(title="A"), Document.objects.get(title="B"), Document.objects.get(title="C") + return Document.objects.get(title="A"), Document.objects.get(title="B"), \ + Document.objects.get(title="C"), Document.objects.get(title="D") def setUp(self) -> None: super(TestRetagger, self).setUp() @@ -40,25 +44,26 @@ class TestRetagger(DirectoriesMixin, TestCase): def test_add_tags(self): call_command('document_retagger', '--tags') - d_first, d_second, d_unrelated = self.get_updated_docs() + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() self.assertEqual(d_first.tags.count(), 1) self.assertEqual(d_second.tags.count(), 1) self.assertEqual(d_unrelated.tags.count(), 2) + self.assertEqual(d_auto.tags.count(), 1) self.assertEqual(d_first.tags.first(), self.tag_first) self.assertEqual(d_second.tags.first(), self.tag_second) def test_add_type(self): call_command('document_retagger', '--document_type') - d_first, d_second, d_unrelated = self.get_updated_docs() + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() self.assertEqual(d_first.document_type, self.doctype_first) self.assertEqual(d_second.document_type, self.doctype_second) def test_add_correspondent(self): call_command('document_retagger', '--correspondent') - d_first, d_second, d_unrelated = self.get_updated_docs() + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() self.assertEqual(d_first.correspondent, self.correspondent_first) self.assertEqual(d_second.correspondent, self.correspondent_second) @@ -68,11 +73,55 @@ class TestRetagger(DirectoriesMixin, TestCase): call_command('document_retagger', '--tags', '--overwrite') - d_first, d_second, d_unrelated = self.get_updated_docs() + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() self.assertIsNotNone(Tag.objects.get(id=self.tag_second.id)) self.assertCountEqual([tag.id for tag in d_first.tags.all()], [self.tag_first.id]) self.assertCountEqual([tag.id for tag in d_second.tags.all()], [self.tag_second.id]) self.assertCountEqual([tag.id for tag in d_unrelated.tags.all()], [self.tag_inbox.id, self.tag_no_match.id]) + self.assertEqual(d_auto.tags.count(), 0) + def test_add_tags_suggest(self): + call_command('document_retagger', '--tags', '--suggest') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.tags.count(), 0) + self.assertEqual(d_second.tags.count(), 0) + self.assertEqual(d_auto.tags.count(), 1) + + def test_add_type_suggest(self): + call_command('document_retagger', '--document_type', '--suggest') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.document_type, None) + self.assertEqual(d_second.document_type, None) + + def test_add_correspondent_suggest(self): + call_command('document_retagger', '--correspondent', '--suggest') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.correspondent, None) + self.assertEqual(d_second.correspondent, None) + + def test_add_tags_suggest_url(self): + call_command('document_retagger', '--tags', '--suggest', '--base-url=http://localhost') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.tags.count(), 0) + self.assertEqual(d_second.tags.count(), 0) + self.assertEqual(d_auto.tags.count(), 1) + + def test_add_type_suggest_url(self): + call_command('document_retagger', '--document_type', '--suggest', '--base-url=http://localhost') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.document_type, None) + self.assertEqual(d_second.document_type, None) + + def test_add_correspondent_suggest_url(self): + call_command('document_retagger', '--correspondent', '--suggest', '--base-url=http://localhost') + d_first, d_second, d_unrelated, d_auto = self.get_updated_docs() + + self.assertEqual(d_first.correspondent, None) + self.assertEqual(d_second.correspondent, None) diff --git a/src/documents/tests/test_management_superuser.py b/src/documents/tests/test_management_superuser.py new file mode 100644 index 000000000..ca28db89c --- /dev/null +++ b/src/documents/tests/test_management_superuser.py @@ -0,0 +1,66 @@ +import os +import shutil +from unittest import mock + +from django.contrib.auth.models import User +from django.core.management import call_command +from django.test import TestCase + +from documents.management.commands.document_thumbnails import _process_document +from documents.models import Document, Tag, Correspondent, DocumentType +from documents.tests.utils import DirectoriesMixin + + +class TestManageSuperUser(DirectoriesMixin, TestCase): + + def reset_environment(self): + if "PAPERLESS_ADMIN_USER" in os.environ: + del os.environ["PAPERLESS_ADMIN_USER"] + if "PAPERLESS_ADMIN_PASSWORD" in os.environ: + del os.environ["PAPERLESS_ADMIN_PASSWORD"] + + def setUp(self) -> None: + super().setUp() + self.reset_environment() + + def tearDown(self) -> None: + super().tearDown() + self.reset_environment() + + def test_no_user(self): + call_command("manage_superuser") + + # just the consumer user. + self.assertEqual(User.objects.count(), 1) + self.assertTrue(User.objects.filter(username="consumer").exists()) + + def test_create(self): + os.environ["PAPERLESS_ADMIN_USER"] = "new_user" + os.environ["PAPERLESS_ADMIN_PASSWORD"] = "123456" + + call_command("manage_superuser") + + user: User = User.objects.get_by_natural_key("new_user") + self.assertTrue(user.check_password("123456")) + + def test_update(self): + os.environ["PAPERLESS_ADMIN_USER"] = "new_user" + os.environ["PAPERLESS_ADMIN_PASSWORD"] = "123456" + + call_command("manage_superuser") + + os.environ["PAPERLESS_ADMIN_USER"] = "new_user" + os.environ["PAPERLESS_ADMIN_PASSWORD"] = "more_secure_pwd_7645" + + call_command("manage_superuser") + + user: User = User.objects.get_by_natural_key("new_user") + self.assertTrue(user.check_password("more_secure_pwd_7645")) + + def test_no_password(self): + os.environ["PAPERLESS_ADMIN_USER"] = "new_user" + + call_command("manage_superuser") + + with self.assertRaises(User.DoesNotExist): + User.objects.get_by_natural_key("new_user") diff --git a/src/documents/tests/test_management_thumbnails.py b/src/documents/tests/test_management_thumbnails.py new file mode 100644 index 000000000..7ecdf0489 --- /dev/null +++ b/src/documents/tests/test_management_thumbnails.py @@ -0,0 +1,52 @@ +import os +import shutil +from unittest import mock + +from django.core.management import call_command +from django.test import TestCase + +from documents.management.commands.document_thumbnails import _process_document +from documents.models import Document, Tag, Correspondent, DocumentType +from documents.tests.utils import DirectoriesMixin + + +class TestMakeThumbnails(DirectoriesMixin, TestCase): + + def make_models(self): + self.d1 = Document.objects.create(checksum="A", title="A", content="first document", mime_type="application/pdf", filename="test.pdf") + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), self.d1.source_path) + + self.d2 = Document.objects.create(checksum="Ass", title="A", content="first document", mime_type="application/pdf", filename="test2.pdf") + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), self.d2.source_path) + + def setUp(self) -> None: + super(TestMakeThumbnails, self).setUp() + self.make_models() + + def test_process_document(self): + self.assertFalse(os.path.isfile(self.d1.thumbnail_path)) + _process_document(self.d1.id) + self.assertTrue(os.path.isfile(self.d1.thumbnail_path)) + + @mock.patch("documents.management.commands.document_thumbnails.shutil.move") + def test_process_document_invalid_mime_type(self, m): + self.d1.mime_type = "asdasdasd" + self.d1.save() + + _process_document(self.d1.id) + + m.assert_not_called() + + def test_command(self): + self.assertFalse(os.path.isfile(self.d1.thumbnail_path)) + self.assertFalse(os.path.isfile(self.d2.thumbnail_path)) + call_command('document_thumbnails') + self.assertTrue(os.path.isfile(self.d1.thumbnail_path)) + self.assertTrue(os.path.isfile(self.d2.thumbnail_path)) + + def test_command_documentid(self): + self.assertFalse(os.path.isfile(self.d1.thumbnail_path)) + self.assertFalse(os.path.isfile(self.d2.thumbnail_path)) + call_command('document_thumbnails', '-d', f"{self.d1.id}") + self.assertTrue(os.path.isfile(self.d1.thumbnail_path)) + self.assertFalse(os.path.isfile(self.d2.thumbnail_path)) diff --git a/src/documents/tests/test_matchables.py b/src/documents/tests/test_matchables.py index 4e4a3e7dc..da0fa66ea 100644 --- a/src/documents/tests/test_matchables.py +++ b/src/documents/tests/test_matchables.py @@ -21,13 +21,15 @@ class TestMatching(TestCase): matching_algorithm=getattr(klass, algorithm) ) for string in true: + doc = Document(content=string) self.assertTrue( - matching.matches(instance, string), + matching.matches(instance, doc), '"%s" should match "%s" but it does not' % (text, string) ) for string in false: + doc = Document(content=string) self.assertFalse( - matching.matches(instance, string), + matching.matches(instance, doc), '"%s" should not match "%s" but it does' % (text, string) ) @@ -169,7 +171,7 @@ class TestMatching(TestCase): def test_match_regex(self): self._test_matching( - r"alpha\w+gamma", + "alpha\w+gamma", "MATCH_REGEX", ( "I have alpha_and_gamma in me", @@ -187,6 +189,16 @@ class TestMatching(TestCase): ) ) + def test_tach_invalid_regex(self): + self._test_matching( + "[[", + "MATCH_REGEX", + [], + [ + "Don't match this" + ] + ) + def test_match_fuzzy(self): self._test_matching( diff --git a/src/documents/tests/test_migration_archive_files.py b/src/documents/tests/test_migration_archive_files.py new file mode 100644 index 000000000..6217ae05f --- /dev/null +++ b/src/documents/tests/test_migration_archive_files.py @@ -0,0 +1,325 @@ +import hashlib +import os +import shutil +from pathlib import Path +from unittest import mock + +from django.conf import settings +from django.test import override_settings + +from documents.parsers import ParseError +from documents.tests.utils import DirectoriesMixin, TestMigrations + + +STORAGE_TYPE_GPG = "gpg" + + +def archive_name_from_filename(filename): + return os.path.splitext(filename)[0] + ".pdf" + + +def archive_path_old(self): + if self.filename: + fname = archive_name_from_filename(self.filename) + else: + fname = "{:07}.pdf".format(self.pk) + + return os.path.join( + settings.ARCHIVE_DIR, + fname + ) + + +def archive_path_new(doc): + if doc.archive_filename is not None: + return os.path.join( + settings.ARCHIVE_DIR, + str(doc.archive_filename) + ) + else: + return None + + +def source_path(doc): + if doc.filename: + fname = str(doc.filename) + else: + fname = "{:07}{}".format(doc.pk, doc.file_type) + if doc.storage_type == STORAGE_TYPE_GPG: + fname += ".gpg" # pragma: no cover + + return os.path.join( + settings.ORIGINALS_DIR, + fname + ) + + +def thumbnail_path(doc): + file_name = "{:07}.png".format(doc.pk) + if doc.storage_type == STORAGE_TYPE_GPG: + file_name += ".gpg" + + return os.path.join( + settings.THUMBNAIL_DIR, + file_name + ) + + +def make_test_document(document_class, title: str, mime_type: str, original: str, original_filename: str, archive: str = None, archive_filename: str = None): + doc = document_class() + doc.filename = original_filename + doc.title = title + doc.mime_type = mime_type + doc.content = "the content, does not matter for this test" + doc.save() + + shutil.copy2(original, source_path(doc)) + with open(original, "rb") as f: + doc.checksum = hashlib.md5(f.read()).hexdigest() + + if archive: + if archive_filename: + doc.archive_filename = archive_filename + shutil.copy2(archive, archive_path_new(doc)) + else: + shutil.copy2(archive, archive_path_old(doc)) + + with open(archive, "rb") as f: + doc.archive_checksum = hashlib.md5(f.read()).hexdigest() + + doc.save() + + Path(thumbnail_path(doc)).touch() + + return doc + + +simple_jpg = os.path.join(os.path.dirname(__file__), "samples", "simple.jpg") +simple_pdf = os.path.join(os.path.dirname(__file__), "samples", "simple.pdf") +simple_pdf2 = os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000002.pdf") +simple_pdf3 = os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000003.pdf") +simple_txt = os.path.join(os.path.dirname(__file__), "samples", "simple.txt") +simple_png = os.path.join(os.path.dirname(__file__), "samples", "simple-noalpha.png") +simple_png2 = os.path.join(os.path.dirname(__file__), "examples", "no-text.png") + + +@override_settings(PAPERLESS_FILENAME_FORMAT="") +class TestMigrateArchiveFiles(DirectoriesMixin, TestMigrations): + + migrate_from = '1011_auto_20210101_2340' + migrate_to = '1012_fix_archive_files' + + def setUpBeforeMigration(self, apps): + Document = apps.get_model("documents", "Document") + + self.unrelated = make_test_document(Document, "unrelated", "application/pdf", simple_pdf3, "unrelated.pdf", simple_pdf) + self.no_text = make_test_document(Document, "no-text", "image/png", simple_png2, "no-text.png", simple_pdf) + self.doc_no_archive = make_test_document(Document, "no_archive", "text/plain", simple_txt, "no_archive.txt") + self.clash1 = make_test_document(Document, "clash", "application/pdf", simple_pdf, "clash.pdf", simple_pdf) + self.clash2 = make_test_document(Document, "clash", "image/jpeg", simple_jpg, "clash.jpg", simple_pdf) + self.clash3 = make_test_document(Document, "clash", "image/png", simple_png, "clash.png", simple_pdf) + self.clash4 = make_test_document(Document, "clash.png", "application/pdf", simple_pdf2, "clash.png.pdf", simple_pdf2) + + self.assertEqual(archive_path_old(self.clash1), archive_path_old(self.clash2)) + self.assertEqual(archive_path_old(self.clash1), archive_path_old(self.clash3)) + self.assertNotEqual(archive_path_old(self.clash1), archive_path_old(self.clash4)) + + def testArchiveFilesMigrated(self): + Document = self.apps.get_model('documents', 'Document') + + for doc in Document.objects.all(): + if doc.archive_checksum: + self.assertIsNotNone(doc.archive_filename) + self.assertTrue(os.path.isfile(archive_path_new(doc))) + else: + self.assertIsNone(doc.archive_filename) + + with open(source_path(doc), "rb") as f: + original_checksum = hashlib.md5(f.read()).hexdigest() + self.assertEqual(original_checksum, doc.checksum) + + if doc.archive_checksum: + self.assertTrue(os.path.isfile(archive_path_new(doc))) + with open(archive_path_new(doc), "rb") as f: + archive_checksum = hashlib.md5(f.read()).hexdigest() + self.assertEqual(archive_checksum, doc.archive_checksum) + + self.assertEqual(Document.objects.filter(archive_checksum__isnull=False).count(), 6) + + def test_filenames(self): + Document = self.apps.get_model('documents', 'Document') + self.assertEqual(Document.objects.get(id=self.unrelated.id).archive_filename, "unrelated.pdf") + self.assertEqual(Document.objects.get(id=self.no_text.id).archive_filename, "no-text.pdf") + self.assertEqual(Document.objects.get(id=self.doc_no_archive.id).archive_filename, None) + self.assertEqual(Document.objects.get(id=self.clash1.id).archive_filename, f"{self.clash1.id:07}.pdf") + self.assertEqual(Document.objects.get(id=self.clash2.id).archive_filename, f"{self.clash2.id:07}.pdf") + self.assertEqual(Document.objects.get(id=self.clash3.id).archive_filename, f"{self.clash3.id:07}.pdf") + self.assertEqual(Document.objects.get(id=self.clash4.id).archive_filename, "clash.png.pdf") + + +@override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}") +class TestMigrateArchiveFilesWithFilenameFormat(TestMigrateArchiveFiles): + + def test_filenames(self): + Document = self.apps.get_model('documents', 'Document') + self.assertEqual(Document.objects.get(id=self.unrelated.id).archive_filename, "unrelated.pdf") + self.assertEqual(Document.objects.get(id=self.no_text.id).archive_filename, "no-text.pdf") + self.assertEqual(Document.objects.get(id=self.doc_no_archive.id).archive_filename, None) + self.assertEqual(Document.objects.get(id=self.clash1.id).archive_filename, "none/clash.pdf") + self.assertEqual(Document.objects.get(id=self.clash2.id).archive_filename, "none/clash_01.pdf") + self.assertEqual(Document.objects.get(id=self.clash3.id).archive_filename, "none/clash_02.pdf") + self.assertEqual(Document.objects.get(id=self.clash4.id).archive_filename, "clash.png.pdf") + + +def fake_parse_wrapper(parser, path, mime_type, file_name): + parser.archive_path = None + parser.text = "the text" + + +@override_settings(PAPERLESS_FILENAME_FORMAT="") +class TestMigrateArchiveFilesErrors(DirectoriesMixin, TestMigrations): + + migrate_from = '1011_auto_20210101_2340' + migrate_to = '1012_fix_archive_files' + auto_migrate = False + + def test_archive_missing(self): + + Document = self.apps.get_model("documents", "Document") + + doc = make_test_document(Document, "clash", "application/pdf", simple_pdf, "clash.pdf", simple_pdf) + os.unlink(archive_path_old(doc)) + + self.assertRaisesMessage(ValueError, "does not exist at: ", self.performMigration) + + def test_parser_missing(self): + Document = self.apps.get_model("documents", "Document") + + doc1 = make_test_document(Document, "document", "invalid/typesss768", simple_png, "document.png", simple_pdf) + doc2 = make_test_document(Document, "document", "invalid/typesss768", simple_jpg, "document.jpg", simple_pdf) + + self.assertRaisesMessage(ValueError, "no parsers are available", self.performMigration) + + @mock.patch("documents.migrations.1012_fix_archive_files.parse_wrapper") + def test_parser_error(self, m): + m.side_effect = ParseError() + Document = self.apps.get_model("documents", "Document") + + doc1 = make_test_document(Document, "document", "image/png", simple_png, "document.png", simple_pdf) + doc2 = make_test_document(Document, "document", "application/pdf", simple_jpg, "document.jpg", simple_pdf) + + self.assertIsNotNone(doc1.archive_checksum) + self.assertIsNotNone(doc2.archive_checksum) + + with self.assertLogs() as capture: + self.performMigration() + + self.assertEqual(m.call_count, 6) + + self.assertEqual( + len(list(filter(lambda log: "Parse error, will try again in 5 seconds" in log, capture.output))), + 4) + + self.assertEqual( + len(list(filter(lambda log: "Unable to regenerate archive document for ID:" in log, capture.output))), + 2) + + Document = self.apps.get_model("documents", "Document") + + doc1 = Document.objects.get(id=doc1.id) + doc2 = Document.objects.get(id=doc2.id) + + self.assertIsNone(doc1.archive_checksum) + self.assertIsNone(doc2.archive_checksum) + self.assertIsNone(doc1.archive_filename) + self.assertIsNone(doc2.archive_filename) + + @mock.patch("documents.migrations.1012_fix_archive_files.parse_wrapper") + def test_parser_no_archive(self, m): + m.side_effect = fake_parse_wrapper + + Document = self.apps.get_model("documents", "Document") + + doc1 = make_test_document(Document, "document", "image/png", simple_png, "document.png", simple_pdf) + doc2 = make_test_document(Document, "document", "application/pdf", simple_jpg, "document.jpg", simple_pdf) + + with self.assertLogs() as capture: + self.performMigration() + + self.assertEqual( + len(list(filter(lambda log: "Parser did not return an archive document for document" in log, capture.output))), + 2) + + Document = self.apps.get_model("documents", "Document") + + doc1 = Document.objects.get(id=doc1.id) + doc2 = Document.objects.get(id=doc2.id) + + self.assertIsNone(doc1.archive_checksum) + self.assertIsNone(doc2.archive_checksum) + self.assertIsNone(doc1.archive_filename) + self.assertIsNone(doc2.archive_filename) + + +@override_settings(PAPERLESS_FILENAME_FORMAT="") +class TestMigrateArchiveFilesBackwards(DirectoriesMixin, TestMigrations): + + migrate_from = '1012_fix_archive_files' + migrate_to = '1011_auto_20210101_2340' + + def setUpBeforeMigration(self, apps): + + Document = apps.get_model("documents", "Document") + + doc_unrelated = make_test_document(Document, "unrelated", "application/pdf", simple_pdf2, "unrelated.txt", simple_pdf2, "unrelated.pdf") + doc_no_archive = make_test_document(Document, "no_archive", "text/plain", simple_txt, "no_archive.txt") + clashB = make_test_document(Document, "clash", "image/jpeg", simple_jpg, "clash.jpg", simple_pdf, "clash_02.pdf") + + def testArchiveFilesReverted(self): + Document = self.apps.get_model('documents', 'Document') + + for doc in Document.objects.all(): + if doc.archive_checksum: + self.assertTrue(os.path.isfile(archive_path_old(doc))) + with open(source_path(doc), "rb") as f: + original_checksum = hashlib.md5(f.read()).hexdigest() + self.assertEqual(original_checksum, doc.checksum) + + if doc.archive_checksum: + self.assertTrue(os.path.isfile(archive_path_old(doc))) + with open(archive_path_old(doc), "rb") as f: + archive_checksum = hashlib.md5(f.read()).hexdigest() + self.assertEqual(archive_checksum, doc.archive_checksum) + + self.assertEqual(Document.objects.filter(archive_checksum__isnull=False).count(), 2) + + +@override_settings(PAPERLESS_FILENAME_FORMAT="{correspondent}/{title}") +class TestMigrateArchiveFilesBackwardsWithFilenameFormat(TestMigrateArchiveFilesBackwards): + pass + + +@override_settings(PAPERLESS_FILENAME_FORMAT="") +class TestMigrateArchiveFilesBackwardsErrors(DirectoriesMixin, TestMigrations): + + migrate_from = '1012_fix_archive_files' + migrate_to = '1011_auto_20210101_2340' + auto_migrate = False + + def test_filename_clash(self): + + Document = self.apps.get_model("documents", "Document") + + self.clashA = make_test_document(Document, "clash", "application/pdf", simple_pdf, "clash.pdf", simple_pdf, "clash_02.pdf") + self.clashB = make_test_document(Document, "clash", "image/jpeg", simple_jpg, "clash.jpg", simple_pdf, "clash_01.pdf") + + self.assertRaisesMessage(ValueError, "would clash with another archive filename", self.performMigration) + + def test_filename_exists(self): + + Document = self.apps.get_model("documents", "Document") + + self.clashA = make_test_document(Document, "clash", "application/pdf", simple_pdf, "clash.pdf", simple_pdf, "clash.pdf") + self.clashB = make_test_document(Document, "clash", "image/jpeg", simple_jpg, "clash.jpg", simple_pdf, "clash_01.pdf") + + self.assertRaisesMessage(ValueError, "file already exists.", self.performMigration) diff --git a/src/documents/tests/test_migration_mime_type.py b/src/documents/tests/test_migration_mime_type.py new file mode 100644 index 000000000..5e825e89d --- /dev/null +++ b/src/documents/tests/test_migration_mime_type.py @@ -0,0 +1,88 @@ +import os +import shutil + +from django.conf import settings +from django.test import override_settings + +from documents.parsers import get_default_file_extension +from documents.tests.utils import DirectoriesMixin, TestMigrations + +STORAGE_TYPE_UNENCRYPTED = "unencrypted" +STORAGE_TYPE_GPG = "gpg" + + +def source_path_before(self): + if self.filename: + fname = str(self.filename) + else: + fname = "{:07}.{}".format(self.pk, self.file_type) + if self.storage_type == STORAGE_TYPE_GPG: + fname += ".gpg" + + return os.path.join( + settings.ORIGINALS_DIR, + fname + ) + + +def file_type_after(self): + return get_default_file_extension(self.mime_type) + + +def source_path_after(doc): + if doc.filename: + fname = str(doc.filename) + else: + fname = "{:07}{}".format(doc.pk, file_type_after(doc)) + if doc.storage_type == STORAGE_TYPE_GPG: + fname += ".gpg" # pragma: no cover + + return os.path.join( + settings.ORIGINALS_DIR, + fname + ) + + +@override_settings(PASSPHRASE="test") +class TestMigrateMimeType(DirectoriesMixin, TestMigrations): + + migrate_from = '1002_auto_20201111_1105' + migrate_to = '1003_mime_types' + + def setUpBeforeMigration(self, apps): + Document = apps.get_model("documents", "Document") + doc = Document.objects.create(title="test", file_type="pdf", filename="file1.pdf") + self.doc_id = doc.id + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), source_path_before(doc)) + + doc2 = Document.objects.create(checksum="B", file_type="pdf", storage_type=STORAGE_TYPE_GPG) + self.doc2_id = doc2.id + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000004.pdf.gpg"), source_path_before(doc2)) + + def testMimeTypesMigrated(self): + Document = self.apps.get_model('documents', 'Document') + + doc = Document.objects.get(id=self.doc_id) + self.assertEqual(doc.mime_type, "application/pdf") + + doc2 = Document.objects.get(id=self.doc2_id) + self.assertEqual(doc2.mime_type, "application/pdf") + + +@override_settings(PASSPHRASE="test") +class TestMigrateMimeTypeBackwards(DirectoriesMixin, TestMigrations): + + migrate_from = '1003_mime_types' + migrate_to = '1002_auto_20201111_1105' + + def setUpBeforeMigration(self, apps): + Document = apps.get_model("documents", "Document") + doc = Document.objects.create(title="test", mime_type="application/pdf", filename="file1.pdf") + self.doc_id = doc.id + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "simple.pdf"), source_path_after(doc)) + + def testMimeTypesReverted(self): + Document = self.apps.get_model('documents', 'Document') + + doc = Document.objects.get(id=self.doc_id) + self.assertEqual(doc.file_type, "pdf") diff --git a/src/documents/tests/test_migration_remove_null_characters.py b/src/documents/tests/test_migration_remove_null_characters.py new file mode 100644 index 000000000..ba6f18539 --- /dev/null +++ b/src/documents/tests/test_migration_remove_null_characters.py @@ -0,0 +1,15 @@ +from documents.tests.utils import DirectoriesMixin, TestMigrations + + +class TestMigrateNullCharacters(DirectoriesMixin, TestMigrations): + + migrate_from = '1014_auto_20210228_1614' + migrate_to = '1015_remove_null_characters' + + def setUpBeforeMigration(self, apps): + Document = apps.get_model("documents", "Document") + self.doc = Document.objects.create(content="aaa\0bbb") + + def testMimeTypesMigrated(self): + Document = self.apps.get_model('documents', 'Document') + self.assertNotIn("\0", Document.objects.get(id=self.doc.id).content) diff --git a/src/documents/tests/test_migration_tag_colors.py b/src/documents/tests/test_migration_tag_colors.py new file mode 100644 index 000000000..2c4b35925 --- /dev/null +++ b/src/documents/tests/test_migration_tag_colors.py @@ -0,0 +1,37 @@ +from documents.tests.utils import DirectoriesMixin, TestMigrations + + +class TestMigrateTagColor(DirectoriesMixin, TestMigrations): + + migrate_from = '1012_fix_archive_files' + migrate_to = '1013_migrate_tag_colour' + + def setUpBeforeMigration(self, apps): + Tag = apps.get_model("documents", "Tag") + self.t1_id = Tag.objects.create(name="tag1").id + self.t2_id = Tag.objects.create(name="tag2", colour=1).id + self.t3_id = Tag.objects.create(name="tag3", colour=5).id + + def testMimeTypesMigrated(self): + Tag = self.apps.get_model('documents', 'Tag') + self.assertEqual(Tag.objects.get(id=self.t1_id).color, "#a6cee3") + self.assertEqual(Tag.objects.get(id=self.t2_id).color, "#a6cee3") + self.assertEqual(Tag.objects.get(id=self.t3_id).color, "#fb9a99") + + +class TestMigrateTagColorBackwards(DirectoriesMixin, TestMigrations): + + migrate_from = '1013_migrate_tag_colour' + migrate_to = '1012_fix_archive_files' + + def setUpBeforeMigration(self, apps): + Tag = apps.get_model("documents", "Tag") + self.t1_id = Tag.objects.create(name="tag1").id + self.t2_id = Tag.objects.create(name="tag2", color="#cab2d6").id + self.t3_id = Tag.objects.create(name="tag3", color="#123456").id + + def testMimeTypesReverted(self): + Tag = self.apps.get_model('documents', 'Tag') + self.assertEqual(Tag.objects.get(id=self.t1_id).colour, 1) + self.assertEqual(Tag.objects.get(id=self.t2_id).colour, 9) + self.assertEqual(Tag.objects.get(id=self.t3_id).colour, 1) diff --git a/src/documents/tests/test_parsers.py b/src/documents/tests/test_parsers.py index 805e4beac..9dd74313f 100644 --- a/src/documents/tests/test_parsers.py +++ b/src/documents/tests/test_parsers.py @@ -68,7 +68,7 @@ class TestParserDiscovery(TestCase): ) -def fake_get_thumbnail(self, path, mimetype): +def fake_get_thumbnail(self, path, mimetype, file_name): return os.path.join(os.path.dirname(__file__), "examples", "no-text.png") @@ -89,15 +89,15 @@ class TestBaseParser(TestCase): def test_get_optimised_thumbnail(self): parser = DocumentParser(None) - parser.get_optimised_thumbnail("any", "not important") + parser.get_optimised_thumbnail("any", "not important", "document.pdf") @mock.patch("documents.parsers.DocumentParser.get_thumbnail", fake_get_thumbnail) @override_settings(OPTIMIZE_THUMBNAILS=False) def test_get_optimised_thumb_disabled(self): parser = DocumentParser(None) - path = parser.get_optimised_thumbnail("any", "not important") - self.assertEqual(path, fake_get_thumbnail(None, None, None)) + path = parser.get_optimised_thumbnail("any", "not important", "document.pdf") + self.assertEqual(path, fake_get_thumbnail(None, None, None, None)) class TestParserAvailability(TestCase): @@ -114,9 +114,10 @@ class TestParserAvailability(TestCase): self.assertEqual(get_default_file_extension('application/zip'), ".zip") self.assertEqual(get_default_file_extension('aasdasd/dgfgf'), "") - self.assertEqual(get_parser_class_for_mime_type('application/pdf'), RasterisedDocumentParser) - self.assertEqual(get_parser_class_for_mime_type('text/plain'), TextDocumentParser) + self.assertIsInstance(get_parser_class_for_mime_type('application/pdf')(logging_group=None), RasterisedDocumentParser) + self.assertIsInstance(get_parser_class_for_mime_type('text/plain')(logging_group=None), TextDocumentParser) self.assertEqual(get_parser_class_for_mime_type('text/sdgsdf'), None) self.assertTrue(is_file_ext_supported('.pdf')) self.assertFalse(is_file_ext_supported('.hsdfh')) + self.assertFalse(is_file_ext_supported('')) diff --git a/src/documents/tests/test_post_consume_handlers.py b/src/documents/tests/test_post_consume_handlers.py deleted file mode 100644 index b4357448c..000000000 --- a/src/documents/tests/test_post_consume_handlers.py +++ /dev/null @@ -1,56 +0,0 @@ -from unittest import mock - -from django.test import TestCase, override_settings - -from documents.models import Document, Tag, Correspondent -from documents.signals.handlers import run_post_consume_script - - -class PostConsumeTestCase(TestCase): - - @mock.patch("documents.signals.handlers.Popen") - @override_settings(POST_CONSUME_SCRIPT=None) - def test_no_post_consume_script(self, m): - doc = Document.objects.create(title="Test", mime_type="application/pdf") - tag1 = Tag.objects.create(name="a") - tag2 = Tag.objects.create(name="b") - doc.tags.add(tag1) - doc.tags.add(tag2) - - run_post_consume_script(None, doc) - - m.assert_not_called() - - @mock.patch("documents.signals.handlers.Popen") - @override_settings(POST_CONSUME_SCRIPT="script") - def test_post_consume_script_simple(self, m): - doc = Document.objects.create(title="Test", mime_type="application/pdf") - - run_post_consume_script(None, doc) - - m.assert_called_once() - - @mock.patch("documents.signals.handlers.Popen") - @override_settings(POST_CONSUME_SCRIPT="script") - def test_post_consume_script_with_correspondent(self, m): - c = Correspondent.objects.create(name="my_bank") - doc = Document.objects.create(title="Test", mime_type="application/pdf", correspondent=c) - tag1 = Tag.objects.create(name="a") - tag2 = Tag.objects.create(name="b") - doc.tags.add(tag1) - doc.tags.add(tag2) - - run_post_consume_script(None, doc) - - m.assert_called_once() - - args, kwargs = m.call_args - - command = args[0] - - self.assertEqual(command[0], "script") - self.assertEqual(command[1], str(doc.pk)) - self.assertEqual(command[5], f"/api/documents/{doc.pk}/download/") - self.assertEqual(command[6], f"/api/documents/{doc.pk}/thumb/") - self.assertEqual(command[7], "my_bank") - self.assertCountEqual(command[8].split(","), ["a", "b"]) diff --git a/src/documents/tests/test_sanity_check.py b/src/documents/tests/test_sanity_check.py index 725e87617..b07ba930a 100644 --- a/src/documents/tests/test_sanity_check.py +++ b/src/documents/tests/test_sanity_check.py @@ -1,23 +1,82 @@ +import logging import os import shutil from pathlib import Path +import filelock +from django.conf import settings from django.test import TestCase from documents.models import Document -from documents.sanity_checker import check_sanity, SanityFailedError +from documents.sanity_checker import check_sanity, SanityCheckMessages from documents.tests.utils import DirectoriesMixin +class TestSanityCheckMessages(TestCase): + + def test_no_messages(self): + messages = SanityCheckMessages() + self.assertEqual(len(messages), 0) + self.assertFalse(messages.has_error()) + self.assertFalse(messages.has_warning()) + with self.assertLogs() as capture: + messages.log_messages() + self.assertEqual(len(capture.output), 1) + self.assertEqual(capture.records[0].levelno, logging.INFO) + self.assertEqual(capture.records[0].message, "Sanity checker detected no issues.") + + def test_info(self): + messages = SanityCheckMessages() + messages.info("Something might be wrong") + self.assertEqual(len(messages), 1) + self.assertFalse(messages.has_error()) + self.assertFalse(messages.has_warning()) + with self.assertLogs() as capture: + messages.log_messages() + self.assertEqual(len(capture.output), 1) + self.assertEqual(capture.records[0].levelno, logging.INFO) + self.assertEqual(capture.records[0].message, "Something might be wrong") + + def test_warning(self): + messages = SanityCheckMessages() + messages.warning("Something is wrong") + self.assertEqual(len(messages), 1) + self.assertFalse(messages.has_error()) + self.assertTrue(messages.has_warning()) + with self.assertLogs() as capture: + messages.log_messages() + self.assertEqual(len(capture.output), 1) + self.assertEqual(capture.records[0].levelno, logging.WARNING) + self.assertEqual(capture.records[0].message, "Something is wrong") + + def test_error(self): + messages = SanityCheckMessages() + messages.error("Something is seriously wrong") + self.assertEqual(len(messages), 1) + self.assertTrue(messages.has_error()) + self.assertFalse(messages.has_warning()) + with self.assertLogs() as capture: + messages.log_messages() + self.assertEqual(len(capture.output), 1) + self.assertEqual(capture.records[0].levelno, logging.ERROR) + self.assertEqual(capture.records[0].message, "Something is seriously wrong") + class TestSanityCheck(DirectoriesMixin, TestCase): def make_test_data(self): - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000001.pdf"), os.path.join(self.dirs.originals_dir, "0000001.pdf")) - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "archive", "0000001.pdf"), os.path.join(self.dirs.archive_dir, "0000001.pdf")) - shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png"), os.path.join(self.dirs.thumbnail_dir, "0000001.png")) + with filelock.FileLock(settings.MEDIA_LOCK): + # just make sure that the lockfile is present. + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "originals", "0000001.pdf"), os.path.join(self.dirs.originals_dir, "0000001.pdf")) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "archive", "0000001.pdf"), os.path.join(self.dirs.archive_dir, "0000001.pdf")) + shutil.copy(os.path.join(os.path.dirname(__file__), "samples", "documents", "thumbnails", "0000001.png"), os.path.join(self.dirs.thumbnail_dir, "0000001.png")) - return Document.objects.create(title="test", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", content="test", pk=1, filename="0000001.pdf", mime_type="application/pdf") + return Document.objects.create(title="test", checksum="42995833e01aea9b3edee44bbfdd7ce1", archive_checksum="62acb0bcbfbcaa62ca6ad3668e4e404b", content="test", pk=1, filename="0000001.pdf", mime_type="application/pdf", archive_filename="0000001.pdf") + + def assertSanityError(self, messageRegex): + messages = check_sanity() + self.assertTrue(messages.has_error()) + self.assertRegex(messages[0]['message'], messageRegex) def test_no_docs(self): self.assertEqual(len(check_sanity()), 0) @@ -29,59 +88,75 @@ class TestSanityCheck(DirectoriesMixin, TestCase): def test_no_thumbnail(self): doc = self.make_test_data() os.remove(doc.thumbnail_path) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Thumbnail of document .* does not exist") def test_thumbnail_no_access(self): doc = self.make_test_data() os.chmod(doc.thumbnail_path, 0o000) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Cannot read thumbnail file of document") os.chmod(doc.thumbnail_path, 0o777) def test_no_original(self): doc = self.make_test_data() os.remove(doc.source_path) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Original of document .* does not exist.") def test_original_no_access(self): doc = self.make_test_data() os.chmod(doc.source_path, 0o000) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Cannot read original file of document") os.chmod(doc.source_path, 0o777) def test_original_checksum_mismatch(self): doc = self.make_test_data() doc.checksum = "WOW" doc.save() - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Checksum mismatch of document") def test_no_archive(self): doc = self.make_test_data() os.remove(doc.archive_path) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Archived version of document .* does not exist.") def test_archive_no_access(self): doc = self.make_test_data() os.chmod(doc.archive_path, 0o000) - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Cannot read archive file of document") os.chmod(doc.archive_path, 0o777) def test_archive_checksum_mismatch(self): doc = self.make_test_data() doc.archive_checksum = "WOW" doc.save() - self.assertEqual(len(check_sanity()), 1) + self.assertSanityError("Checksum mismatch of archived document") def test_empty_content(self): doc = self.make_test_data() doc.content = "" doc.save() - self.assertEqual(len(check_sanity()), 1) + messages = check_sanity() + self.assertFalse(messages.has_error()) + self.assertFalse(messages.has_warning()) + self.assertEqual(len(messages), 1) + self.assertRegex(messages[0]['message'], "Document .* has no content.") def test_orphaned_file(self): doc = self.make_test_data() Path(self.dirs.originals_dir, "orphaned").touch() - self.assertEqual(len(check_sanity()), 1) + messages = check_sanity() + self.assertFalse(messages.has_error()) + self.assertTrue(messages.has_warning()) + self.assertEqual(len(messages), 1) + self.assertRegex(messages[0]['message'], "Orphaned file in media dir") - def test_all(self): - Document.objects.create(title="test", checksum="dgfhj", archive_checksum="dfhg", content="", pk=1, filename="0000001.pdf") - string = str(SanityFailedError(check_sanity())) + def test_archive_filename_no_checksum(self): + doc = self.make_test_data() + doc.archive_checksum = None + doc.save() + self.assertSanityError("has an archive file, but its checksum is missing.") + + def test_archive_checksum_no_filename(self): + doc = self.make_test_data() + doc.archive_filename = None + doc.save() + self.assertSanityError("has an archive file checksum, but no archive filename.") diff --git a/src/documents/tests/test_settings.py b/src/documents/tests/test_settings.py new file mode 100644 index 000000000..0036daee7 --- /dev/null +++ b/src/documents/tests/test_settings.py @@ -0,0 +1,34 @@ +import logging +from unittest import mock + +from django.test import TestCase + +from paperless.settings import default_task_workers, default_threads_per_worker + + +class TestSettings(TestCase): + + @mock.patch("paperless.settings.multiprocessing.cpu_count") + def test_single_core(self, cpu_count): + cpu_count.return_value = 1 + + default_workers = default_task_workers() + + default_threads = default_threads_per_worker(default_workers) + + self.assertEqual(default_workers, 1) + self.assertEqual(default_threads, 1) + + def test_workers_threads(self): + for i in range(1, 64): + with mock.patch("paperless.settings.multiprocessing.cpu_count") as cpu_count: + cpu_count.return_value = i + + default_workers = default_task_workers() + + default_threads = default_threads_per_worker(default_workers) + + self.assertTrue(default_workers >= 1) + self.assertTrue(default_threads >= 1) + + self.assertTrue(default_workers * default_threads <= i, f"{i}") diff --git a/src/documents/tests/test_tasks.py b/src/documents/tests/test_tasks.py index 6d04e58e1..dab8ebfb9 100644 --- a/src/documents/tests/test_tasks.py +++ b/src/documents/tests/test_tasks.py @@ -1,10 +1,13 @@ -from datetime import datetime +import os +from unittest import mock +from django.conf import settings from django.test import TestCase from django.utils import timezone from documents import tasks -from documents.models import Document +from documents.models import Document, Tag, Correspondent, DocumentType +from documents.sanity_checker import SanityCheckMessages, SanityCheckFailedException from documents.tests.utils import DirectoriesMixin @@ -20,5 +23,88 @@ class TestTasks(DirectoriesMixin, TestCase): tasks.index_optimize() - def test_train_classifier(self): + @mock.patch("documents.tasks.load_classifier") + def test_train_classifier_no_auto_matching(self, load_classifier): tasks.train_classifier() + load_classifier.assert_not_called() + + @mock.patch("documents.tasks.load_classifier") + def test_train_classifier_with_auto_tag(self, load_classifier): + load_classifier.return_value = None + Tag.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test") + tasks.train_classifier() + load_classifier.assert_called_once() + self.assertFalse(os.path.isfile(settings.MODEL_FILE)) + + @mock.patch("documents.tasks.load_classifier") + def test_train_classifier_with_auto_type(self, load_classifier): + load_classifier.return_value = None + DocumentType.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test") + tasks.train_classifier() + load_classifier.assert_called_once() + self.assertFalse(os.path.isfile(settings.MODEL_FILE)) + + @mock.patch("documents.tasks.load_classifier") + def test_train_classifier_with_auto_correspondent(self, load_classifier): + load_classifier.return_value = None + Correspondent.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test") + tasks.train_classifier() + load_classifier.assert_called_once() + self.assertFalse(os.path.isfile(settings.MODEL_FILE)) + + def test_train_classifier(self): + c = Correspondent.objects.create(matching_algorithm=Tag.MATCH_AUTO, name="test") + doc = Document.objects.create(correspondent=c, content="test", title="test") + self.assertFalse(os.path.isfile(settings.MODEL_FILE)) + + tasks.train_classifier() + self.assertTrue(os.path.isfile(settings.MODEL_FILE)) + mtime = os.stat(settings.MODEL_FILE).st_mtime + + tasks.train_classifier() + self.assertTrue(os.path.isfile(settings.MODEL_FILE)) + mtime2 = os.stat(settings.MODEL_FILE).st_mtime + self.assertEqual(mtime, mtime2) + + doc.content = "test2" + doc.save() + tasks.train_classifier() + self.assertTrue(os.path.isfile(settings.MODEL_FILE)) + mtime3 = os.stat(settings.MODEL_FILE).st_mtime + self.assertNotEqual(mtime2, mtime3) + + @mock.patch("documents.tasks.sanity_checker.check_sanity") + def test_sanity_check_success(self, m): + m.return_value = SanityCheckMessages() + self.assertEqual(tasks.sanity_check(), "No issues detected.") + m.assert_called_once() + + @mock.patch("documents.tasks.sanity_checker.check_sanity") + def test_sanity_check_error(self, m): + messages = SanityCheckMessages() + messages.error("Some error") + m.return_value = messages + self.assertRaises(SanityCheckFailedException, tasks.sanity_check) + m.assert_called_once() + + @mock.patch("documents.tasks.sanity_checker.check_sanity") + def test_sanity_check_warning(self, m): + messages = SanityCheckMessages() + messages.warning("Some warning") + m.return_value = messages + self.assertEqual(tasks.sanity_check(), "Sanity check exited with warnings. See log.") + m.assert_called_once() + + @mock.patch("documents.tasks.sanity_checker.check_sanity") + def test_sanity_check_info(self, m): + messages = SanityCheckMessages() + messages.info("Some info") + m.return_value = messages + self.assertEqual(tasks.sanity_check(), "Sanity check exited with infos. See log.") + m.assert_called_once() + + def test_bulk_update_documents(self): + doc1 = Document.objects.create(title="test", content="my document", checksum="wow", added=timezone.now(), + created=timezone.now(), modified=timezone.now()) + + tasks.bulk_update_documents([doc1.pk]) diff --git a/src/documents/tests/test_views.py b/src/documents/tests/test_views.py new file mode 100644 index 000000000..7f7a07a61 --- /dev/null +++ b/src/documents/tests/test_views.py @@ -0,0 +1,30 @@ +from django.conf import settings +from django.contrib.auth.models import User +from django.test import TestCase + + +class TestViews(TestCase): + + def setUp(self) -> None: + self.user = User.objects.create_user("testuser") + + def test_login_redirect(self): + response = self.client.get('/') + self.assertEqual(response.status_code, 302) + self.assertEqual(response.url, "/accounts/login/?next=/") + + def test_index(self): + self.client.force_login(self.user) + for (language_given, language_actual) in [("", "en-US"), ("en-US", "en-US"), ("de", "de-DE"), ("en", "en-US"), ("en-us", "en-US"), ("fr", "fr-FR"), ("jp", "en-US")]: + if language_given: + self.client.cookies.load({settings.LANGUAGE_COOKIE_NAME: language_given}) + elif settings.LANGUAGE_COOKIE_NAME in self.client.cookies.keys(): + self.client.cookies.pop(settings.LANGUAGE_COOKIE_NAME) + + response = self.client.get('/', ) + self.assertEqual(response.status_code, 200) + self.assertEqual(response.context_data['webmanifest'], f"frontend/{language_actual}/manifest.webmanifest") + self.assertEqual(response.context_data['styles_css'], f"frontend/{language_actual}/styles.css") + self.assertEqual(response.context_data['runtime_js'], f"frontend/{language_actual}/runtime.js") + self.assertEqual(response.context_data['polyfills_js'], f"frontend/{language_actual}/polyfills.js") + self.assertEqual(response.context_data['main_js'], f"frontend/{language_actual}/main.js") diff --git a/src/documents/tests/utils.py b/src/documents/tests/utils.py index 7f9d50ed5..da8d3d429 100644 --- a/src/documents/tests/utils.py +++ b/src/documents/tests/utils.py @@ -4,7 +4,10 @@ import tempfile from collections import namedtuple from contextlib import contextmanager -from django.test import override_settings +from django.apps import apps +from django.db import connection +from django.db.migrations.executor import MigrationExecutor +from django.test import override_settings, TransactionTestCase def setup_directories(): @@ -19,12 +22,15 @@ def setup_directories(): dirs.originals_dir = os.path.join(dirs.media_dir, "documents", "originals") dirs.thumbnail_dir = os.path.join(dirs.media_dir, "documents", "thumbnails") dirs.archive_dir = os.path.join(dirs.media_dir, "documents", "archive") + dirs.logging_dir = os.path.join(dirs.data_dir, "log") os.makedirs(dirs.index_dir, exist_ok=True) os.makedirs(dirs.originals_dir, exist_ok=True) os.makedirs(dirs.thumbnail_dir, exist_ok=True) os.makedirs(dirs.archive_dir, exist_ok=True) + os.makedirs(dirs.logging_dir, exist_ok=True) + dirs.settings_override = override_settings( DATA_DIR=dirs.data_dir, SCRATCH_DIR=dirs.scratch_dir, @@ -33,8 +39,10 @@ def setup_directories(): THUMBNAIL_DIR=dirs.thumbnail_dir, ARCHIVE_DIR=dirs.archive_dir, CONSUMPTION_DIR=dirs.consumption_dir, + LOGGING_DIR=dirs.logging_dir, INDEX_DIR=dirs.index_dir, - MODEL_FILE=os.path.join(dirs.data_dir, "classification_model.pickle") + MODEL_FILE=os.path.join(dirs.data_dir, "classification_model.pickle"), + MEDIA_LOCK=os.path.join(dirs.media_dir, "media.lock") ) dirs.settings_override.enable() @@ -74,3 +82,45 @@ class DirectoriesMixin: def tearDown(self) -> None: super(DirectoriesMixin, self).tearDown() remove_dirs(self.dirs) + + +class TestMigrations(TransactionTestCase): + + @property + def app(self): + return apps.get_containing_app_config(type(self).__module__).name + + migrate_from = None + migrate_to = None + auto_migrate = True + + def setUp(self): + super(TestMigrations, self).setUp() + + assert self.migrate_from and self.migrate_to, \ + "TestCase '{}' must define migrate_from and migrate_to properties".format(type(self).__name__) + self.migrate_from = [(self.app, self.migrate_from)] + self.migrate_to = [(self.app, self.migrate_to)] + executor = MigrationExecutor(connection) + old_apps = executor.loader.project_state(self.migrate_from).apps + + # Reverse to the original migration + executor.migrate(self.migrate_from) + + self.setUpBeforeMigration(old_apps) + + self.apps = old_apps + + if self.auto_migrate: + self.performMigration() + + def performMigration(self): + # Run the migration to test + executor = MigrationExecutor(connection) + executor.loader.build_graph() # reload. + executor.migrate(self.migrate_to) + + self.apps = executor.loader.project_state(self.migrate_to).apps + + def setUpBeforeMigration(self, apps): + pass diff --git a/src/documents/views.py b/src/documents/views.py old mode 100755 new mode 100644 index b42ae1f96..d8fcca2a9 --- a/src/documents/views.py +++ b/src/documents/views.py @@ -1,18 +1,25 @@ +import logging import os import tempfile +import uuid +import zipfile from datetime import datetime from time import mktime from django.conf import settings -from django.db.models import Count, Max +from django.db.models import Count, Max, Case, When, IntegerField +from django.db.models.functions import Lower from django.http import HttpResponse, HttpResponseBadRequest, Http404 +from django.utils.translation import get_language from django.views.decorators.cache import cache_control from django.views.generic import TemplateView from django_filters.rest_framework import DjangoFilterBackend from django_q.tasks import async_task from rest_framework import parsers from rest_framework.decorators import action +from rest_framework.exceptions import NotFound from rest_framework.filters import OrderingFilter, SearchFilter +from rest_framework.generics import GenericAPIView from rest_framework.mixins import ( DestroyModelMixin, ListModelMixin, @@ -25,41 +32,76 @@ from rest_framework.views import APIView from rest_framework.viewsets import ( GenericViewSet, ModelViewSet, - ReadOnlyModelViewSet + ViewSet ) -import documents.index as index from paperless.db import GnuPG from paperless.views import StandardPagination +from .bulk_download import OriginalAndArchiveStrategy, OriginalsOnlyStrategy, \ + ArchiveOnlyStrategy +from .classifier import load_classifier from .filters import ( CorrespondentFilterSet, DocumentFilterSet, TagFilterSet, - DocumentTypeFilterSet, - LogFilterSet + DocumentTypeFilterSet ) -from .models import Correspondent, Document, Log, Tag, DocumentType +from .matching import match_correspondents, match_tags, match_document_types +from .models import Correspondent, Document, Tag, DocumentType, SavedView from .parsers import get_parser_class_for_mime_type from .serialisers import ( CorrespondentSerializer, DocumentSerializer, - LogSerializer, + TagSerializerVersion1, TagSerializer, DocumentTypeSerializer, - PostDocumentSerializer + PostDocumentSerializer, + SavedViewSerializer, + BulkEditSerializer, + DocumentListSerializer, + BulkDownloadSerializer ) +logger = logging.getLogger("paperless.api") + + class IndexView(TemplateView): template_name = "index.html" + def get_language(self): + # This is here for the following reason: + # Django identifies languages in the form "en-us" + # However, angular generates locales as "en-US". + # this translates between these two forms. + lang = get_language() + if "-" in lang: + first = lang[:lang.index("-")] + second = lang[lang.index("-")+1:] + return f"{first}-{second.upper()}" + else: + return lang + + def get_context_data(self, **kwargs): + context = super().get_context_data(**kwargs) + context['cookie_prefix'] = settings.COOKIE_PREFIX + context['username'] = self.request.user.username + context['full_name'] = self.request.user.get_full_name() + context['styles_css'] = f"frontend/{self.get_language()}/styles.css" + context['runtime_js'] = f"frontend/{self.get_language()}/runtime.js" + context['polyfills_js'] = f"frontend/{self.get_language()}/polyfills.js" # NOQA: E501 + context['main_js'] = f"frontend/{self.get_language()}/main.js" + context['webmanifest'] = f"frontend/{self.get_language()}/manifest.webmanifest" # NOQA: E501 + context['apple_touch_icon'] = f"frontend/{self.get_language()}/apple-touch-icon.png" # NOQA: E501 + return context + class CorrespondentViewSet(ModelViewSet): model = Correspondent queryset = Correspondent.objects.annotate( document_count=Count('documents'), - last_correspondence=Max('documents__created')).order_by('name') + last_correspondence=Max('documents__created')).order_by(Lower('name')) serializer_class = CorrespondentSerializer pagination_class = StandardPagination @@ -78,9 +120,14 @@ class TagViewSet(ModelViewSet): model = Tag queryset = Tag.objects.annotate( - document_count=Count('documents')).order_by('name') + document_count=Count('documents')).order_by(Lower('name')) + + def get_serializer_class(self): + if int(self.request.version) == 1: + return TagSerializerVersion1 + else: + return TagSerializer - serializer_class = TagSerializer pagination_class = StandardPagination permission_classes = (IsAuthenticated,) filter_backends = (DjangoFilterBackend, OrderingFilter) @@ -92,7 +139,7 @@ class DocumentTypeViewSet(ModelViewSet): model = DocumentType queryset = DocumentType.objects.annotate( - document_count=Count('documents')).order_by('name') + document_count=Count('documents')).order_by(Lower('name')) serializer_class = DocumentTypeSerializer pagination_class = StandardPagination @@ -125,13 +172,29 @@ class DocumentViewSet(RetrieveModelMixin, "added", "archive_serial_number") + def get_queryset(self): + return Document.objects.distinct() + + def get_serializer(self, *args, **kwargs): + fields_param = self.request.query_params.get('fields', None) + if fields_param: + fields = fields_param.split(",") + else: + fields = None + serializer_class = self.get_serializer_class() + kwargs.setdefault('context', self.get_serializer_context()) + kwargs.setdefault('fields', fields) + return serializer_class(*args, **kwargs) + def update(self, request, *args, **kwargs): response = super(DocumentViewSet, self).update( request, *args, **kwargs) + from documents import index index.add_or_update_document(self.get_object()) return response def destroy(self, request, *args, **kwargs): + from documents import index index.remove_document_from_index(self.get_object()) return super(DocumentViewSet, self).destroy(request, *args, **kwargs) @@ -144,7 +207,7 @@ class DocumentViewSet(RetrieveModelMixin, def file_response(self, pk, request, disposition): doc = Document.objects.get(id=pk) - if not self.original_requested(request) and os.path.isfile(doc.archive_path): # NOQA: E501 + if not self.original_requested(request) and doc.has_archive_version: # NOQA: E501 file_handle = doc.archive_file filename = doc.get_public_filename(archive=True) mime_type = 'application/pdf' @@ -167,40 +230,70 @@ class DocumentViewSet(RetrieveModelMixin, parser_class = get_parser_class_for_mime_type(mime_type) if parser_class: - parser = parser_class(logging_group=None) - return parser.extract_metadata(file, mime_type) + parser = parser_class(progress_callback=None, logging_group=None) + + try: + return parser.extract_metadata(file, mime_type) + except Exception as e: + # TODO: cover GPG errors, remove later. + return [] else: return [] + def get_filesize(self, filename): + if os.path.isfile(filename): + return os.stat(filename).st_size + else: + return None + @action(methods=['get'], detail=True) def metadata(self, request, pk=None): try: doc = Document.objects.get(pk=pk) - - meta = { - "original_checksum": doc.checksum, - "original_size": os.stat(doc.source_path).st_size, - "original_mime_type": doc.mime_type, - "media_filename": doc.filename, - "has_archive_version": os.path.isfile(doc.archive_path), - "original_metadata": self.get_metadata( - doc.source_path, doc.mime_type) - } - - if doc.archive_checksum and os.path.isfile(doc.archive_path): - meta['archive_checksum'] = doc.archive_checksum - meta['archive_size'] = os.stat(doc.archive_path).st_size, - meta['archive_metadata'] = self.get_metadata( - doc.archive_path, "application/pdf") - else: - meta['archive_checksum'] = None - meta['archive_size'] = None - meta['archive_metadata'] = None - - return Response(meta) except Document.DoesNotExist: raise Http404() + meta = { + "original_checksum": doc.checksum, + "original_size": self.get_filesize(doc.source_path), + "original_mime_type": doc.mime_type, + "media_filename": doc.filename, + "has_archive_version": doc.has_archive_version, + "original_metadata": self.get_metadata( + doc.source_path, doc.mime_type), + "archive_checksum": doc.archive_checksum, + "archive_media_filename": doc.archive_filename + } + + if doc.has_archive_version: + meta['archive_size'] = self.get_filesize(doc.archive_path) + meta['archive_metadata'] = self.get_metadata( + doc.archive_path, "application/pdf") + else: + meta['archive_size'] = None + meta['archive_metadata'] = None + + return Response(meta) + + @action(methods=['get'], detail=True) + def suggestions(self, request, pk=None): + try: + doc = Document.objects.get(pk=pk) + except Document.DoesNotExist: + raise Http404() + + classifier = load_classifier() + + return Response({ + "correspondents": [ + c.id for c in match_correspondents(doc, classifier) + ], + "tags": [t.id for t in match_tags(doc, classifier)], + "document_types": [ + dt.id for dt in match_document_types(doc, classifier) + ] + }) + @action(methods=['get'], detail=True) def preview(self, request, pk=None): try: @@ -214,7 +307,14 @@ class DocumentViewSet(RetrieveModelMixin, @cache_control(public=False, max_age=315360000) def thumb(self, request, pk=None): try: - return HttpResponse(Document.objects.get(id=pk).thumbnail_file, + doc = Document.objects.get(id=pk) + if doc.storage_type == Document.STORAGE_TYPE_GPG: + handle = GnuPG.decrypted(doc.thumbnail_file) + else: + handle = doc.thumbnail_file + # TODO: Send ETag information and use that to send new thumbnails + # if available + return HttpResponse(handle, content_type='image/png') except (FileNotFoundError, Document.DoesNotExist): raise Http404() @@ -228,35 +328,138 @@ class DocumentViewSet(RetrieveModelMixin, raise Http404() -class LogViewSet(ReadOnlyModelViewSet): - model = Log +class SearchResultSerializer(DocumentSerializer): - queryset = Log.objects.all() - serializer_class = LogSerializer + def to_representation(self, instance): + doc = Document.objects.get(id=instance['id']) + r = super(SearchResultSerializer, self).to_representation(doc) + r['__search_hit__'] = { + "score": instance.score, + "highlights": instance.highlights("content", + text=doc.content) if doc else None, # NOQA: E501 + "rank": instance.rank + } + + return r + + +class UnifiedSearchViewSet(DocumentViewSet): + + def __init__(self, *args, **kwargs): + super(UnifiedSearchViewSet, self).__init__(*args, **kwargs) + self.searcher = None + + def get_serializer_class(self): + if self._is_search_request(): + return SearchResultSerializer + else: + return DocumentSerializer + + def _is_search_request(self): + return ("query" in self.request.query_params or + "more_like_id" in self.request.query_params) + + def filter_queryset(self, queryset): + if self._is_search_request(): + from documents import index + + if "query" in self.request.query_params: + query_class = index.DelayedFullTextQuery + elif "more_like_id" in self.request.query_params: + query_class = index.DelayedMoreLikeThisQuery + else: + raise ValueError() + + return query_class( + self.searcher, + self.request.query_params, + self.paginator.get_page_size(self.request)) + else: + return super(UnifiedSearchViewSet, self).filter_queryset(queryset) + + def list(self, request, *args, **kwargs): + if self._is_search_request(): + from documents import index + try: + with index.open_index_searcher() as s: + self.searcher = s + return super(UnifiedSearchViewSet, self).list(request) + except NotFound: + raise + except Exception as e: + return HttpResponseBadRequest(str(e)) + else: + return super(UnifiedSearchViewSet, self).list(request) + + +class LogViewSet(ViewSet): + + permission_classes = (IsAuthenticated,) + + log_files = ["paperless", "mail"] + + def retrieve(self, request, pk=None, *args, **kwargs): + if pk not in self.log_files: + raise Http404() + + filename = os.path.join(settings.LOGGING_DIR, f"{pk}.log") + + if not os.path.isfile(filename): + raise Http404() + + with open(filename, "r") as f: + lines = [line.rstrip() for line in f.readlines()] + + return Response(lines) + + def list(self, request, *args, **kwargs): + return Response(self.log_files) + + +class SavedViewViewSet(ModelViewSet): + model = SavedView + + queryset = SavedView.objects.all() + serializer_class = SavedViewSerializer pagination_class = StandardPagination permission_classes = (IsAuthenticated,) - filter_backends = (DjangoFilterBackend, OrderingFilter) - filterset_class = LogFilterSet - ordering_fields = ("created",) + + def get_queryset(self): + user = self.request.user + return SavedView.objects.filter(user=user) + + def perform_create(self, serializer): + serializer.save(user=self.request.user) -class PostDocumentView(APIView): +class BulkEditView(GenericAPIView): + + permission_classes = (IsAuthenticated,) + serializer_class = BulkEditSerializer + parser_classes = (parsers.JSONParser,) + + def post(self, request, *args, **kwargs): + serializer = self.get_serializer(data=request.data) + serializer.is_valid(raise_exception=True) + + method = serializer.validated_data.get("method") + parameters = serializer.validated_data.get("parameters") + documents = serializer.validated_data.get("documents") + + try: + # TODO: parameter validation + result = method(documents, **parameters) + return Response({"result": result}) + except Exception as e: + return HttpResponseBadRequest(str(e)) + + +class PostDocumentView(GenericAPIView): permission_classes = (IsAuthenticated,) serializer_class = PostDocumentSerializer parser_classes = (parsers.MultiPartParser,) - def get_serializer_context(self): - return { - 'request': self.request, - 'format': self.format_kwarg, - 'view': self - } - - def get_serializer(self, *args, **kwargs): - kwargs['context'] = self.get_serializer_context() - return self.serializer_class(*args, **kwargs) - def post(self, request, *args, **kwargs): serializer = self.get_serializer(data=request.data) @@ -277,74 +480,73 @@ class PostDocumentView(APIView): delete=False) as f: f.write(doc_data) os.utime(f.name, times=(t, t)) + temp_filename = f.name + + task_id = str(uuid.uuid4()) + + async_task("documents.tasks.consume_file", + temp_filename, + override_filename=doc_name, + override_title=title, + override_correspondent_id=correspondent_id, + override_document_type_id=document_type_id, + override_tag_ids=tag_ids, + task_id=task_id, + task_name=os.path.basename(doc_name)[:100]) - async_task("documents.tasks.consume_file", - f.name, - override_filename=doc_name, - override_title=title, - override_correspondent_id=correspondent_id, - override_document_type_id=document_type_id, - override_tag_ids=tag_ids, - task_name=os.path.basename(doc_name)[:100]) return Response("OK") -class SearchView(APIView): +class SelectionDataView(GenericAPIView): permission_classes = (IsAuthenticated,) + serializer_class = DocumentListSerializer + parser_classes = (parsers.MultiPartParser, parsers.JSONParser) - def __init__(self, *args, **kwargs): - super(SearchView, self).__init__(*args, **kwargs) - self.ix = index.open_index() + def post(self, request, format=None): + serializer = self.get_serializer(data=request.data) + serializer.is_valid(raise_exception=True) - def add_infos_to_hit(self, r): - doc = Document.objects.get(id=r['id']) - return {'id': r['id'], - 'highlights': r.highlights("content", text=doc.content), - 'score': r.score, - 'rank': r.rank, - 'document': DocumentSerializer(doc).data, - 'title': r['title'] - } + ids = serializer.validated_data.get('documents') - def get(self, request, format=None): - if 'query' not in request.query_params: - return Response({ - 'count': 0, - 'page': 0, - 'page_count': 0, - 'results': []}) + correspondents = Correspondent.objects.annotate( + document_count=Count(Case( + When(documents__id__in=ids, then=1), + output_field=IntegerField() + ))) - query = request.query_params['query'] - try: - page = int(request.query_params.get('page', 1)) - except (ValueError, TypeError): - page = 1 + tags = Tag.objects.annotate(document_count=Count(Case( + When(documents__id__in=ids, then=1), + output_field=IntegerField() + ))) - if page < 1: - page = 1 + types = DocumentType.objects.annotate(document_count=Count(Case( + When(documents__id__in=ids, then=1), + output_field=IntegerField() + ))) - try: - with index.query_page(self.ix, query, page) as (result_page, - corrected_query): - return Response( - {'count': len(result_page), - 'page': result_page.pagenum, - 'page_count': result_page.pagecount, - 'corrected_query': corrected_query, - 'results': list(map(self.add_infos_to_hit, result_page))}) - except Exception as e: - return HttpResponseBadRequest(str(e)) + r = Response({ + "selected_correspondents": [{ + "id": t.id, + "document_count": t.document_count + } for t in correspondents], + "selected_tags": [{ + "id": t.id, + "document_count": t.document_count + } for t in tags], + "selected_document_types": [{ + "id": t.id, + "document_count": t.document_count + } for t in types] + }) + + return r class SearchAutoCompleteView(APIView): permission_classes = (IsAuthenticated,) - def __init__(self, *args, **kwargs): - super(SearchAutoCompleteView, self).__init__(*args, **kwargs) - self.ix = index.open_index() - def get(self, request, format=None): if 'term' in request.query_params: term = request.query_params['term'] @@ -358,7 +560,11 @@ class SearchAutoCompleteView(APIView): else: limit = 10 - return Response(index.autocomplete(self.ix, term, limit)) + from documents import index + + ix = index.open_index() + + return Response(index.autocomplete(ix, term, limit)) class StatisticsView(APIView): @@ -366,8 +572,55 @@ class StatisticsView(APIView): permission_classes = (IsAuthenticated,) def get(self, request, format=None): - return Response({ - 'documents_total': Document.objects.all().count(), - 'documents_inbox': Document.objects.filter( + documents_total = Document.objects.all().count() + if Tag.objects.filter(is_inbox_tag=True).exists(): + documents_inbox = Document.objects.filter( tags__is_inbox_tag=True).distinct().count() + else: + documents_inbox = None + + return Response({ + 'documents_total': documents_total, + 'documents_inbox': documents_inbox, }) + + +class BulkDownloadView(GenericAPIView): + + permission_classes = (IsAuthenticated,) + serializer_class = BulkDownloadSerializer + parser_classes = (parsers.JSONParser,) + + def post(self, request, format=None): + serializer = self.get_serializer(data=request.data) + serializer.is_valid(raise_exception=True) + + ids = serializer.validated_data.get('documents') + compression = serializer.validated_data.get('compression') + content = serializer.validated_data.get('content') + + os.makedirs(settings.SCRATCH_DIR, exist_ok=True) + temp = tempfile.NamedTemporaryFile( + dir=settings.SCRATCH_DIR, + suffix="-compressed-archive", + delete=False) + + if content == 'both': + strategy_class = OriginalAndArchiveStrategy + elif content == 'originals': + strategy_class = OriginalsOnlyStrategy + else: + strategy_class = ArchiveOnlyStrategy + + with zipfile.ZipFile(temp.name, "w", compression) as zipf: + strategy = strategy_class(zipf) + for id in ids: + doc = Document.objects.get(id=id) + strategy.add_document(doc) + + with open(temp.name, "rb") as f: + response = HttpResponse(f, content_type="application/zip") + response["Content-Disposition"] = '{}; filename="{}"'.format( + "attachment", "documents.zip") + + return response diff --git a/src/locale/cs_CZ/LC_MESSAGES/django.po b/src/locale/cs_CZ/LC_MESSAGES/django.po new file mode 100644 index 000000000..367d4086a --- /dev/null +++ b/src/locale/cs_CZ/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Czech\n" +"Language: cs_CZ\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=4; plural=(n==1) ? 0 : (n>=2 && n<=4) ? 1 : 3;\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: cs\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokumenty" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Jakékoliv slovo" + +#: documents/models.py:33 +msgid "All words" +msgstr "Všechna slova" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Přesná shoda" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Regulární výraz" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Fuzzy slovo" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatický" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "název" + +#: documents/models.py:45 +msgid "match" +msgstr "shoda" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algoritmus pro shodu" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "je ignorováno" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "korespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "korespondenti" + +#: documents/models.py:81 +msgid "color" +msgstr "barva" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "tag přichozí" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Označí tento tag jako tag pro příchozí: Všechny nově zkonzumované dokumenty budou označeny tagem pro přichozí" + +#: documents/models.py:94 +msgid "tag" +msgstr "tag" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "tagy" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "typ dokumentu" + +#: documents/models.py:102 +msgid "document types" +msgstr "typy dokumentu" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Nešifrované" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Šifrované pomocí GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titulek" + +#: documents/models.py:137 +msgid "content" +msgstr "obsah" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Nezpracovaná, pouze textová data dokumentu. Toto pole je používáno především pro vyhledávání." + +#: documents/models.py:144 +msgid "mime type" +msgstr "mime typ" + +#: documents/models.py:155 +msgid "checksum" +msgstr "kontrolní součet" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Kontrolní součet původního dokumentu" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "kontrolní součet archivu" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Kontrolní součet archivovaného dokumentu." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "vytvořeno" + +#: documents/models.py:176 +msgid "modified" +msgstr "upraveno" + +#: documents/models.py:180 +msgid "storage type" +msgstr "typ úložiště" + +#: documents/models.py:188 +msgid "added" +msgstr "přidáno" + +#: documents/models.py:192 +msgid "filename" +msgstr "název souboru" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Aktuální název souboru v úložišti" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "sériové číslo archivu" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Pozice dokumentu ve vašem archivu fyzických dokumentů" + +#: documents/models.py:223 +msgid "document" +msgstr "dokument" + +#: documents/models.py:224 +msgid "documents" +msgstr "dokumenty" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "informace" + +#: documents/models.py:313 +msgid "warning" +msgstr "varování" + +#: documents/models.py:314 +msgid "error" +msgstr "chyba" + +#: documents/models.py:315 +msgid "critical" +msgstr "kritická" + +#: documents/models.py:319 +msgid "group" +msgstr "skupina" + +#: documents/models.py:322 +msgid "message" +msgstr "zpráva" + +#: documents/models.py:325 +msgid "level" +msgstr "úroveň" + +#: documents/models.py:332 +msgid "log" +msgstr "záznam" + +#: documents/models.py:333 +msgid "logs" +msgstr "záznamy" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "uložený pohled" + +#: documents/models.py:345 +msgid "saved views" +msgstr "uložené pohledy" + +#: documents/models.py:348 +msgid "user" +msgstr "uživatel" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "zobrazit v dashboardu" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "zobrazit v postranním menu" + +#: documents/models.py:361 +msgid "sort field" +msgstr "pole na řazení" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "třídit opačně" + +#: documents/models.py:373 +msgid "title contains" +msgstr "titulek obsahuje" + +#: documents/models.py:374 +msgid "content contains" +msgstr "obsah obsahuje" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN je" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "korespondent je" + +#: documents/models.py:377 +msgid "document type is" +msgstr "typ dokumentu je" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "je v příchozích" + +#: documents/models.py:379 +msgid "has tag" +msgstr "má tag" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "má jakýkoliv tag" + +#: documents/models.py:381 +msgid "created before" +msgstr "vytvořeno před" + +#: documents/models.py:382 +msgid "created after" +msgstr "vytvořeno po" + +#: documents/models.py:383 +msgid "created year is" +msgstr "rok vytvoření je" + +#: documents/models.py:384 +msgid "created month is" +msgstr "měsíc vytvoření je" + +#: documents/models.py:385 +msgid "created day is" +msgstr "den vytvoření je" + +#: documents/models.py:386 +msgid "added before" +msgstr "přidáno před" + +#: documents/models.py:387 +msgid "added after" +msgstr "přidáno po" + +#: documents/models.py:388 +msgid "modified before" +msgstr "upraveno před" + +#: documents/models.py:389 +msgid "modified after" +msgstr "upraveno po" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "nemá tag" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "" + +#: documents/models.py:394 +msgid "more like this" +msgstr "" + +#: documents/models.py:405 +msgid "rule type" +msgstr "typ pravidla" + +#: documents/models.py:409 +msgid "value" +msgstr "hodnota" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "filtrovací pravidlo" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "filtrovací pravidla" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Typ souboru %(type)s není podporován" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng se načítá..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Odhlášeno od Paperless-ng" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Byli jste úspěšně odhlášeni. Nashledanou!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Přihlašte se znovu" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng přihlášení" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Prosím přihlaste se." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Vaše uživatelské jméno a heslo se neshodují. Prosím, zkuste to znovu." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Uživatelské jméno" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Heslo" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Přihlásit se" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Němčina" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Holandština" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Francouzština" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Správa Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtr" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless zpracuje pouze emaily které odpovídají VŠEM níže zadaným filtrům." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Akce" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Akce provedena na emailu. Tato akce je provedena jen pokud byly dokumenty zkonzumovány z emailu. Emaily bez příloh zůstanou nedotčeny." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadata" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Automaticky přiřadit metadata dokumentům zkonzumovaných z tohoto pravidla. Pokud zde nepřiřadíte tagy, typy nebo korespondenty, paperless stále zpracuje všechna shodující-se pravidla které jste definovali." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless pošta" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "emailový účet" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "emailové účty" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Žádné šifrování" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Používat SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Používat STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP port" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Toto je většinou 143 pro nešifrovaná připojení/připojení používající STARTTLS a 993 pro SSL připojení." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP bezpečnost" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "uživatelské jméno" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "heslo" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "mailové pravidlo" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "mailová pravidla" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Zpracovávat jen přílohy" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Zpracovat všechny soubory, včetně vložených příloh" + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Označit jako přečtené, nezpracovávat přečtené emaily" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Označit email, nezpracovávat označené emaily" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Přesunout do specifikované složky" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Odstranit" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Použít předmět jako titulek" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Použít název souboru u přílohy jako titulek" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Nepřiřazovat korespondenta" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Použít emailovou adresu" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Použít jméno (nebo emailovou adresu pokud jméno není dostupné)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Použít korespondenta vybraného níže" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "pořadí" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "účet" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "složka" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrovat z" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "název filtru" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "tělo filtru" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "název souboru u přílohy filtru" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Konzumovat jen dokumenty které přesně odpovídají tomuto názvu souboru pokud specifikováno. Zástupné znaky jako *.pdf nebo *invoice* jsou povoleny. Nezáleží na velikosti písmen." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "maximální stáří" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Specifikováno ve dnech." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "typ přílohy" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Vložené přílohy zahrnují vložené obrázky, takže je nejlepší tuto možnost kombinovat s filtrem na název souboru" + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "akce" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parametr akce" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "nastavit titulek z" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "přiřadit tento tag" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "přiřadit tento typ dokumentu" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "přiřadit korespondenta z" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "přiřadit tohoto korespondenta" + diff --git a/src/locale/de_DE/LC_MESSAGES/django.po b/src/locale/de_DE/LC_MESSAGES/django.po new file mode 100644 index 000000000..4a41c3c05 --- /dev/null +++ b/src/locale/de_DE/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-07-05 11:17\n" +"Last-Translator: \n" +"Language-Team: German\n" +"Language: de_DE\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: de\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokumente" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Irgendein Wort" + +#: documents/models.py:33 +msgid "All words" +msgstr "Alle Wörter" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Exakte Übereinstimmung" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Regular expression / Reguläre Ausdrücke" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Ungenaues Wort" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatisch" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "Name" + +#: documents/models.py:45 +msgid "match" +msgstr "Zuweisungsmuster" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "Zuweisungsalgorithmus" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "Groß-/Kleinschreibung irrelevant" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "Korrespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "Korrespondenten" + +#: documents/models.py:81 +msgid "color" +msgstr "Farbe" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "Posteingangs-Tag" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Markiert das Tag als Posteingangs-Tag. Neue Dokumente werden immer mit diesem Tag versehen." + +#: documents/models.py:94 +msgid "tag" +msgstr "Tag" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "Tags" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "Dokumenttyp" + +#: documents/models.py:102 +msgid "document types" +msgstr "Dokumenttypen" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Nicht verschlüsselt" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Verschlüsselt mit GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "Titel" + +#: documents/models.py:137 +msgid "content" +msgstr "Inhalt" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Der Inhalt des Dokuments in Textform. Dieses Feld wird primär für die Suche verwendet." + +#: documents/models.py:144 +msgid "mime type" +msgstr "MIME-Typ" + +#: documents/models.py:155 +msgid "checksum" +msgstr "Prüfsumme" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Die Prüfsumme des originalen Dokuments." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "Archiv-Prüfsumme" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Die Prüfsumme des archivierten Dokuments." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "Ausgestellt" + +#: documents/models.py:176 +msgid "modified" +msgstr "Geändert" + +#: documents/models.py:180 +msgid "storage type" +msgstr "Speichertyp" + +#: documents/models.py:188 +msgid "added" +msgstr "Hinzugefügt" + +#: documents/models.py:192 +msgid "filename" +msgstr "Dateiname" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Aktueller Dateiname im Datenspeicher" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "Archiv-Dateiname" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Aktueller Dateiname im Archiv" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "Archiv-Seriennummer" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Die Position dieses Dokuments in Ihrem physischen Dokumentenarchiv." + +#: documents/models.py:223 +msgid "document" +msgstr "Dokument" + +#: documents/models.py:224 +msgid "documents" +msgstr "Dokumente" + +#: documents/models.py:311 +msgid "debug" +msgstr "Debug" + +#: documents/models.py:312 +msgid "information" +msgstr "Information" + +#: documents/models.py:313 +msgid "warning" +msgstr "Warnung" + +#: documents/models.py:314 +msgid "error" +msgstr "Fehler" + +#: documents/models.py:315 +msgid "critical" +msgstr "Kritisch" + +#: documents/models.py:319 +msgid "group" +msgstr "Gruppe" + +#: documents/models.py:322 +msgid "message" +msgstr "Nachricht" + +#: documents/models.py:325 +msgid "level" +msgstr "Level" + +#: documents/models.py:332 +msgid "log" +msgstr "Protokoll" + +#: documents/models.py:333 +msgid "logs" +msgstr "Protokoll" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "Gespeicherte Ansicht" + +#: documents/models.py:345 +msgid "saved views" +msgstr "Gespeicherte Ansichten" + +#: documents/models.py:348 +msgid "user" +msgstr "Benutzer" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "Auf Startseite zeigen" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "In Seitenleiste zeigen" + +#: documents/models.py:361 +msgid "sort field" +msgstr "Sortierfeld" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "Umgekehrte Sortierung" + +#: documents/models.py:373 +msgid "title contains" +msgstr "Titel enthält" + +#: documents/models.py:374 +msgid "content contains" +msgstr "Inhalt enthält" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN ist" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "Korrespondent ist" + +#: documents/models.py:377 +msgid "document type is" +msgstr "Dokumenttyp ist" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "Ist im Posteingang" + +#: documents/models.py:379 +msgid "has tag" +msgstr "Hat Tag" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "Hat irgendein Tag" + +#: documents/models.py:381 +msgid "created before" +msgstr "Ausgestellt vor" + +#: documents/models.py:382 +msgid "created after" +msgstr "Ausgestellt nach" + +#: documents/models.py:383 +msgid "created year is" +msgstr "Ausgestellt im Jahr" + +#: documents/models.py:384 +msgid "created month is" +msgstr "Ausgestellt im Monat" + +#: documents/models.py:385 +msgid "created day is" +msgstr "Ausgestellt am Tag" + +#: documents/models.py:386 +msgid "added before" +msgstr "Hinzugefügt vor" + +#: documents/models.py:387 +msgid "added after" +msgstr "Hinzugefügt nach" + +#: documents/models.py:388 +msgid "modified before" +msgstr "Geändert vor" + +#: documents/models.py:389 +msgid "modified after" +msgstr "Geändert nach" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "Hat nicht folgendes Tag" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "Dokument hat keine ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "Titel oder Inhalt enthält" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "Volltextsuche" + +#: documents/models.py:394 +msgid "more like this" +msgstr "Ähnliche Dokumente" + +#: documents/models.py:405 +msgid "rule type" +msgstr "Regeltyp" + +#: documents/models.py:409 +msgid "value" +msgstr "Wert" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "Filterregel" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "Filterregeln" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Ungültiger regulärer Ausdruck: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Ungültige Farbe." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Dateityp %(type)s nicht unterstützt" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng wird geladen..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng abgemeldet" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Sie wurden erfolgreich abgemeldet. Auf Wiedersehen!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Erneut anmelden" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng Anmeldung" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Bitte melden Sie sich an." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Ihr Benutzername und Kennwort stimmen nicht überein. Bitte versuchen Sie es erneut." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Benutzername" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Kennwort" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Anmelden" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Englisch (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Englisch (UK)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Deutsch" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Niederländisch" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Französisch" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugiesisch (Brasilien)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugiesisch" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italienisch" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumänisch" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russisch" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spanisch" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polnisch" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Schwedisch" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng Administration" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Authentifizierung" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Erweiterte Einstellungen" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filter" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless wird nur E-Mails verarbeiten, für die alle der hier angegebenen Filter zutreffen." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Aktionen" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Die Aktion, die auf E-Mails angewendet werden soll. Diese Aktion wird nur auf E-Mails angewendet, aus denen Anhänge verarbeitet wurden. E-Mails ohne Anhänge werden vollständig ignoriert." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadaten" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Folgende Metadaten werden Dokumenten dieser Regel automatisch zugewiesen. Wenn Sie hier nichts auswählen wird Paperless weiterhin alle Zuweisungsalgorithmen ausführen und Metadaten auf Basis des Dokumentinhalts zuweisen." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless E-Mail" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "E-Mail-Konto" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "E-Mail-Konten" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Keine Verschlüsselung" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "SSL benutzen" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "STARTTLS benutzen" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP-Server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP-Port" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Dies ist in der Regel 143 für unverschlüsselte und STARTTLS-Verbindungen und 993 für SSL-Verbindungen." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP-Sicherheit" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "Benutzername" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "Kennwort" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Zeichensatz" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Der Zeichensatz, der bei der Kommunikation mit dem Mailserver verwendet werden soll, wie z.B. 'UTF-8' oder 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "E-Mail-Regel" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "E-Mail-Regeln" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Nur Anhänge verarbeiten." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Alle Dateien verarbeiten, auch 'inline'-Anhänge." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Als gelesen markieren, gelesene E-Mails nicht verarbeiten" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Als wichtig markieren, markierte E-Mails nicht verarbeiten" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "In angegebenen Ordner verschieben" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Löschen" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Betreff als Titel verwenden" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Dateiname des Anhangs als Titel verwenden" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Keinen Korrespondenten zuweisen" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "E-Mail-Adresse benutzen" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Absendername benutzen (oder E-Mail-Adressen, wenn nicht verfügbar)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Nachfolgend ausgewählten Korrespondent verwenden" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "Reihenfolge" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "Konto" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "Ordner" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Unterordner müssen durch Punkte getrennt werden." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "Absender filtern" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "Betreff filtern" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "Nachrichteninhalt filtern" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "Anhang-Dateiname filtern" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Wenn angegeben werden nur Dateien verarbeitet, die diesem Dateinamen exakt entsprechen. Platzhalter wie *.pdf oder *rechnung* sind erlaubt. Groß- und Kleinschreibung ist irrelevant." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "Maximales Alter" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Angegeben in Tagen." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "Dateianhangstyp" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "'Inline'-Anhänge schließen eingebettete Bilder mit ein, daher sollte diese Einstellung mit einem Dateinamenfilter kombiniert werden." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "Aktion" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "Parameter für Aktion" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Zusätzlicher Parameter für die oben ausgewählte Aktion, zum Beispiel der Zielordner für die Aktion \"In angegebenen Ordner verschieben\". Unterordner müssen durch Punkte getrennt werden." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "Titel zuweisen von" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "Dieses Tag zuweisen" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "Diesen Dokumenttyp zuweisen" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "Korrespondent zuweisen von" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "Diesen Korrespondent zuweisen" + diff --git a/src/locale/en_GB/LC_MESSAGES/django.po b/src/locale/en_GB/LC_MESSAGES/django.po new file mode 100644 index 000000000..c36d321d1 --- /dev/null +++ b/src/locale/en_GB/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-06-19 21:20\n" +"Last-Translator: \n" +"Language-Team: English, United Kingdom\n" +"Language: en_GB\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: en-GB\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documents" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Any word" + +#: documents/models.py:33 +msgid "All words" +msgstr "All words" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Exact match" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Regular expression" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Fuzzy word" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatic" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "name" + +#: documents/models.py:45 +msgid "match" +msgstr "match" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "matching algorithm" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "is insensitive" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "correspondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "correspondents" + +#: documents/models.py:81 +msgid "color" +msgstr "colour" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "is inbox tag" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." + +#: documents/models.py:94 +msgid "tag" +msgstr "tag" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "tags" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "document type" + +#: documents/models.py:102 +msgid "document types" +msgstr "document types" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Unencrypted" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Encrypted with GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "title" + +#: documents/models.py:137 +msgid "content" +msgstr "content" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "The raw, text-only data of the document. This field is primarily used for searching." + +#: documents/models.py:144 +msgid "mime type" +msgstr "mime type" + +#: documents/models.py:155 +msgid "checksum" +msgstr "checksum" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "The checksum of the original document." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "archive checksum" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "The checksum of the archived document." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "created" + +#: documents/models.py:176 +msgid "modified" +msgstr "modified" + +#: documents/models.py:180 +msgid "storage type" +msgstr "storage type" + +#: documents/models.py:188 +msgid "added" +msgstr "added" + +#: documents/models.py:192 +msgid "filename" +msgstr "filename" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Current filename in storage" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "archive filename" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Current archive filename in storage" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "archive serial number" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "The position of this document in your physical document archive." + +#: documents/models.py:223 +msgid "document" +msgstr "document" + +#: documents/models.py:224 +msgid "documents" +msgstr "documents" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "information" + +#: documents/models.py:313 +msgid "warning" +msgstr "warning" + +#: documents/models.py:314 +msgid "error" +msgstr "error" + +#: documents/models.py:315 +msgid "critical" +msgstr "critical" + +#: documents/models.py:319 +msgid "group" +msgstr "group" + +#: documents/models.py:322 +msgid "message" +msgstr "message" + +#: documents/models.py:325 +msgid "level" +msgstr "level" + +#: documents/models.py:332 +msgid "log" +msgstr "log" + +#: documents/models.py:333 +msgid "logs" +msgstr "logs" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "saved view" + +#: documents/models.py:345 +msgid "saved views" +msgstr "saved views" + +#: documents/models.py:348 +msgid "user" +msgstr "user" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "show on dashboard" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "show in sidebar" + +#: documents/models.py:361 +msgid "sort field" +msgstr "sort field" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "sort reverse" + +#: documents/models.py:373 +msgid "title contains" +msgstr "title contains" + +#: documents/models.py:374 +msgid "content contains" +msgstr "content contains" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN is" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "correspondent is" + +#: documents/models.py:377 +msgid "document type is" +msgstr "document type is" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "is in inbox" + +#: documents/models.py:379 +msgid "has tag" +msgstr "has tag" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "has any tag" + +#: documents/models.py:381 +msgid "created before" +msgstr "created before" + +#: documents/models.py:382 +msgid "created after" +msgstr "created after" + +#: documents/models.py:383 +msgid "created year is" +msgstr "created year is" + +#: documents/models.py:384 +msgid "created month is" +msgstr "created month is" + +#: documents/models.py:385 +msgid "created day is" +msgstr "created day is" + +#: documents/models.py:386 +msgid "added before" +msgstr "added before" + +#: documents/models.py:387 +msgid "added after" +msgstr "added after" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modified before" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modified after" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "does not have tag" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "does not have ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "title or content contains" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "fulltext query" + +#: documents/models.py:394 +msgid "more like this" +msgstr "more like this" + +#: documents/models.py:405 +msgid "rule type" +msgstr "rule type" + +#: documents/models.py:409 +msgid "value" +msgstr "value" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "filter rule" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "filter rules" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Invalid regular expression: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Invalid colour." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "File type %(type)s not supported" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng is loading..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng signed out" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "You have been successfully logged out. Bye!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Sign in again" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng sign in" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Please sign in." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Your username and password didn't match. Please try again." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Username" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Password" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Sign in" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "English (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "English (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "German" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Dutch" + +#: paperless/settings.py:307 +msgid "French" +msgstr "French" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portuguese (Brazil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portuguese" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italian" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Romanian" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russian" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spanish" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polish" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Swedish" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng administration" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Authentication" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Advanced settings" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filter" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless will only process mails that match ALL of the filters given below." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Actions" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadata" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless mail" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "mail account" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "mail accounts" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "No encryption" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Use SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Use STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP port" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP security" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "username" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "password" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "mail rule" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "mail rules" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Only process attachments." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Process all files, including 'inline' attachments." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Mark as read, don't process read mails" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Flag the mail, don't process flagged mails" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Move to specified folder" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Delete" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Use subject as title" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Use attachment filename as title" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Do not assign a correspondent" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Use mail address" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Use name (or mail address if not available)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Use correspondent selected below" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "order" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "account" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "folder" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filter from" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filter subject" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filter body" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filter attachment filename" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "maximum age" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Specified in days." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "attachment type" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Inline attachments include embedded images, so it's best to combine this option with a filename filter." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "action" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "action parameter" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "assign title from" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "assign this tag" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "assign this document type" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "assign correspondent from" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "assign this correspondent" + diff --git a/src/locale/en_US/LC_MESSAGES/django.po b/src/locale/en_US/LC_MESSAGES/django.po new file mode 100644 index 000000000..f152691ba --- /dev/null +++ b/src/locale/en_US/LC_MESSAGES/django.po @@ -0,0 +1,718 @@ +# SOME DESCRIPTIVE TITLE. +# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER +# This file is distributed under the same license as the PACKAGE package. +# FIRST AUTHOR , YEAR. +# +#, fuzzy +msgid "" +msgstr "" +"Project-Id-Version: PACKAGE VERSION\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" +"Last-Translator: FULL NAME \n" +"Language-Team: LANGUAGE \n" +"Language: \n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "" + +#: documents/models.py:32 +msgid "Any word" +msgstr "" + +#: documents/models.py:33 +msgid "All words" +msgstr "" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "" + +#: documents/models.py:45 +msgid "match" +msgstr "" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "" + +#: documents/models.py:81 +msgid "color" +msgstr "" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "" + +#: documents/models.py:89 +msgid "" +"Marks this tag as an inbox tag: All newly consumed documents will be tagged " +"with inbox tags." +msgstr "" + +#: documents/models.py:94 +msgid "tag" +msgstr "" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "" + +#: documents/models.py:102 +msgid "document types" +msgstr "" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "" + +#: documents/models.py:124 +msgid "title" +msgstr "" + +#: documents/models.py:137 +msgid "content" +msgstr "" + +#: documents/models.py:139 +msgid "" +"The raw, text-only data of the document. This field is primarily used for " +"searching." +msgstr "" + +#: documents/models.py:144 +msgid "mime type" +msgstr "" + +#: documents/models.py:155 +msgid "checksum" +msgstr "" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "" + +#: documents/models.py:176 +msgid "modified" +msgstr "" + +#: documents/models.py:180 +msgid "storage type" +msgstr "" + +#: documents/models.py:188 +msgid "added" +msgstr "" + +#: documents/models.py:192 +msgid "filename" +msgstr "" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "" + +#: documents/models.py:223 +msgid "document" +msgstr "" + +#: documents/models.py:224 +msgid "documents" +msgstr "" + +#: documents/models.py:311 +msgid "debug" +msgstr "" + +#: documents/models.py:312 +msgid "information" +msgstr "" + +#: documents/models.py:313 +msgid "warning" +msgstr "" + +#: documents/models.py:314 +msgid "error" +msgstr "" + +#: documents/models.py:315 +msgid "critical" +msgstr "" + +#: documents/models.py:319 +msgid "group" +msgstr "" + +#: documents/models.py:322 +msgid "message" +msgstr "" + +#: documents/models.py:325 +msgid "level" +msgstr "" + +#: documents/models.py:332 +msgid "log" +msgstr "" + +#: documents/models.py:333 +msgid "logs" +msgstr "" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "" + +#: documents/models.py:345 +msgid "saved views" +msgstr "" + +#: documents/models.py:348 +msgid "user" +msgstr "" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "" + +#: documents/models.py:361 +msgid "sort field" +msgstr "" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "" + +#: documents/models.py:373 +msgid "title contains" +msgstr "" + +#: documents/models.py:374 +msgid "content contains" +msgstr "" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "" + +#: documents/models.py:377 +msgid "document type is" +msgstr "" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "" + +#: documents/models.py:379 +msgid "has tag" +msgstr "" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "" + +#: documents/models.py:381 +msgid "created before" +msgstr "" + +#: documents/models.py:382 +msgid "created after" +msgstr "" + +#: documents/models.py:383 +msgid "created year is" +msgstr "" + +#: documents/models.py:384 +msgid "created month is" +msgstr "" + +#: documents/models.py:385 +msgid "created day is" +msgstr "" + +#: documents/models.py:386 +msgid "added before" +msgstr "" + +#: documents/models.py:387 +msgid "added after" +msgstr "" + +#: documents/models.py:388 +msgid "modified before" +msgstr "" + +#: documents/models.py:389 +msgid "modified after" +msgstr "" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "" + +#: documents/models.py:394 +msgid "more like this" +msgstr "" + +#: documents/models.py:405 +msgid "rule type" +msgstr "" + +#: documents/models.py:409 +msgid "value" +msgstr "" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "" + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "" + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:305 +msgid "German" +msgstr "" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "" + +#: paperless/settings.py:307 +msgid "French" +msgstr "" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "" + +#: paperless_mail/admin.py:39 +msgid "" +"Paperless will only process mails that match ALL of the filters given below." +msgstr "" + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "" + +#: paperless_mail/admin.py:51 +msgid "" +"The action applied to the mail. This action is only performed when documents " +"were consumed from the mail. Mails without attachments will remain entirely " +"untouched." +msgstr "" + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "" + +#: paperless_mail/admin.py:60 +msgid "" +"Assign metadata to documents consumed from this rule automatically. If you " +"do not assign tags, types or correspondents here, paperless will still " +"process all matching rules that you have defined." +msgstr "" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "" + +#: paperless_mail/models.py:36 +msgid "" +"This is usually 143 for unencrypted and STARTTLS connections, and 993 for " +"SSL connections." +msgstr "" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "" +"The character set to use when communicating with the mail server, such as " +"'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "" + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "" + +#: paperless_mail/models.py:150 +msgid "" +"Only consume documents which entirely match this filename if specified. " +"Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "" + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "" + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "" + +#: paperless_mail/models.py:164 +msgid "" +"Inline attachments include embedded images, so it's best to combine this " +"option with a filename filter." +msgstr "" + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "" + +#: paperless_mail/models.py:177 +msgid "" +"Additional parameter for the action selected above, i.e., the target folder " +"of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "" diff --git a/src/locale/es_ES/LC_MESSAGES/django.po b/src/locale/es_ES/LC_MESSAGES/django.po new file mode 100644 index 000000000..00f9dc769 --- /dev/null +++ b/src/locale/es_ES/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-07-29 20:57\n" +"Last-Translator: \n" +"Language-Team: Spanish\n" +"Language: es_ES\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: es-ES\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documentos" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Cualquier palabra" + +#: documents/models.py:33 +msgid "All words" +msgstr "Todas las palabras" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Coincidencia exacta" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Expresión regular" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Palabra borrosa" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automático" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nombre" + +#: documents/models.py:45 +msgid "match" +msgstr "coincidencia" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "Algoritmo de coincidencia" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "es insensible" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "interlocutor" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "interlocutores" + +#: documents/models.py:81 +msgid "color" +msgstr "color" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "es etiqueta de bandeja" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marca esta etiqueta como una etiqueta de bandeja: todos los documentos recién consumidos serán etiquetados con las etiquetas de bandeja." + +#: documents/models.py:94 +msgid "tag" +msgstr "etiqueta" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "etiquetas" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "tipo de documento" + +#: documents/models.py:102 +msgid "document types" +msgstr "tipos de documento" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Sin cifrar" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Cifrado con GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "título" + +#: documents/models.py:137 +msgid "content" +msgstr "contenido" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Los datos de texto en bruto del documento. Este campo se utiliza principalmente para las búsquedas." + +#: documents/models.py:144 +msgid "mime type" +msgstr "tipo MIME" + +#: documents/models.py:155 +msgid "checksum" +msgstr "Cadena de verificación" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "La cadena de verificación del documento original." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "cadena de comprobación del archivo" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "La cadena de verificación del documento archivado." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "creado" + +#: documents/models.py:176 +msgid "modified" +msgstr "modificado" + +#: documents/models.py:180 +msgid "storage type" +msgstr "tipo de almacenamiento" + +#: documents/models.py:188 +msgid "added" +msgstr "añadido" + +#: documents/models.py:192 +msgid "filename" +msgstr "nombre del archivo" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nombre de archivo actual en disco" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nombre de archivo" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Nombre de archivo actual en disco" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "número de serie del archivo" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Posición de este documento en tu archivo físico de documentos." + +#: documents/models.py:223 +msgid "document" +msgstr "documento" + +#: documents/models.py:224 +msgid "documents" +msgstr "documentos" + +#: documents/models.py:311 +msgid "debug" +msgstr "depuración" + +#: documents/models.py:312 +msgid "information" +msgstr "información" + +#: documents/models.py:313 +msgid "warning" +msgstr "alerta" + +#: documents/models.py:314 +msgid "error" +msgstr "error" + +#: documents/models.py:315 +msgid "critical" +msgstr "crítico" + +#: documents/models.py:319 +msgid "group" +msgstr "grupo" + +#: documents/models.py:322 +msgid "message" +msgstr "mensaje" + +#: documents/models.py:325 +msgid "level" +msgstr "nivel" + +#: documents/models.py:332 +msgid "log" +msgstr "log" + +#: documents/models.py:333 +msgid "logs" +msgstr "logs" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "vista guardada" + +#: documents/models.py:345 +msgid "saved views" +msgstr "vistas guardadas" + +#: documents/models.py:348 +msgid "user" +msgstr "usuario" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "mostrar en el panel de control" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "mostrar en barra lateral" + +#: documents/models.py:361 +msgid "sort field" +msgstr "campo de ordenación" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "ordenar al revés" + +#: documents/models.py:373 +msgid "title contains" +msgstr "el título contiene" + +#: documents/models.py:374 +msgid "content contains" +msgstr "el contenido contiene" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN es" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "interlocutor es" + +#: documents/models.py:377 +msgid "document type is" +msgstr "el tipo de documento es" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "está en la bandeja de entrada" + +#: documents/models.py:379 +msgid "has tag" +msgstr "tiene la etiqueta" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "tiene cualquier etiqueta" + +#: documents/models.py:381 +msgid "created before" +msgstr "creado antes" + +#: documents/models.py:382 +msgid "created after" +msgstr "creado después" + +#: documents/models.py:383 +msgid "created year is" +msgstr "el año de creación es" + +#: documents/models.py:384 +msgid "created month is" +msgstr "el mes de creación es" + +#: documents/models.py:385 +msgid "created day is" +msgstr "creado el día" + +#: documents/models.py:386 +msgid "added before" +msgstr "agregado antes de" + +#: documents/models.py:387 +msgid "added after" +msgstr "agregado después de" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modificado después de" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modificado antes de" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "no tiene la etiqueta" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "no tiene ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "el título o cuerpo contiene" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "consulta de texto completo" + +#: documents/models.py:394 +msgid "more like this" +msgstr "más contenido similar" + +#: documents/models.py:405 +msgid "rule type" +msgstr "tipo de regla" + +#: documents/models.py:409 +msgid "value" +msgstr "valor" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "regla de filtrado" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "reglas de filtrado" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Expresión irregular inválida: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Color inválido." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Tipo de fichero %(type)s no suportado" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng está cargándose..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng Sesión cerrada" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Has cerrado la sesión satisfactoriamente. ¡Adiós!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Iniciar sesión de nuevo" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng Iniciar sesión" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Por favor, inicie sesión" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Tu usuario y contraseña no coinciden. Inténtalo de nuevo." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Usuario" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Contraseña" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Iniciar sesión" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Inglés (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Inglés (Gran Bretaña)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Alemán" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Alemán" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Francés" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugués (Brasil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugués" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiano" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumano" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Ruso" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Español" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polaco" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Sueco" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng Administración" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Autentificación" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Configuración avanzada" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtro" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless solo procesará los correos que coincidan con TODOS los filtros escritos abajo." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Acciones" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "La acción se aplicó al correo. Esta acción sólo se realiza cuando los documentos se consumen desde el correo. Los correos sin archivos adjuntos permanecerán totalmente intactos." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadatos" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Asignar metadatos a documentos consumidos por esta regla automáticamente. Si no asigna etiquetas, tipos o interlocutores aquí, paperless procesará igualmente todas las reglas que haya definido." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Correo Paperless" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "cuenta de correo" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "cuentas de correo" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Sin encriptar" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Usar SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Usar STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Servidor IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Puerto IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Normalmente 143 para conexiones sin encriptar y STARTTLS, y 993 para conexiones SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Seguridad IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "usuario" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "contraseña" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "conjunto de caracteres" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "El conjunto de caracteres a usar al comunicarse con el servidor de correo, como 'UTF-8' o 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "regla de correo" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "reglas de correo" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Solo procesar ficheros adjuntos." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Procesar todos los ficheros, incluyendo ficheros 'incrustados'." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Marcar como leído, no procesar archivos leídos" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Marcar el correo, no procesar correos marcados" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Mover a carpeta específica" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Borrar" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Usar asunto como título" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Usar nombre del fichero adjunto como título" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "No asignar interlocutor" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Usar dirección de correo" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Usar nombre (o dirección de correo sino está disponible)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Usar el interlocutor seleccionado a continuación" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "orden" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "cuenta" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "carpeta" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Las subcarpetas deben estar separadas por puntos." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrar desde" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrar asunto" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrar cuerpo" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrar nombre del fichero adjunto" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Sólo consumirá documentos que coincidan completamente con este nombre de archivo si se especifica. Se permiten comodines como *.pdf o *factura*. No diferencia mayúsculas." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "antigüedad máxima" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Especificado en días." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "tipo de fichero adjunto" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Adjuntos incrustados incluyen imágenes, por lo que es mejor combina resta opción un filtro de nombre de fichero." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "acción" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parámetro de acción" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Parámetro adicional para la acción seleccionada arriba. Ej. la carpeta de destino de la acción \"mover a carpeta\". Las subcarpetas deben estar separadas por puntos." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "asignar título desde" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "asignar esta etiqueta" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "asignar este tipo de documento" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "asignar interlocutor desde" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "asignar este interlocutor" + diff --git a/src/locale/fr_FR/LC_MESSAGES/django.po b/src/locale/fr_FR/LC_MESSAGES/django.po new file mode 100644 index 000000000..3ec137cc1 --- /dev/null +++ b/src/locale/fr_FR/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-17 13:13\n" +"Last-Translator: \n" +"Language-Team: French\n" +"Language: fr_FR\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n > 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: fr\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documents" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Un des mots" + +#: documents/models.py:33 +msgid "All words" +msgstr "Tous les mots" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Concordance exacte" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Expression régulière" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Mot approximatif" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatique" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nom" + +#: documents/models.py:45 +msgid "match" +msgstr "rapprochement" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algorithme de rapprochement" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "est insensible à la casse" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "correspondant" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "correspondants" + +#: documents/models.py:81 +msgid "color" +msgstr "couleur" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "est une étiquette de boîte de réception" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marque cette étiquette comme étiquette de boîte de réception : ces étiquettes sont affectées à tous les documents nouvellement traités." + +#: documents/models.py:94 +msgid "tag" +msgstr "étiquette" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "étiquettes" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "type de document" + +#: documents/models.py:102 +msgid "document types" +msgstr "types de document" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Non chiffré" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Chiffré avec GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titre" + +#: documents/models.py:137 +msgid "content" +msgstr "contenu" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Les données brutes du document, en format texte uniquement. Ce champ est principalement utilisé pour la recherche." + +#: documents/models.py:144 +msgid "mime type" +msgstr "type mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "somme de contrôle" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "La somme de contrôle du document original." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "somme de contrôle de l'archive" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "La somme de contrôle du document archivé." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "créé le" + +#: documents/models.py:176 +msgid "modified" +msgstr "modifié" + +#: documents/models.py:180 +msgid "storage type" +msgstr "forme d'enregistrement :" + +#: documents/models.py:188 +msgid "added" +msgstr "date d'ajout" + +#: documents/models.py:192 +msgid "filename" +msgstr "nom du fichier" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nom du fichier courant en base de données" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nom de fichier de l'archive" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Nom du fichier d'archive courant en base de données" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "numéro de série de l'archive" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Le classement de ce document dans votre archive de documents physiques." + +#: documents/models.py:223 +msgid "document" +msgstr "document" + +#: documents/models.py:224 +msgid "documents" +msgstr "documents" + +#: documents/models.py:311 +msgid "debug" +msgstr "débogage" + +#: documents/models.py:312 +msgid "information" +msgstr "information" + +#: documents/models.py:313 +msgid "warning" +msgstr "avertissement" + +#: documents/models.py:314 +msgid "error" +msgstr "erreur" + +#: documents/models.py:315 +msgid "critical" +msgstr "critique" + +#: documents/models.py:319 +msgid "group" +msgstr "groupe" + +#: documents/models.py:322 +msgid "message" +msgstr "message" + +#: documents/models.py:325 +msgid "level" +msgstr "niveau" + +#: documents/models.py:332 +msgid "log" +msgstr "journal" + +#: documents/models.py:333 +msgid "logs" +msgstr "journaux" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "vue enregistrée" + +#: documents/models.py:345 +msgid "saved views" +msgstr "vues enregistrées" + +#: documents/models.py:348 +msgid "user" +msgstr "utilisateur" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "montrer sur le tableau de bord" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "montrer dans la barre latérale" + +#: documents/models.py:361 +msgid "sort field" +msgstr "champ de tri" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "tri inverse" + +#: documents/models.py:373 +msgid "title contains" +msgstr "le titre contient" + +#: documents/models.py:374 +msgid "content contains" +msgstr "le contenu contient" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "le NSA est" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "le correspondant est" + +#: documents/models.py:377 +msgid "document type is" +msgstr "le type de document est" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "est dans la boîte de réception" + +#: documents/models.py:379 +msgid "has tag" +msgstr "porte l'étiquette" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "porte l'une des étiquettes" + +#: documents/models.py:381 +msgid "created before" +msgstr "créé avant" + +#: documents/models.py:382 +msgid "created after" +msgstr "créé après" + +#: documents/models.py:383 +msgid "created year is" +msgstr "l'année de création est" + +#: documents/models.py:384 +msgid "created month is" +msgstr "le mois de création est" + +#: documents/models.py:385 +msgid "created day is" +msgstr "le jour de création est" + +#: documents/models.py:386 +msgid "added before" +msgstr "ajouté avant" + +#: documents/models.py:387 +msgid "added after" +msgstr "ajouté après" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modifié avant" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modifié après" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "ne porte pas d'étiquette" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "ne porte pas de NSA" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "le titre ou le contenu contient" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "recherche en texte intégral" + +#: documents/models.py:394 +msgid "more like this" +msgstr "documents relatifs" + +#: documents/models.py:405 +msgid "rule type" +msgstr "type de règle" + +#: documents/models.py:409 +msgid "value" +msgstr "valeur" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "règle de filtrage" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "règles de filtrage" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Expression régulière incorrecte : %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Couleur incorrecte." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Type de fichier %(type)s non pris en charge" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng est en cours de chargement..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Déconnecté de Paperless-ng" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Vous avez été déconnecté avec succès. Au revoir !" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Se reconnecter" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Connexion à Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Veuillez vous connecter." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Votre nom d'utilisateur et votre mot de passe ne correspondent pas. Veuillez réessayer." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Nom d'utilisateur" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Mot de passe" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "S'identifier" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Anglais (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Anglais (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Allemand" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Néerlandais" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Français" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugais (Brésil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugais" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italien" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Roumain" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russe" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Espagnol" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polonais" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Suédois" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Administration de Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Authentification" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Paramètres avancés" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtrage" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless-ng ne traitera que les courriers qui correspondent à TOUS les filtres ci-dessous." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Actions" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Action appliquée au courriel. Cette action n'est exécutée que lorsque les documents ont été traités depuis des courriels. Les courriels sans pièces jointes demeurent totalement inchangés." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Métadonnées" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Affecter automatiquement des métadonnées aux documents traités à partir de cette règle. Si vous n'affectez pas d'étiquette, de type ou de correspondant ici, Paperless-ng appliquera toutes les autres règles de rapprochement que vous avez définies." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless-ng pour le courriel" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "compte de messagerie" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "comptes de messagerie" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Pas de chiffrement" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Utiliser SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Utiliser STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Serveur IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Port IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Généralement 143 pour les connexions non chiffrées et STARTTLS, et 993 pour les connexions SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Sécurité IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "nom d'utilisateur" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "mot de passe" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "jeu de caractères" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Le jeu de caractères à utiliser lors de la communication avec le serveur de messagerie, par exemple 'UTF-8' ou 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "règle de courriel" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "règles de courriel" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Ne traiter que les pièces jointes." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Traiter tous les fichiers, y compris les pièces jointes \"en ligne\"." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Marquer comme lu, ne pas traiter les courriels lus" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Marquer le courriel, ne pas traiter les courriels marqués" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Déplacer vers le dossier spécifié" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Supprimer" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Utiliser le sujet en tant que titre" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Utiliser le nom de la pièce jointe en tant que titre" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Ne pas affecter de correspondant" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Utiliser l'adresse électronique" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Utiliser le nom (ou l'adresse électronique s'il n'est pas disponible)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Utiliser le correspondant sélectionné ci-dessous" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "ordre" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "compte" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "répertoire" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Les sous-dossiers doivent être séparés par des points." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrer l'expéditeur" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrer le sujet" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrer le corps du message" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrer le nom de fichier de la pièce jointe" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Ne traiter que les documents correspondant intégralement à ce nom de fichier s'il est spécifié. Les jokers tels que *.pdf ou *facture* sont autorisés. La casse n'est pas prise en compte." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "âge maximum" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "En jours." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "type de pièce jointe" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Les pièces jointes en ligne comprennent les images intégrées, il est donc préférable de combiner cette option avec un filtre de nom de fichier." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "action" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "paramètre d'action" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Paramètre supplémentaire pour l'action sélectionnée ci-dessus, par exemple le dossier cible de l'action de déplacement vers un dossier. Les sous-dossiers doivent être séparés par des points." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "affecter le titre depuis" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "affecter cette étiquette" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "affecter ce type de document" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "affecter le correspondant depuis" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "affecter ce correspondant" + diff --git a/src/locale/hu_HU/LC_MESSAGES/django.po b/src/locale/hu_HU/LC_MESSAGES/django.po new file mode 100644 index 000000000..870a35195 --- /dev/null +++ b/src/locale/hu_HU/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Hungarian\n" +"Language: hu_HU\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: hu\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokumentumok" + +#: documents/models.py:32 +msgid "Any word" +msgstr "" + +#: documents/models.py:33 +msgid "All words" +msgstr "" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Regexp" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatikus" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "" + +#: documents/models.py:45 +msgid "match" +msgstr "" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "" + +#: documents/models.py:81 +msgid "color" +msgstr "" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "" + +#: documents/models.py:94 +msgid "tag" +msgstr "" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "" + +#: documents/models.py:102 +msgid "document types" +msgstr "" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "" + +#: documents/models.py:124 +msgid "title" +msgstr "" + +#: documents/models.py:137 +msgid "content" +msgstr "" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "" + +#: documents/models.py:144 +msgid "mime type" +msgstr "" + +#: documents/models.py:155 +msgid "checksum" +msgstr "" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "" + +#: documents/models.py:176 +msgid "modified" +msgstr "" + +#: documents/models.py:180 +msgid "storage type" +msgstr "" + +#: documents/models.py:188 +msgid "added" +msgstr "" + +#: documents/models.py:192 +msgid "filename" +msgstr "" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "" + +#: documents/models.py:223 +msgid "document" +msgstr "" + +#: documents/models.py:224 +msgid "documents" +msgstr "" + +#: documents/models.py:311 +msgid "debug" +msgstr "" + +#: documents/models.py:312 +msgid "information" +msgstr "" + +#: documents/models.py:313 +msgid "warning" +msgstr "" + +#: documents/models.py:314 +msgid "error" +msgstr "" + +#: documents/models.py:315 +msgid "critical" +msgstr "" + +#: documents/models.py:319 +msgid "group" +msgstr "" + +#: documents/models.py:322 +msgid "message" +msgstr "" + +#: documents/models.py:325 +msgid "level" +msgstr "" + +#: documents/models.py:332 +msgid "log" +msgstr "" + +#: documents/models.py:333 +msgid "logs" +msgstr "" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "" + +#: documents/models.py:345 +msgid "saved views" +msgstr "" + +#: documents/models.py:348 +msgid "user" +msgstr "" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "" + +#: documents/models.py:361 +msgid "sort field" +msgstr "" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "" + +#: documents/models.py:373 +msgid "title contains" +msgstr "" + +#: documents/models.py:374 +msgid "content contains" +msgstr "" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "" + +#: documents/models.py:377 +msgid "document type is" +msgstr "" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "" + +#: documents/models.py:379 +msgid "has tag" +msgstr "" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "" + +#: documents/models.py:381 +msgid "created before" +msgstr "" + +#: documents/models.py:382 +msgid "created after" +msgstr "" + +#: documents/models.py:383 +msgid "created year is" +msgstr "" + +#: documents/models.py:384 +msgid "created month is" +msgstr "" + +#: documents/models.py:385 +msgid "created day is" +msgstr "" + +#: documents/models.py:386 +msgid "added before" +msgstr "" + +#: documents/models.py:387 +msgid "added after" +msgstr "" + +#: documents/models.py:388 +msgid "modified before" +msgstr "" + +#: documents/models.py:389 +msgid "modified after" +msgstr "" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "" + +#: documents/models.py:394 +msgid "more like this" +msgstr "" + +#: documents/models.py:405 +msgid "rule type" +msgstr "" + +#: documents/models.py:409 +msgid "value" +msgstr "" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "" + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "" + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Angol (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Német" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "" + +#: paperless/settings.py:307 +msgid "French" +msgstr "" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Szűrő" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "" + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Műveletek" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "" + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metaadat" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "" + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Törlés" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "" + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "" + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "" + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "" + diff --git a/src/locale/it_IT/LC_MESSAGES/django.po b/src/locale/it_IT/LC_MESSAGES/django.po new file mode 100644 index 000000000..3d89a62b0 --- /dev/null +++ b/src/locale/it_IT/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-17 11:06\n" +"Last-Translator: \n" +"Language-Team: Italian\n" +"Language: it_IT\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: it\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documenti" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Qualsiasi parola" + +#: documents/models.py:33 +msgid "All words" +msgstr "Tutte le parole" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Corrispondenza esatta" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Espressione regolare" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Parole fuzzy" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatico" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nome" + +#: documents/models.py:45 +msgid "match" +msgstr "corrispondenza" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algoritmo di corrispondenza" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "non distingue maiuscole e minuscole" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "corrispondente" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "corrispondenti" + +#: documents/models.py:81 +msgid "color" +msgstr "colore" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "è tag di arrivo" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Contrassegna questo tag come tag in arrivo: tutti i documenti elaborati verranno taggati con questo tag." + +#: documents/models.py:94 +msgid "tag" +msgstr "tag" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "tag" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "tipo di documento" + +#: documents/models.py:102 +msgid "document types" +msgstr "tipi di documento" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Non criptato" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Criptato con GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titolo" + +#: documents/models.py:137 +msgid "content" +msgstr "contenuto" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "I dati grezzi o solo testo del documento. Questo campo è usato principalmente per la ricerca." + +#: documents/models.py:144 +msgid "mime type" +msgstr "tipo mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "checksum" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Il checksum del documento originale." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "checksum dell'archivio" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Il checksum del documento archiviato." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "creato il" + +#: documents/models.py:176 +msgid "modified" +msgstr "modificato il" + +#: documents/models.py:180 +msgid "storage type" +msgstr "tipo di storage" + +#: documents/models.py:188 +msgid "added" +msgstr "aggiunto il" + +#: documents/models.py:192 +msgid "filename" +msgstr "nome del file" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nome del file corrente nello storage" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "Nome file in archivio" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Il nome del file nell'archiviazione" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "numero seriale dell'archivio" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Posizione di questo documento all'interno dell'archivio fisico." + +#: documents/models.py:223 +msgid "document" +msgstr "documento" + +#: documents/models.py:224 +msgid "documents" +msgstr "documenti" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "informazione" + +#: documents/models.py:313 +msgid "warning" +msgstr "avvertimento" + +#: documents/models.py:314 +msgid "error" +msgstr "errore" + +#: documents/models.py:315 +msgid "critical" +msgstr "critico" + +#: documents/models.py:319 +msgid "group" +msgstr "gruppo" + +#: documents/models.py:322 +msgid "message" +msgstr "messaggio" + +#: documents/models.py:325 +msgid "level" +msgstr "livello" + +#: documents/models.py:332 +msgid "log" +msgstr "log" + +#: documents/models.py:333 +msgid "logs" +msgstr "log" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "vista salvata" + +#: documents/models.py:345 +msgid "saved views" +msgstr "viste salvate" + +#: documents/models.py:348 +msgid "user" +msgstr "utente" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "mostra sul cruscotto" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "mostra nella barra laterale" + +#: documents/models.py:361 +msgid "sort field" +msgstr "campo di ordinamento" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "ordine invertito" + +#: documents/models.py:373 +msgid "title contains" +msgstr "il titolo contiene" + +#: documents/models.py:374 +msgid "content contains" +msgstr "il contenuto contiene" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN è" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "la corrispondenza è" + +#: documents/models.py:377 +msgid "document type is" +msgstr "il tipo di documento è" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "è in arrivo" + +#: documents/models.py:379 +msgid "has tag" +msgstr "ha etichetta" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "ha qualsiasi etichetta" + +#: documents/models.py:381 +msgid "created before" +msgstr "creato prima del" + +#: documents/models.py:382 +msgid "created after" +msgstr "creato dopo il" + +#: documents/models.py:383 +msgid "created year is" +msgstr "l'anno di creazione è" + +#: documents/models.py:384 +msgid "created month is" +msgstr "il mese di creazione è" + +#: documents/models.py:385 +msgid "created day is" +msgstr "il giorno di creazione è" + +#: documents/models.py:386 +msgid "added before" +msgstr "aggiunto prima del" + +#: documents/models.py:387 +msgid "added after" +msgstr "aggiunto dopo il" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modificato prima del" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modificato dopo" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "non ha tag" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "non ha ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "il titolo o il contenuto contiene" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "query fulltext" + +#: documents/models.py:394 +msgid "more like this" +msgstr "altro come questo" + +#: documents/models.py:405 +msgid "rule type" +msgstr "tipo di regola" + +#: documents/models.py:409 +msgid "value" +msgstr "valore" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "regola filtro" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "regole filtro" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Espressione regolare non valida: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Colore non valido." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Il tipo di file %(type)s non è supportato" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng è in caricamento..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng è uscito" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Ti sei disconnesso. A presto!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Accedi nuovamente" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Accedi a Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Accedi" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Il nome utente e la password non sono corretti. Riprovare." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Nome utente" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Password" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Accedi" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Inglese (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Inglese (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Tedesco" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Olandese" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Francese" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portoghese (Brasile)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portoghese" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiano" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumeno" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russo" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spagnolo" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polacco" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Svedese" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Amministrazione di Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Autenticazione" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Impostazioni avanzate" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtro" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless-ng processerà solo la posta che rientra in TUTTI i filtri impostati." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Azioni" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "L'azione che viene applicata alla email. Questa azione viene eseguita solo quando dei documenti vengono elaborati dalla email. Le email senza allegati vengono ignorate." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadati" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Assegna automaticamente i metadati ai documenti elaborati da questa regola. Se non assegni qui dei tag, tipi di documenti o corrispondenti, Paperless userà comunque le regole corrispondenti che hai configurato." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Email Paperless" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "account email" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "account email" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Nessuna crittografia" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Usa SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Usa STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Server IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Porta IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Di solito si usa 143 per STARTTLS o nessuna crittografia e 993 per SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Sicurezza IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "nome utente" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "password" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "set di caratteri" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Il set di caratteri da usare quando si comunica con il server di posta, come 'UTF-8' o 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "regola email" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "regole email" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Elabora solo gli allegati." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Elabora tutti i file, inclusi gli allegati nel corpo." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Segna come letto, non elaborare le email lette" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Contrassegna la email, non elaborare le email elaborate." + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Sposta in una cartella" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Elimina" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Usa oggetto come titolo" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Usa il nome dell'allegato come titolo" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Non assegnare un corrispondente" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Usa indirizzo email" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Usa nome (o indirizzo email se non disponibile)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Usa il corrispondente selezionato qui sotto" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "priorità" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "account" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "cartella" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Le sottocartelle devono essere separate da punti." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtra da" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtra oggetto" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtra corpo" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtra nome allegato" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Elabora i documenti che corrispondono a questo nome. Puoi usare wildcard come *.pdf o *fattura*. Non fa differenza fra maiuscole e minuscole." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "età massima" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Definito in giorni." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "tipo di allegato" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Gli allegati in linea includono le immagini nel corpo, quindi è meglio combinare questa opzione con il filtro nome." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "azione" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parametro azione" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Parametro aggiuntivo per l'azione selezionata, ad esempio la cartella di destinazione per l'azione che sposta una cartella. Le sottocartelle devono essere separate da punti." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "assegna tittolo da" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "assegna questo tag" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "assegna questo tipo di documento" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "assegna corrispondente da" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "assegna questo corrispondente" + diff --git a/src/locale/la_LA/LC_MESSAGES/django.po b/src/locale/la_LA/LC_MESSAGES/django.po new file mode 100644 index 000000000..d631e4911 --- /dev/null +++ b/src/locale/la_LA/LC_MESSAGES/django.po @@ -0,0 +1,642 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-02-28 12:40+0100\n" +"PO-Revision-Date: 2021-03-06 21:39\n" +"Last-Translator: \n" +"Language-Team: Latin\n" +"Language: la_LA\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: la-LA\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "" + +#: documents/models.py:32 +msgid "Any word" +msgstr "" + +#: documents/models.py:33 +msgid "All words" +msgstr "" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:109 +msgid "name" +msgstr "" + +#: documents/models.py:45 +msgid "match" +msgstr "" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "" + +#: documents/models.py:81 +msgid "color" +msgstr "" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "" + +#: documents/models.py:94 +msgid "tag" +msgstr "" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "" + +#: documents/models.py:102 +msgid "document types" +msgstr "" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "" + +#: documents/models.py:124 +msgid "title" +msgstr "" + +#: documents/models.py:137 +msgid "content" +msgstr "" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "" + +#: documents/models.py:144 +msgid "mime type" +msgstr "" + +#: documents/models.py:155 +msgid "checksum" +msgstr "" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "" + +#: documents/models.py:176 +msgid "modified" +msgstr "" + +#: documents/models.py:180 +msgid "storage type" +msgstr "" + +#: documents/models.py:188 +msgid "added" +msgstr "" + +#: documents/models.py:192 +msgid "filename" +msgstr "" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "" + +#: documents/models.py:223 +msgid "document" +msgstr "" + +#: documents/models.py:224 +msgid "documents" +msgstr "" + +#: documents/models.py:311 +msgid "debug" +msgstr "" + +#: documents/models.py:312 +msgid "information" +msgstr "" + +#: documents/models.py:313 +msgid "warning" +msgstr "" + +#: documents/models.py:314 +msgid "error" +msgstr "" + +#: documents/models.py:315 +msgid "critical" +msgstr "" + +#: documents/models.py:319 +msgid "group" +msgstr "" + +#: documents/models.py:322 +msgid "message" +msgstr "" + +#: documents/models.py:325 +msgid "level" +msgstr "" + +#: documents/models.py:332 +msgid "log" +msgstr "" + +#: documents/models.py:333 +msgid "logs" +msgstr "" + +#: documents/models.py:344 documents/models.py:394 +msgid "saved view" +msgstr "" + +#: documents/models.py:345 +msgid "saved views" +msgstr "" + +#: documents/models.py:348 +msgid "user" +msgstr "" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "" + +#: documents/models.py:361 +msgid "sort field" +msgstr "" + +#: documents/models.py:364 +msgid "sort reverse" +msgstr "" + +#: documents/models.py:370 +msgid "title contains" +msgstr "" + +#: documents/models.py:371 +msgid "content contains" +msgstr "" + +#: documents/models.py:372 +msgid "ASN is" +msgstr "" + +#: documents/models.py:373 +msgid "correspondent is" +msgstr "" + +#: documents/models.py:374 +msgid "document type is" +msgstr "" + +#: documents/models.py:375 +msgid "is in inbox" +msgstr "" + +#: documents/models.py:376 +msgid "has tag" +msgstr "" + +#: documents/models.py:377 +msgid "has any tag" +msgstr "" + +#: documents/models.py:378 +msgid "created before" +msgstr "" + +#: documents/models.py:379 +msgid "created after" +msgstr "" + +#: documents/models.py:380 +msgid "created year is" +msgstr "" + +#: documents/models.py:381 +msgid "created month is" +msgstr "" + +#: documents/models.py:382 +msgid "created day is" +msgstr "" + +#: documents/models.py:383 +msgid "added before" +msgstr "" + +#: documents/models.py:384 +msgid "added after" +msgstr "" + +#: documents/models.py:385 +msgid "modified before" +msgstr "" + +#: documents/models.py:386 +msgid "modified after" +msgstr "" + +#: documents/models.py:387 +msgid "does not have tag" +msgstr "" + +#: documents/models.py:398 +msgid "rule type" +msgstr "" + +#: documents/models.py:402 +msgid "value" +msgstr "" + +#: documents/models.py:408 +msgid "filter rule" +msgstr "" + +#: documents/models.py:409 +msgid "filter rules" +msgstr "" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expresssion: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "" + +#: documents/templates/index.html:21 +msgid "Paperless-ng is loading..." +msgstr "" + +#: documents/templates/registration/logged_out.html:13 +msgid "Paperless-ng signed out" +msgstr "" + +#: documents/templates/registration/logged_out.html:41 +msgid "You have been successfully logged out. Bye!" +msgstr "" + +#: documents/templates/registration/logged_out.html:42 +msgid "Sign in again" +msgstr "" + +#: documents/templates/registration/login.html:13 +msgid "Paperless-ng sign in" +msgstr "" + +#: documents/templates/registration/login.html:42 +msgid "Please sign in." +msgstr "" + +#: documents/templates/registration/login.html:45 +msgid "Your username and password didn't match. Please try again." +msgstr "" + +#: documents/templates/registration/login.html:48 +msgid "Username" +msgstr "" + +#: documents/templates/registration/login.html:49 +msgid "Password" +msgstr "" + +#: documents/templates/registration/login.html:54 +msgid "Sign in" +msgstr "" + +#: paperless/settings.py:297 +msgid "English (US)" +msgstr "" + +#: paperless/settings.py:298 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:299 +msgid "German" +msgstr "" + +#: paperless/settings.py:300 +msgid "Dutch" +msgstr "" + +#: paperless/settings.py:301 +msgid "French" +msgstr "" + +#: paperless/settings.py:302 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:303 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:304 +msgid "Romanian" +msgstr "" + +#: paperless/urls.py:118 +msgid "Paperless-ng administration" +msgstr "" + +#: paperless_mail/admin.py:25 +msgid "Filter" +msgstr "" + +#: paperless_mail/admin.py:27 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Actions" +msgstr "" + +#: paperless_mail/admin.py:39 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "" + +#: paperless_mail/admin.py:46 +msgid "Metadata" +msgstr "" + +#: paperless_mail/admin.py:48 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "" + +#: paperless_mail/models.py:60 +msgid "mail rule" +msgstr "" + +#: paperless_mail/models.py:61 +msgid "mail rules" +msgstr "" + +#: paperless_mail/models.py:67 +msgid "Only process attachments." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "Process all files, including 'inline' attachments." +msgstr "" + +#: paperless_mail/models.py:78 +msgid "Mark as read, don't process read mails" +msgstr "" + +#: paperless_mail/models.py:79 +msgid "Flag the mail, don't process flagged mails" +msgstr "" + +#: paperless_mail/models.py:80 +msgid "Move to specified folder" +msgstr "" + +#: paperless_mail/models.py:81 +msgid "Delete" +msgstr "" + +#: paperless_mail/models.py:88 +msgid "Use subject as title" +msgstr "" + +#: paperless_mail/models.py:89 +msgid "Use attachment filename as title" +msgstr "" + +#: paperless_mail/models.py:99 +msgid "Do not assign a correspondent" +msgstr "" + +#: paperless_mail/models.py:101 +msgid "Use mail address" +msgstr "" + +#: paperless_mail/models.py:103 +msgid "Use name (or mail address if not available)" +msgstr "" + +#: paperless_mail/models.py:105 +msgid "Use correspondent selected below" +msgstr "" + +#: paperless_mail/models.py:113 +msgid "order" +msgstr "" + +#: paperless_mail/models.py:120 +msgid "account" +msgstr "" + +#: paperless_mail/models.py:124 +msgid "folder" +msgstr "" + +#: paperless_mail/models.py:128 +msgid "filter from" +msgstr "" + +#: paperless_mail/models.py:131 +msgid "filter subject" +msgstr "" + +#: paperless_mail/models.py:134 +msgid "filter body" +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter attachment filename" +msgstr "" + +#: paperless_mail/models.py:140 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "" + +#: paperless_mail/models.py:146 +msgid "maximum age" +msgstr "" + +#: paperless_mail/models.py:148 +msgid "Specified in days." +msgstr "" + +#: paperless_mail/models.py:151 +msgid "attachment type" +msgstr "" + +#: paperless_mail/models.py:154 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "" + +#: paperless_mail/models.py:159 +msgid "action" +msgstr "" + +#: paperless_mail/models.py:165 +msgid "action parameter" +msgstr "" + +#: paperless_mail/models.py:167 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action." +msgstr "" + +#: paperless_mail/models.py:173 +msgid "assign title from" +msgstr "" + +#: paperless_mail/models.py:183 +msgid "assign this tag" +msgstr "" + +#: paperless_mail/models.py:191 +msgid "assign this document type" +msgstr "" + +#: paperless_mail/models.py:195 +msgid "assign correspondent from" +msgstr "" + +#: paperless_mail/models.py:205 +msgid "assign this correspondent" +msgstr "" + diff --git a/src/locale/lb_LU/LC_MESSAGES/django.po b/src/locale/lb_LU/LC_MESSAGES/django.po new file mode 100644 index 000000000..bda9d5ebc --- /dev/null +++ b/src/locale/lb_LU/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-07-16 14:22\n" +"Last-Translator: \n" +"Language-Team: Luxembourgish\n" +"Language: lb_LU\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: lb\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokumenter" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Iergendee Wuert" + +#: documents/models.py:33 +msgid "All words" +msgstr "All d'Wierder" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Exakten Treffer" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Regulären Ausdrock" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Ongenaut Wuert" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatesch" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "Numm" + +#: documents/models.py:45 +msgid "match" +msgstr "Zouweisungsmuster" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "Zouweisungsalgorithmus" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "Grouss-/Klengschreiwung ignoréieren" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "Korrespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "Korrespondenten" + +#: documents/models.py:81 +msgid "color" +msgstr "Faarf" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "Postaganks-Etikett" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Dës Etikett als Postaganks-Etikett markéieren: All nei importéiert Dokumenter kréien ëmmer dës Etikett zougewisen." + +#: documents/models.py:94 +msgid "tag" +msgstr "Etikett" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "Etiketten" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "Dokumententyp" + +#: documents/models.py:102 +msgid "document types" +msgstr "Dokumententypen" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Onverschlësselt" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Verschlësselt mat GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "Titel" + +#: documents/models.py:137 +msgid "content" +msgstr "Inhalt" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "De réien Textinhalt vum Dokument. Dëst Feld gëtt primär fir d'Sich benotzt." + +#: documents/models.py:144 +msgid "mime type" +msgstr "MIME-Typ" + +#: documents/models.py:155 +msgid "checksum" +msgstr "Préifzomm" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "D'Préifzomm vum Original-Dokument." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "Archiv-Préifzomm" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "D'Préifzomm vum archivéierten Dokument." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "erstallt" + +#: documents/models.py:176 +msgid "modified" +msgstr "verännert" + +#: documents/models.py:180 +msgid "storage type" +msgstr "Späichertyp" + +#: documents/models.py:188 +msgid "added" +msgstr "derbäigesat" + +#: documents/models.py:192 +msgid "filename" +msgstr "Fichiersnumm" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Aktuellen Dateinumm am Späicher" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "Archiv-Dateinumm" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Aktuellen Dateinumm am Archiv" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "Archiv-Seriennummer" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "D'Positioun vun dësem Dokument am physeschen Dokumentenarchiv." + +#: documents/models.py:223 +msgid "document" +msgstr "Dokument" + +#: documents/models.py:224 +msgid "documents" +msgstr "Dokumenter" + +#: documents/models.py:311 +msgid "debug" +msgstr "Fehlersiich" + +#: documents/models.py:312 +msgid "information" +msgstr "Informatioun" + +#: documents/models.py:313 +msgid "warning" +msgstr "Warnung" + +#: documents/models.py:314 +msgid "error" +msgstr "Feeler" + +#: documents/models.py:315 +msgid "critical" +msgstr "kritesch" + +#: documents/models.py:319 +msgid "group" +msgstr "Grupp" + +#: documents/models.py:322 +msgid "message" +msgstr "Message" + +#: documents/models.py:325 +msgid "level" +msgstr "Niveau" + +#: documents/models.py:332 +msgid "log" +msgstr "Protokoll" + +#: documents/models.py:333 +msgid "logs" +msgstr "Protokoller" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "Gespäichert Usiicht" + +#: documents/models.py:345 +msgid "saved views" +msgstr "Gespäichert Usiichten" + +#: documents/models.py:348 +msgid "user" +msgstr "Benotzer" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "Op der Startsäit uweisen" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "An der Säiteleescht uweisen" + +#: documents/models.py:361 +msgid "sort field" +msgstr "Zortéierfeld" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "ëmgedréint zortéieren" + +#: documents/models.py:373 +msgid "title contains" +msgstr "Titel enthält" + +#: documents/models.py:374 +msgid "content contains" +msgstr "Inhalt enthält" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN ass" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "Korrespondent ass" + +#: documents/models.py:377 +msgid "document type is" +msgstr "Dokumententyp ass" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "ass am Postagank" + +#: documents/models.py:379 +msgid "has tag" +msgstr "huet Etikett" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "huet iergendeng Etikett" + +#: documents/models.py:381 +msgid "created before" +msgstr "erstallt virun" + +#: documents/models.py:382 +msgid "created after" +msgstr "erstallt no" + +#: documents/models.py:383 +msgid "created year is" +msgstr "Erstellungsjoer ass" + +#: documents/models.py:384 +msgid "created month is" +msgstr "Erstellungsmount ass" + +#: documents/models.py:385 +msgid "created day is" +msgstr "Erstellungsdag ass" + +#: documents/models.py:386 +msgid "added before" +msgstr "dobäigesat virun" + +#: documents/models.py:387 +msgid "added after" +msgstr "dobäigesat no" + +#: documents/models.py:388 +msgid "modified before" +msgstr "verännert virun" + +#: documents/models.py:389 +msgid "modified after" +msgstr "verännert no" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "huet dës Etikett net" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "huet keng ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "Titel oder Inhalt enthalen" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "Volltextsich" + +#: documents/models.py:394 +msgid "more like this" +msgstr "ähnlech Dokumenter" + +#: documents/models.py:405 +msgid "rule type" +msgstr "Reegeltyp" + +#: documents/models.py:409 +msgid "value" +msgstr "Wäert" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "Filterreegel" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "Filterreegelen" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Ongëltege regulären Ausdrock: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Ongëlteg Faarf." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Fichierstyp %(type)s net ënnerstëtzt" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng gëtt gelueden..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng ofgemellt" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Dir hutt Iech erfollegräich ofgemellt. Bis geschwënn!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Nees umellen" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Umeldung bei Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "W. e. g. umellen." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Äre Benotzernumm a Passwuert stëmmen net iwwereneen. Probéiert w. e. g. nach emol." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Benotzernumm" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Passwuert" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Umellen" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Englesch (USA)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Englesch (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Däitsch" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Hollännesch" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Franséisch" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugisesch (Brasilien)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugisesch" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italienesch" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumänesch" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russesch" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spuenesch" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polnesch" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Schwedesch" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng-Administratioun" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Authentifizéierung" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Erweidert Astellungen" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filter" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless wäert nëmmen E-Maile veraarbechten, fir déi all déi hei definéiert Filteren zoutreffen." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Aktiounen" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "D'Aktioun, déi op E-Mailen applizéiert sill ginn. Dës Aktioun gëtt just ausgefouert, wann Dokumenter aus der E-Mail veraarbecht goufen. E-Mailen ouni Unhank bleiwe komplett onberéiert." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadaten" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Den Dokumenter, déi iwwer dës Reegel veraarbecht ginn, automatesch Metadaten zouweisen. Wann hei keng Etiketten, Typen oder Korrespondenten zougewise ginn, veraarbecht Paperless-ng trotzdeem all zoutreffend Reegelen déi definéiert sinn." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless E-Mail" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "Mailkont" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "Mailkonten" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Keng Verschlësselung" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "SSL benotzen" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "STARTTLS benotzen" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP-Server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP-Port" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Dëst ass normalerweis 143 fir onverschësselt oder STARTTLS-Connectiounen an 993 fir SSL-Connectiounen." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP-Sécherheet" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "Benotzernumm" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "Passwuert" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Zeechesaz" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Den Zeechesaz dee benotzt gëtt wa mam Mailserver kommunizéiert gëtt, wéi beispillsweis 'UTF-8' oder 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "E-Mail-Reegel" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "E-Mail-Reegelen" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Just Unhäng veraarbechten." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "All d'Fichiere veraarbechten, inklusiv \"inline\"-Unhäng." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Als gelies markéieren, gelies Mailen net traitéieren" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Als wichteg markéieren, markéiert E-Mailen net veraarbechten" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "An e virdefinéierten Dossier réckelen" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Läschen" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Sujet als TItel notzen" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Numm vum Dateiunhank als Titel benotzen" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Kee Korrespondent zouweisen" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "E-Mail-Adress benotzen" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Numm benotzen (oder E-Mail-Adress falls den Numm net disponibel ass)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Korrespondent benotzen deen hei drënner ausgewielt ass" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "Reiefolleg" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "Kont" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "Dossier" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Ënnerdossiere mussen duerch Punkte getrennt ginn." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "Ofsenderfilter" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "Sujets-Filter" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "Contenu-Filter" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "Filter fir den Numm vum Unhank" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Just Dokumenter traitéieren, déi exakt dësen Dateinumm hunn (falls definéiert). Platzhalter wéi *.pdf oder *invoice* sinn erlaabt. D'Grouss-/Klengschreiwung gëtt ignoréiert." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "Maximalen Alter" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "An Deeg." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "Fichierstyp vum Unhank" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "\"Inline\"-Unhänk schléissen och agebonne Biller mat an, dofir sollt dës Astellung mat engem Filter fir den Numm vum Unhank kombinéiert ginn." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "Aktioun" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "Parameter fir Aktioun" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Zousätzleche Parameter fir d'Aktioun déi hei driwwer ausgewielt ass, zum Beispill den Numm vum Zildossier fir d'Aktioun \"An e virdefinéierten Dossier réckelen\". Ënnerdossiere musse mat Punkte getrennt ginn." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "Titel zouweisen aus" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "dës Etikett zouweisen" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "Dësen Dokumententyp zouweisen" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "Korrespondent zouweisen aus" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "Dëse Korrespondent zouweisen" + diff --git a/src/locale/nl_NL/LC_MESSAGES/django.po b/src/locale/nl_NL/LC_MESSAGES/django.po new file mode 100644 index 000000000..1272add6f --- /dev/null +++ b/src/locale/nl_NL/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-22 10:12\n" +"Last-Translator: \n" +"Language-Team: Dutch\n" +"Language: nl_NL\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: nl\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documenten" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Elk woord" + +#: documents/models.py:33 +msgid "All words" +msgstr "Alle woorden" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Exacte overeenkomst" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Reguliere expressie" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Gelijkaardig woord" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatisch" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "naam" + +#: documents/models.py:45 +msgid "match" +msgstr "Overeenkomst" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "Algoritme voor het bepalen van de overeenkomst" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "is niet hoofdlettergevoelig" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "correspondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "correspondenten" + +#: documents/models.py:81 +msgid "color" +msgstr "Kleur" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "is \"Postvak in\"-label" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Markeert dit label als een \"Postvak in\"-label: alle nieuw verwerkte documenten krijgen de \"Postvak in\"-labels." + +#: documents/models.py:94 +msgid "tag" +msgstr "label" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "labels" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "documenttype" + +#: documents/models.py:102 +msgid "document types" +msgstr "documenttypen" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Niet versleuteld" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Versleuteld met GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titel" + +#: documents/models.py:137 +msgid "content" +msgstr "inhoud" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "De onbewerkte gegevens van het document. Dit veld wordt voornamelijk gebruikt om te zoeken." + +#: documents/models.py:144 +msgid "mime type" +msgstr "mimetype" + +#: documents/models.py:155 +msgid "checksum" +msgstr "checksum" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "De checksum van het oorspronkelijke document." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "archief checksum" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "De checksum van het gearchiveerde document." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "aangemaakt" + +#: documents/models.py:176 +msgid "modified" +msgstr "gewijzigd" + +#: documents/models.py:180 +msgid "storage type" +msgstr "type opslag" + +#: documents/models.py:188 +msgid "added" +msgstr "toegevoegd" + +#: documents/models.py:192 +msgid "filename" +msgstr "bestandsnaam" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Huidige bestandsnaam in opslag" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "Bestandsnaam in archief" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Huidige bestandsnaam in archief" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "serienummer in archief" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "De positie van dit document in je fysieke documentenarchief." + +#: documents/models.py:223 +msgid "document" +msgstr "document" + +#: documents/models.py:224 +msgid "documents" +msgstr "documenten" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "informatie" + +#: documents/models.py:313 +msgid "warning" +msgstr "waarschuwing" + +#: documents/models.py:314 +msgid "error" +msgstr "fout" + +#: documents/models.py:315 +msgid "critical" +msgstr "kritisch" + +#: documents/models.py:319 +msgid "group" +msgstr "groep" + +#: documents/models.py:322 +msgid "message" +msgstr "bericht" + +#: documents/models.py:325 +msgid "level" +msgstr "niveau" + +#: documents/models.py:332 +msgid "log" +msgstr "bericht" + +#: documents/models.py:333 +msgid "logs" +msgstr "berichten" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "opgeslagen view" + +#: documents/models.py:345 +msgid "saved views" +msgstr "opgeslagen views" + +#: documents/models.py:348 +msgid "user" +msgstr "gebruiker" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "weergeven op dashboard" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "weergeven in zijbalk" + +#: documents/models.py:361 +msgid "sort field" +msgstr "sorteerveld" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "omgekeerd sorteren" + +#: documents/models.py:373 +msgid "title contains" +msgstr "titel bevat" + +#: documents/models.py:374 +msgid "content contains" +msgstr "inhoud bevat" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN is" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "correspondent is" + +#: documents/models.py:377 +msgid "document type is" +msgstr "documenttype is" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "zit in \"Postvak in\"" + +#: documents/models.py:379 +msgid "has tag" +msgstr "heeft label" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "heeft één van de labels" + +#: documents/models.py:381 +msgid "created before" +msgstr "aangemaakt voor" + +#: documents/models.py:382 +msgid "created after" +msgstr "aangemaakt na" + +#: documents/models.py:383 +msgid "created year is" +msgstr "aangemaakt jaar is" + +#: documents/models.py:384 +msgid "created month is" +msgstr "aangemaakte maand is" + +#: documents/models.py:385 +msgid "created day is" +msgstr "aangemaakte dag is" + +#: documents/models.py:386 +msgid "added before" +msgstr "toegevoegd voor" + +#: documents/models.py:387 +msgid "added after" +msgstr "toegevoegd na" + +#: documents/models.py:388 +msgid "modified before" +msgstr "gewijzigd voor" + +#: documents/models.py:389 +msgid "modified after" +msgstr "gewijzigd na" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "heeft geen label" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "heeft geen ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "titel of inhoud bevat" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "inhoud doorzoeken" + +#: documents/models.py:394 +msgid "more like this" +msgstr "meer zoals dit" + +#: documents/models.py:405 +msgid "rule type" +msgstr "type regel" + +#: documents/models.py:409 +msgid "value" +msgstr "waarde" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "filterregel" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "filterregels" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Ongeldige reguliere expressie: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Ongeldig kleur." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Bestandstype %(type)s niet ondersteund" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng is aan het laden..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng - afmelden" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Je bent nu afgemeld. Tot later!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Meld je opnieuw aan" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng - aanmelden" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Gelieve aan te melden." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Je gebruikersnaam en wachtwoord komen niet overeen. Probeer opnieuw." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Gebruikersnaam" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Wachtwoord" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Aanmelden" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Engels (US)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Engels (Brits)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Duits" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Nederlands" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Frans" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugees (Brazilië)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugees" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiaans" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Roemeens" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russisch" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spaans" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Pools" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Zweeds" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng administratie" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Authenticatie" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Geavanceerde instellingen" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filter" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless verwerkt alleen e-mails die voldoen aan ALLE onderstaande filters." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Acties" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "De actie die wordt toegepast op de mail. Deze actie wordt alleen uitgevoerd wanneer documenten verwerkt werden uit de mail. Mails zonder bijlage blijven onaangeroerd." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadata" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Automatisch metadata toewijzen aan documenten vanuit deze regel. Indien je geen labels, documenttypes of correspondenten toewijst, zal Paperless nog steeds alle regels verwerken die je hebt gedefinieerd." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless email" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "email account" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "email accounts" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Geen versleuteling" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Gebruik SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Gebruik STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP-server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP-poort" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Dit is gewoonlijk 143 voor onversleutelde of STARTTLS verbindingen, en 993 voor SSL verbindingen." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP-beveiliging" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "gebruikersnaam" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "wachtwoord" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Tekenset" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Tekenset die gebruikt moet worden bij communicatie met de mailserver, zoals 'UTF-8' of 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "email-regel" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "email-regels" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Alleen bijlagen verwerken" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Verwerk alle bestanden, inclusief 'inline' bijlagen." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Markeer als gelezen, verwerk geen gelezen mails" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Markeer de mail, verwerk geen mails met markering" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Verplaats naar gegeven map" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Verwijder" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Gebruik onderwerp als titel" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Gebruik naam van bijlage als titel" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Wijs geen correspondent toe" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Gebruik het email-adres" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Gebruik de naam, en anders het email-adres" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Gebruik de hieronder aangeduide correspondent" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "volgorde" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "account" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "map" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Submappen moeten gescheiden worden door punten." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filter afzender" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filter onderwerp" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filter inhoud" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "Filter bestandsnaam van bijlage" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Alleen documenten verwerken die volledig overeenkomen, indien aangegeven. Je kunt jokertekens gebruiken, zoals *.pdf of *factuur*. Dit is niet hoofdlettergevoelig." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "Maximale leeftijd" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Aangegeven in dagen" + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "Type bijlage" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "\"Inline\" bijlagen bevatten vaak ook afbeeldingen. In dit geval valt het aan te raden om ook een filter voor de bestandsnaam op te geven." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "actie" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "actie parameters" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Extra parameter voor de hierboven geselecteerde actie, bijvoorbeeld: de doelmap voor de \"verplaats naar map\"-actie. Submappen moeten gescheiden worden door punten." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "wijs titel toe van" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "wijs dit etiket toe" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "wijs dit documenttype toe" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "wijs correspondent toe van" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "wijs deze correspondent toe" + diff --git a/src/locale/pl_PL/LC_MESSAGES/django.po b/src/locale/pl_PL/LC_MESSAGES/django.po new file mode 100644 index 000000000..350204e08 --- /dev/null +++ b/src/locale/pl_PL/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-08-07 13:02\n" +"Last-Translator: \n" +"Language-Team: Polish\n" +"Language: pl_PL\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=4; plural=(n==1 ? 0 : (n%10>=2 && n%10<=4) && (n%100<12 || n%100>14) ? 1 : n!=1 && (n%10>=0 && n%10<=1) || (n%10>=5 && n%10<=9) || (n%100>=12 && n%100<=14) ? 2 : 3);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: pl\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokumenty" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Dowolne słowo" + +#: documents/models.py:33 +msgid "All words" +msgstr "Wszystkie słowa" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Dokładne dopasowanie" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Wyrażenie regularne" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Dopasowanie rozmyte" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatyczny" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nazwa" + +#: documents/models.py:45 +msgid "match" +msgstr "dopasowanie" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algorytm dopasowania" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "bez rozróżniania wielkości liter" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "korespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "korespondenci" + +#: documents/models.py:81 +msgid "color" +msgstr "kolor" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "jest tagiem skrzynki odbiorczej" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Zaznacza ten tag jako tag skrzynki odbiorczej: Wszystkie nowo przetworzone dokumenty będą oznaczone tagami skrzynki odbiorczej." + +#: documents/models.py:94 +msgid "tag" +msgstr "tag" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "tagi" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "typ dokumentu" + +#: documents/models.py:102 +msgid "document types" +msgstr "typy dokumentów" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Niezaszyfrowane" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Zaszyfrowane przy użyciu GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "tytuł" + +#: documents/models.py:137 +msgid "content" +msgstr "zawartość" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Surowe, tekstowe dane dokumentu. To pole jest używane głównie do wyszukiwania." + +#: documents/models.py:144 +msgid "mime type" +msgstr "typ mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "suma kontrolna" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Suma kontrolna oryginalnego dokumentu." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "suma kontrolna archiwum" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Suma kontrolna zarchiwizowanego dokumentu." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "utworzono" + +#: documents/models.py:176 +msgid "modified" +msgstr "zmodyfikowano" + +#: documents/models.py:180 +msgid "storage type" +msgstr "typ przechowywania" + +#: documents/models.py:188 +msgid "added" +msgstr "dodano" + +#: documents/models.py:192 +msgid "filename" +msgstr "nazwa pliku" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Aktualna nazwa pliku w pamięci" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nazwa pliku archiwum" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Aktualna nazwa pliku archiwum w pamięci" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "numer seryjny archiwum" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Pozycja tego dokumentu w archiwum dokumentów fizycznych." + +#: documents/models.py:223 +msgid "document" +msgstr "dokument" + +#: documents/models.py:224 +msgid "documents" +msgstr "dokumenty" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "informacja" + +#: documents/models.py:313 +msgid "warning" +msgstr "ostrzeżenie" + +#: documents/models.py:314 +msgid "error" +msgstr "błąd" + +#: documents/models.py:315 +msgid "critical" +msgstr "krytyczne" + +#: documents/models.py:319 +msgid "group" +msgstr "grupa" + +#: documents/models.py:322 +msgid "message" +msgstr "wiadomość" + +#: documents/models.py:325 +msgid "level" +msgstr "poziom" + +#: documents/models.py:332 +msgid "log" +msgstr "log" + +#: documents/models.py:333 +msgid "logs" +msgstr "logi" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "zapisany widok" + +#: documents/models.py:345 +msgid "saved views" +msgstr "zapisane widoki" + +#: documents/models.py:348 +msgid "user" +msgstr "użytkownik" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "pokaż na pulpicie" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "pokaż na pasku bocznym" + +#: documents/models.py:361 +msgid "sort field" +msgstr "pole sortowania" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "sortuj malejąco" + +#: documents/models.py:373 +msgid "title contains" +msgstr "tytuł zawiera" + +#: documents/models.py:374 +msgid "content contains" +msgstr "zawartość zawiera" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "numer archiwum jest" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "korespondentem jest" + +#: documents/models.py:377 +msgid "document type is" +msgstr "typ dokumentu jest" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "jest w skrzynce odbiorczej" + +#: documents/models.py:379 +msgid "has tag" +msgstr "ma tag" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "ma dowolny tag" + +#: documents/models.py:381 +msgid "created before" +msgstr "utworzony przed" + +#: documents/models.py:382 +msgid "created after" +msgstr "utworzony po" + +#: documents/models.py:383 +msgid "created year is" +msgstr "rok utworzenia to" + +#: documents/models.py:384 +msgid "created month is" +msgstr "miesiąc utworzenia to" + +#: documents/models.py:385 +msgid "created day is" +msgstr "dzień utworzenia to" + +#: documents/models.py:386 +msgid "added before" +msgstr "dodany przed" + +#: documents/models.py:387 +msgid "added after" +msgstr "dodany po" + +#: documents/models.py:388 +msgid "modified before" +msgstr "zmodyfikowany przed" + +#: documents/models.py:389 +msgid "modified after" +msgstr "zmodyfikowany po" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "nie ma tagu" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "nie ma numeru archiwum" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "tytuł lub zawartość zawiera" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "zapytanie pełnotekstowe" + +#: documents/models.py:394 +msgid "more like this" +msgstr "podobne dokumenty" + +#: documents/models.py:405 +msgid "rule type" +msgstr "typ reguły" + +#: documents/models.py:409 +msgid "value" +msgstr "wartość" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "reguła filtrowania" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "reguły filtrowania" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Nieprawidłowe wyrażenie regularne: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Nieprawidłowy kolor." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Typ pliku %(type)s nie jest obsługiwany" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Ładowanie Paperless-ng..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Wylogowano z Paperless-ng" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Poprawnie wylogowano. Do zobaczenia!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Zaloguj się ponownie" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Logowanie do Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Proszę się zalogować." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Twoja nazwa użytkownika i hasło nie są zgodne. Spróbuj ponownie." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Użytkownik" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Hasło" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Zaloguj się" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Angielski (USA)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Angielski (Wielka Brytania)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Niemiecki" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Holenderski" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Francuski" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugalski (Brazylia)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugalski" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Włoski" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumuński" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Rosyjski" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Hiszpański" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polski" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Szwedzki" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Administracja Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Uwierzytelnianie" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Ustawienia zaawansowane" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtry" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless przetworzy tylko wiadomości pasujące do WSZYSTKICH filtrów podanych poniżej." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Akcje" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Akcja zastosowana do wiadomości. Ta akcja jest wykonywana tylko wtedy, gdy dokumenty zostały przetworzone z wiadomości. Poczta bez załączników pozostanie całkowicie niezmieniona." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadane" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Przypisz metadane do dokumentów zużywanych z tej reguły automatycznie. Jeśli nie przypisujesz tutaj tagów, typów lub korespondentów, Paperless będzie nadal przetwarzał wszystkie zdefiniowane przez Ciebie reguły." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Poczta Paperless" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "konto pocztowe" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "konta pocztowe" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Brak szyfrowania" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Użyj SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Użyj STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Serwer IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Port IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Zwykle jest to 143 dla połączeń niezaszyfrowanych i STARTTLS oraz 993 dla połączeń SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Zabezpieczenia IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "nazwa użytkownika" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "hasło" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Kodowanie" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Zestaw znaków używany podczas komunikowania się z serwerem poczty, np. \"UTF-8\" lub \"US-ASCII\"." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "reguła wiadomości" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "reguły wiadomości" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Przetwarzaj tylko załączniki." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Przetwarzaj wszystkie pliki, łącznie z załącznikami „inline”." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Oznacz jako przeczytane, nie przetwarzaj przeczytanych wiadomości" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Oznacz wiadomość, nie przetwarzaj oznaczonych wiadomości" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Przenieś do określonego folderu" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Usuń" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Użyj tematu jako tytułu" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Użyj nazwy pliku załącznika jako tytułu" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Nie przypisuj korespondenta" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Użyj adresu e-mail" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Użyj nazwy nadawcy (lub adresu e-mail, jeśli jest niedostępna)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Użyj korespondenta wybranego poniżej" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "kolejność" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "konto" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "folder" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Podfoldery muszą być oddzielone kropkami." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtruj po nadawcy" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtruj po temacie" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtruj po treści" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtruj po nazwie pliku załącznika" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Przetwarzaj tylko dokumenty, które całkowicie pasują do tej nazwy pliku, jeśli jest podana. Wzorce dopasowania jak *.pdf lub *faktura* są dozwolone. Wielkość liter nie jest rozróżniana." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "nie starsze niż" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "dni." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "typ załącznika" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Załączniki typu \"inline\" zawierają osadzone obrazy, więc najlepiej połączyć tę opcję z filtrem nazwy pliku." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "akcja" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parametr akcji" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Dodatkowy parametr dla akcji wybranej powyżej, tj. docelowy folder akcji \"Przenieś do określonego folderu\". Podfoldery muszą być oddzielone kropkami." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "przypisz tytuł" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "przypisz ten tag" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "przypisz ten typ dokumentu" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "przypisz korespondenta z" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "przypisz tego korespondenta" + diff --git a/src/locale/pt_BR/LC_MESSAGES/django.po b/src/locale/pt_BR/LC_MESSAGES/django.po new file mode 100644 index 000000000..d10d68364 --- /dev/null +++ b/src/locale/pt_BR/LC_MESSAGES/django.po @@ -0,0 +1,699 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Portuguese, Brazilian\n" +"Language: pt_BR\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: pt-BR\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documentos" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Qualquer palavra" + +#: documents/models.py:33 +msgid "All words" +msgstr "Todas as palavras" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Detecção exata" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Expressão regular" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Palavra difusa (fuzzy)" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automático" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nome" + +#: documents/models.py:45 +msgid "match" +msgstr "detecção" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algoritmo de detecção" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "diferencia maiúsculas de minúsculas" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "correspondente" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "correspondentes" + +#: documents/models.py:81 +msgid "color" +msgstr "cor" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "é etiqueta caixa de entrada" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marca essa etiqueta como caixa de entrada: Todos os novos documentos consumidos terão as etiquetas de caixa de entrada." + +#: documents/models.py:94 +msgid "tag" +msgstr "etiqueta" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "etiquetas" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "tipo de documento" + +#: documents/models.py:102 +msgid "document types" +msgstr "tipos de documento" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Não encriptado" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Encriptado com GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "título" + +#: documents/models.py:137 +msgid "content" +msgstr "conteúdo" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "O conteúdo de texto bruto do documento. Esse campo é usado principalmente para busca." + +#: documents/models.py:144 +msgid "mime type" +msgstr "tipo mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "some de verificação" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "A soma de verificação original do documento." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "Soma de verificação de arquivamento." + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "A soma de verificação do documento arquivado." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "criado" + +#: documents/models.py:176 +msgid "modified" +msgstr "modificado" + +#: documents/models.py:180 +msgid "storage type" +msgstr "tipo de armazenamento" + +#: documents/models.py:188 +msgid "added" +msgstr "adicionado" + +#: documents/models.py:192 +msgid "filename" +msgstr "nome do arquivo" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nome do arquivo atual armazenado" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nome do arquivo para arquivamento" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Nome do arquivo para arquivamento armazenado" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "número de sério de arquivamento" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "A posição deste documento no seu arquivamento físico." + +#: documents/models.py:223 +msgid "document" +msgstr "documento" + +#: documents/models.py:224 +msgid "documents" +msgstr "documentos" + +#: documents/models.py:311 +msgid "debug" +msgstr "debug" + +#: documents/models.py:312 +msgid "information" +msgstr "informação" + +#: documents/models.py:313 +msgid "warning" +msgstr "aviso" + +#: documents/models.py:314 +msgid "error" +msgstr "erro" + +#: documents/models.py:315 +msgid "critical" +msgstr "crítico" + +#: documents/models.py:319 +msgid "group" +msgstr "grupo" + +#: documents/models.py:322 +msgid "message" +msgstr "mensagem" + +#: documents/models.py:325 +msgid "level" +msgstr "nível" + +#: documents/models.py:332 +msgid "log" +msgstr "log" + +#: documents/models.py:333 +msgid "logs" +msgstr "logs" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "visualização" + +#: documents/models.py:345 +msgid "saved views" +msgstr "visualizações" + +#: documents/models.py:348 +msgid "user" +msgstr "usuário" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "exibir no painel de controle" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "exibir no painel lateral" + +#: documents/models.py:361 +msgid "sort field" +msgstr "ordenar campo" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "odernar reverso" + +#: documents/models.py:373 +msgid "title contains" +msgstr "título contém" + +#: documents/models.py:374 +msgid "content contains" +msgstr "conteúdo contém" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "NSA é" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "correspondente é" + +#: documents/models.py:377 +msgid "document type is" +msgstr "tipo de documento é" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "é caixa de entrada" + +#: documents/models.py:379 +msgid "has tag" +msgstr "contém etiqueta" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "contém qualquer etiqueta" + +#: documents/models.py:381 +msgid "created before" +msgstr "criado antes de" + +#: documents/models.py:382 +msgid "created after" +msgstr "criado depois de" + +#: documents/models.py:383 +msgid "created year is" +msgstr "ano de criação é" + +#: documents/models.py:384 +msgid "created month is" +msgstr "mês de criação é" + +#: documents/models.py:385 +msgid "created day is" +msgstr "dia de criação é" + +#: documents/models.py:386 +msgid "added before" +msgstr "adicionado antes de" + +#: documents/models.py:387 +msgid "added after" +msgstr "adicionado depois de" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modificado antes de" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modificado depois de" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "não tem etiqueta" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "título ou conteúdo contém" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "" + +#: documents/models.py:394 +msgid "more like this" +msgstr "" + +#: documents/models.py:405 +msgid "rule type" +msgstr "tipo de regra" + +#: documents/models.py:409 +msgid "value" +msgstr "valor" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "regra de filtragem" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "regras de filtragem" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Expressão regular inválida: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Cor inválida." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Tipo de arquivo %(type)s não suportado" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng está carregando..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng saiu" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Sua sessão foi encerrada com sucesso. Até mais!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Entre novamente" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Entrar no Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Por favor, entre na sua conta" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Seu usuário e senha estão incorretos. Por favor, tente novamente." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Usuário" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Senha" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Entrar" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Inglês (EUA)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Inglês (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Alemão" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Holandês" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Francês" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Português (Brasil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiano" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Romeno" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Administração do Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtro" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless processará somente e-mails que se encaixam em TODOS os filtros abaixo." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Ações" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "A ação se aplica ao e-mail. Essa ação só é executada quando documentos foram consumidos do e-mail. E-mails sem anexos permanecerão intactos." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadados" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Atribua metadados aos documentos consumidos por esta regra automaticamente. Se você não atribuir etiquetas, tipos ou correspondentes aqui, paperless ainda sim processará todas as regras de detecção que você definiu." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless mail" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "conta de e-mail" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "contas de e-mail" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Sem encriptação" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Usar SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Usar STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Servidor IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Porta IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "É geralmente 143 para não encriptado e conexões STARTTLS, e 993 para conexões SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "segurança IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "usuário" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "senha" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "regra de e-mail" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "regras de e-mail" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Processar somente anexos." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Processar todos os arquivos, incluindo anexos 'inline'." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Marcar como lido, não processar e-mails lidos" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Sinalizar o e-mail, não processar e-mails sinalizados" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Mover para pasta especificada" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Excluir" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Usar assunto como título" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Usar nome do arquivo anexo como título" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Não atribuir um correspondente" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Usar endereço de e-mail" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Usar nome (ou endereço de e-mail se não disponível)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Usar correspondente selecionado abaixo" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "ordem" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "conta" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "pasta" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrar de" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrar assunto" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrar corpo" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrar nome do arquivo anexo" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Consumir somente documentos que correspondem a este nome de arquivo se especificado.\n" +"Curingas como *.pdf ou *invoice* são permitidos. Sem diferenciação de maiúsculas e minúsculas." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "idade máxima" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Especificada em dias." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "tipo de anexo" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Anexos inline incluem imagens inseridas, por isso é melhor combinar essa opção com um filtro de nome de arquivo." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "ação" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parâmetro da ação" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "atribuir título de" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "atribuir esta etiqueta" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "atribuir este tipo de documento" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "atribuir correspondente de" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "atribuir este correspondente" + diff --git a/src/locale/pt_PT/LC_MESSAGES/django.po b/src/locale/pt_PT/LC_MESSAGES/django.po new file mode 100644 index 000000000..4f4f25968 --- /dev/null +++ b/src/locale/pt_PT/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 20:45\n" +"Last-Translator: \n" +"Language-Team: Portuguese\n" +"Language: pt_PT\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: pt-PT\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documentos" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Qualquer palavra" + +#: documents/models.py:33 +msgid "All words" +msgstr "Todas as palavras" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Detecção exata" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Expressão regular" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Palavra difusa (fuzzy)" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automático" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nome" + +#: documents/models.py:45 +msgid "match" +msgstr "correspondência" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algoritmo correspondente" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "é insensível" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "correspondente" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "correspondentes" + +#: documents/models.py:81 +msgid "color" +msgstr "cor" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "é etiqueta de novo" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marca esta etiqueta como uma etiqueta de entrada. Todos os documentos recentemente consumidos serão etiquetados com a etiqueta de entrada." + +#: documents/models.py:94 +msgid "tag" +msgstr "etiqueta" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "etiquetas" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "tipo de documento" + +#: documents/models.py:102 +msgid "document types" +msgstr "tipos de documento" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Não encriptado" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Encriptado com GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "título" + +#: documents/models.py:137 +msgid "content" +msgstr "conteúdo" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Os dados de texto, em cru, do documento. Este campo é utilizado principalmente para pesquisar." + +#: documents/models.py:144 +msgid "mime type" +msgstr "tipo mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "soma de verificação" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "A soma de verificação do documento original." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "arquivar soma de verificação" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "A soma de verificação do documento arquivado." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "criado" + +#: documents/models.py:176 +msgid "modified" +msgstr "modificado" + +#: documents/models.py:180 +msgid "storage type" +msgstr "tipo de armazenamento" + +#: documents/models.py:188 +msgid "added" +msgstr "adicionado" + +#: documents/models.py:192 +msgid "filename" +msgstr "nome de ficheiro" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nome do arquivo atual no armazenamento" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nome do ficheiro de arquivo" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Nome do arquivo atual em no armazenamento" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "numero de série de arquivo" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "A posição do documento no seu arquivo físico de documentos." + +#: documents/models.py:223 +msgid "document" +msgstr "documento" + +#: documents/models.py:224 +msgid "documents" +msgstr "documentos" + +#: documents/models.py:311 +msgid "debug" +msgstr "depurar" + +#: documents/models.py:312 +msgid "information" +msgstr "informação" + +#: documents/models.py:313 +msgid "warning" +msgstr "aviso" + +#: documents/models.py:314 +msgid "error" +msgstr "erro" + +#: documents/models.py:315 +msgid "critical" +msgstr "crítico" + +#: documents/models.py:319 +msgid "group" +msgstr "grupo" + +#: documents/models.py:322 +msgid "message" +msgstr "mensagem" + +#: documents/models.py:325 +msgid "level" +msgstr "nível" + +#: documents/models.py:332 +msgid "log" +msgstr "registo" + +#: documents/models.py:333 +msgid "logs" +msgstr "registos" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "vista guardada" + +#: documents/models.py:345 +msgid "saved views" +msgstr "vistas guardadas" + +#: documents/models.py:348 +msgid "user" +msgstr "utilizador" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "exibir no painel de controlo" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "mostrar na navegação lateral" + +#: documents/models.py:361 +msgid "sort field" +msgstr "ordenar campo" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "ordenar inversamente" + +#: documents/models.py:373 +msgid "title contains" +msgstr "o título contém" + +#: documents/models.py:374 +msgid "content contains" +msgstr "o conteúdo contém" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "O NSA é" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "o correspondente é" + +#: documents/models.py:377 +msgid "document type is" +msgstr "o tipo de documento é" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "está na entrada" + +#: documents/models.py:379 +msgid "has tag" +msgstr "tem etiqueta" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "tem qualquer etiqueta" + +#: documents/models.py:381 +msgid "created before" +msgstr "criado antes" + +#: documents/models.py:382 +msgid "created after" +msgstr "criado depois" + +#: documents/models.py:383 +msgid "created year is" +msgstr "ano criada é" + +#: documents/models.py:384 +msgid "created month is" +msgstr "mês criado é" + +#: documents/models.py:385 +msgid "created day is" +msgstr "dia criado é" + +#: documents/models.py:386 +msgid "added before" +msgstr "adicionada antes" + +#: documents/models.py:387 +msgid "added after" +msgstr "adicionado depois de" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modificado antes de" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modificado depois de" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "não tem etiqueta" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "não possui um NSA" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "título ou conteúdo contém" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "consulta de texto completo" + +#: documents/models.py:394 +msgid "more like this" +msgstr "mais como este" + +#: documents/models.py:405 +msgid "rule type" +msgstr "tipo de regra" + +#: documents/models.py:409 +msgid "value" +msgstr "valor" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "regra de filtragem" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "regras de filtragem" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Expressão regular inválida: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Cor invalida." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Tipo de arquivo %(type)s não suportado" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "O paperless-ng está a carregar..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng com sessão terminada" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Terminou a sessão com sucesso. Adeus!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Iniciar sessão novamente" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Inicio de sessão Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Por favor inicie sessão." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "O utilizador e a senha não correspondem. Tente novamente." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Nome de utilizador" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Palavra-passe" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Iniciar sessão" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Inglês (EUA)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "English (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Deutsch" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Nederlandse" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Français" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Português (Brasil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Português" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiano" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Romeno" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Russo" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Espanhol" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polaco" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Sueco" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Administração do Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Autenticação" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Definições avançadas" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtro" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "O Paperless apenas irá processar emails que coincidem com TODOS os filtros dados abaixo." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Ações" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "A ação aplicada a correio. Esta ação apenas será efetuada com documentos que tenham sido consumidos através do correio. E-mails sem anexos permanecerão intactos." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadados" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Atribuir meta-dados aos documentos consumidos automaticamente através desta regra. Se você não atribuir etiquetas, tipos ou correspondentes aqui, o paperless ainda assim processará todas as regras correspondentes que tenha definido." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Correio Paperless" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "conta de email" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "contas de email" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Sem encriptação" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Utilizar SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Utilizar STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Servidor IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Porto IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Por norma é o 143 sem encriptação e conexões STARTTLS, e o 993 para conexões com SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Segurança IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "nome de utilizador" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "palavra-passe" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "conjunto de caracteres" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "O conjunto de caracteres a utilizar ao comunicar com um servidor de email, tal como 'UTF-8' ou 'US-ASCII'." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "regra de correio" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "regras de correio" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Processar anexos apenas." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Processar todos os ficheiros, incluindo ficheiros 'embutidos (inline)'." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Marcar como lido, não processar emails lidos" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Marcar o email, não processar emails marcados" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Mover para uma diretoria específica" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Excluir" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Utilizar o assunto como título" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Utilizar o nome do anexo como título" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Não atribuir um correspondente" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Utilizar o endereço de email" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Utilizar nome (ou endereço de email se não disponível)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Utilizar o correspondente selecionado abaixo" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "ordem" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "conta" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "directoria" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Sub-pastas devem ser separadas por pontos." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrar de" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrar assunto" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrar corpo" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrar nome do arquivo anexo" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Consumir apenas documentos que correspondam inteiramente ao nome de arquivo se especificado. Genéricos como *.pdf ou *fatura* são permitidos. Não é sensível a letras maiúsculas/minúsculas." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "idade máxima" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Especificado em dias." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "tipo de anexo" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Anexos embutidos incluem imagens incorporadas, por isso é melhor combinar esta opção com um filtro de nome do arquivo." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "ação" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parâmetro de ação" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Parâmetros adicionais para a ação selecionada acima, isto é, a pasta alvo da ação mover para pasta. Sub-pastas devem ser separadas por pontos." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "atribuir titulo de" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "atribuir esta etiqueta" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "atribuir este tipo de documento" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "atribuir correspondente de" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "atribuir este correspondente" + diff --git a/src/locale/ro_RO/LC_MESSAGES/django.po b/src/locale/ro_RO/LC_MESSAGES/django.po new file mode 100644 index 000000000..1fbb3da38 --- /dev/null +++ b/src/locale/ro_RO/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-08-16 09:06\n" +"Last-Translator: \n" +"Language-Team: Romanian\n" +"Language: ro_RO\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=3; plural=(n==1 ? 0 : (n==0 || (n%100>0 && n%100<20)) ? 1 : 2);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: ro\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Documente" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Orice cuvânt" + +#: documents/models.py:33 +msgid "All words" +msgstr "Toate cuvintele" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Potrivire exactă" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Expresie regulată" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Mod neatent" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automat" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "nume" + +#: documents/models.py:45 +msgid "match" +msgstr "potrivire" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "algoritm de potrivire" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "nu ține cont de majuscule" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "corespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "corespondenți" + +#: documents/models.py:81 +msgid "color" +msgstr "culoare" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "este etichetă inbox" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Marchează aceasta eticheta ca etichetă inbox: Toate documentele nou consumate primesc aceasta eticheta." + +#: documents/models.py:94 +msgid "tag" +msgstr "etichetă" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "etichete" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "tip de document" + +#: documents/models.py:102 +msgid "document types" +msgstr "tipuri de document" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Necriptat" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Criptat cu GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titlu" + +#: documents/models.py:137 +msgid "content" +msgstr "conținut" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Textul brut al documentului. Acest camp este folosit in principal pentru căutare." + +#: documents/models.py:144 +msgid "mime type" +msgstr "tip MIME" + +#: documents/models.py:155 +msgid "checksum" +msgstr "sumă de control" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Suma de control a documentului original." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "suma de control a arhivei" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Suma de control a documentului arhivat." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "creat" + +#: documents/models.py:176 +msgid "modified" +msgstr "modificat" + +#: documents/models.py:180 +msgid "storage type" +msgstr "tip de stocare" + +#: documents/models.py:188 +msgid "added" +msgstr "adăugat" + +#: documents/models.py:192 +msgid "filename" +msgstr "nume fișier" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Numele curent al fișierului stocat" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "nume fișier arhiva" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Numele curent al arhivei stocate" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "număr serial in arhiva" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Poziția acestui document in arhiva fizica." + +#: documents/models.py:223 +msgid "document" +msgstr "document" + +#: documents/models.py:224 +msgid "documents" +msgstr "documente" + +#: documents/models.py:311 +msgid "debug" +msgstr "depanare" + +#: documents/models.py:312 +msgid "information" +msgstr "informații" + +#: documents/models.py:313 +msgid "warning" +msgstr "avertizare" + +#: documents/models.py:314 +msgid "error" +msgstr "eroare" + +#: documents/models.py:315 +msgid "critical" +msgstr "critic" + +#: documents/models.py:319 +msgid "group" +msgstr "grup" + +#: documents/models.py:322 +msgid "message" +msgstr "mesaj" + +#: documents/models.py:325 +msgid "level" +msgstr "nivel" + +#: documents/models.py:332 +msgid "log" +msgstr "jurnal" + +#: documents/models.py:333 +msgid "logs" +msgstr "jurnale" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "vizualizare" + +#: documents/models.py:345 +msgid "saved views" +msgstr "vizualizări" + +#: documents/models.py:348 +msgid "user" +msgstr "utilizator" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "afișează pe tabloul de bord" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "afișează in bara laterala" + +#: documents/models.py:361 +msgid "sort field" +msgstr "sortează camp" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "sortează invers" + +#: documents/models.py:373 +msgid "title contains" +msgstr "titlul conține" + +#: documents/models.py:374 +msgid "content contains" +msgstr "conținutul conține" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "Avizul prealabil de expediție este" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "corespondentul este" + +#: documents/models.py:377 +msgid "document type is" +msgstr "tipul documentului este" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "este în inbox" + +#: documents/models.py:379 +msgid "has tag" +msgstr "are eticheta" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "are orice eticheta" + +#: documents/models.py:381 +msgid "created before" +msgstr "creat înainte de" + +#: documents/models.py:382 +msgid "created after" +msgstr "creat după" + +#: documents/models.py:383 +msgid "created year is" +msgstr "anul creării este" + +#: documents/models.py:384 +msgid "created month is" +msgstr "luna creării este" + +#: documents/models.py:385 +msgid "created day is" +msgstr "ziua creării este" + +#: documents/models.py:386 +msgid "added before" +msgstr "adăugat înainte de" + +#: documents/models.py:387 +msgid "added after" +msgstr "adăugat după" + +#: documents/models.py:388 +msgid "modified before" +msgstr "modificat înainte de" + +#: documents/models.py:389 +msgid "modified after" +msgstr "modificat după" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "nu are etichetă" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "nu are aviz prealabil de expediție" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "titlul sau conținutul conține" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "query fulltext" + +#: documents/models.py:394 +msgid "more like this" +msgstr "mai multe ca aceasta" + +#: documents/models.py:405 +msgid "rule type" +msgstr "tip de regula" + +#: documents/models.py:409 +msgid "value" +msgstr "valoare" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "regulă de filtrare" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "reguli de filtrare" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Expresie regulată invalida: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Culoare invalidă." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Tip de fișier %(type)s nesuportat" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng se încarca..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng s-a deconectat" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Ați fost deconectat cu succes. La revedere!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Conectați-vă din nou" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Conectare Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Vă rugăm conectați-vă." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Numele si parola nu sunt corecte. Vă rugăm incercați din nou." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Nume" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Parolă" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Conectare" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Engleză (Americană)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Engleză (Britanică)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Germană" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Olandeză" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Franceză" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugheză (Brazilia)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugheză" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italiană" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Română" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Rusă" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spaniolă" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Poloneză" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Suedeză" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Administrare Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Autentificare" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Setări avansate" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filtru" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless va procesa doar mail-urile care corespund TUTUROR filtrelor date mai jos." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Acțiuni" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Acțiunea aplicată tuturor email-urilor. Aceasta este realizată doar când sunt consumate documente din email. Cele fara atașamente nu vor fi procesate." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadate" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Atribuie metadate documentelor consumate prin aceasta regula în mod automat. Chiar dacă nu sunt atribuite etichete, tipuri sau corespondenți, Paperless va procesa toate regulile definite care se potrivesc." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Email Paperless" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "cont de email" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "conturi de email" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Fără criptare" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Folosește SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Folosește STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "server IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "port IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "De obicei este 143 pentru conexiuni necriptate și STARTTLS, sau 993 pentru conexiuni SSL." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "securitate IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "nume" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "parolă" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Set de caractere" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Setul de caractere folosit la comunicarea cu serverul de e-mail, cum ar fi \"UTF-8\" sau \"US-ASCII\"." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "regulă email" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "reguli email" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Procesează doar atașamentele." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Procesează toate fișierele, inclusiv atașamentele „inline”." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Marchează ca citit, nu procesa email-uri citite" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Marchează, nu procesa email-uri marcate" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Mută în directorul specificat" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Șterge" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Utilizează subiectul ca titlu" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Utilizează numele fișierului atașat ca titlu" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Nu atribui un corespondent" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Folosește adresa de email" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Folosește numele (dacă nu exista, folosește adresa de email)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Folosește corespondentul selectat mai jos" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "ordonează" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "cont" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "director" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Subdosarele trebuie separate prin puncte." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrează de la" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrează subiect" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrează corpul email-ului" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrează numele fișierului atașat" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Consumă doar documentele care se potrivesc în întregime cu acest nume de fișier, dacă este specificat. Simbolul * ține locul oricărui șir de caractere. Majusculele nu contează." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "vârsta maximă" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Specificată in zile." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "tip atașament" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Atașamentele \"inline\" includ și imaginile încorporate, deci această opțiune funcționează cel mai bine combinată cu un filtru pentru numele fișierului." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "acţiune" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "parametru acțiune" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Parametru adițional pentru acțiunea definită mai sus (ex. directorul în care să se realizeze o mutare). Subdosarele trebuie separate prin puncte." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "atribuie titlu din" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "atribuie această etichetă" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "atribuie acest tip" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "atribuie corespondent din" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "atribuie acest corespondent" + diff --git a/src/locale/ru_RU/LC_MESSAGES/django.po b/src/locale/ru_RU/LC_MESSAGES/django.po new file mode 100644 index 000000000..2e38b0737 --- /dev/null +++ b/src/locale/ru_RU/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Russian\n" +"Language: ru_RU\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=4; plural=((n%10==1 && n%100!=11) ? 0 : ((n%10 >= 2 && n%10 <=4 && (n%100 < 12 || n%100 > 14)) ? 1 : ((n%10 == 0 || (n%10 >= 5 && n%10 <=9)) || (n%100 >= 11 && n%100 <= 14)) ? 2 : 3));\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: ru\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Документы" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Любые слова" + +#: documents/models.py:33 +msgid "All words" +msgstr "Все слова" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Точное соответствие" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Регулярное выражение" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "\"Нечёткий\" режим" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Автоматически" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "имя" + +#: documents/models.py:45 +msgid "match" +msgstr "соответствие" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "алгоритм сопоставления" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "без учёта регистра" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "корреспондент" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "корреспонденты" + +#: documents/models.py:81 +msgid "color" +msgstr "цвет" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "это входящий тег" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Отметить этот тег как «Входящий»: все вновь добавленные документы будут помечены тегами «Входящие»." + +#: documents/models.py:94 +msgid "tag" +msgstr "тег" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "теги" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "тип документа" + +#: documents/models.py:102 +msgid "document types" +msgstr "типы документов" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "не зашифровано" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Зашифровано с помощью GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "заголовок" + +#: documents/models.py:137 +msgid "content" +msgstr "содержимое" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Это поле используется в основном для поиска." + +#: documents/models.py:144 +msgid "mime type" +msgstr "тип Mime" + +#: documents/models.py:155 +msgid "checksum" +msgstr "контрольная сумма" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Контрольная сумма оригинального документа." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "контрольная сумма архива" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Контрольная сумма архивного документа." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "создано" + +#: documents/models.py:176 +msgid "modified" +msgstr "изменено" + +#: documents/models.py:180 +msgid "storage type" +msgstr "тип хранилища" + +#: documents/models.py:188 +msgid "added" +msgstr "добавлено" + +#: documents/models.py:192 +msgid "filename" +msgstr "имя файла" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Текущее имя файла в хранилище" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "имя файла архива" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Текущее имя файла архива в хранилище" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "архивный номер (АН)" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Позиция этого документа в вашем физическом архиве документов." + +#: documents/models.py:223 +msgid "document" +msgstr "документ" + +#: documents/models.py:224 +msgid "documents" +msgstr "документы" + +#: documents/models.py:311 +msgid "debug" +msgstr "отладка" + +#: documents/models.py:312 +msgid "information" +msgstr "информация" + +#: documents/models.py:313 +msgid "warning" +msgstr "предупреждение" + +#: documents/models.py:314 +msgid "error" +msgstr "ошибка" + +#: documents/models.py:315 +msgid "critical" +msgstr "критическая" + +#: documents/models.py:319 +msgid "group" +msgstr "группа" + +#: documents/models.py:322 +msgid "message" +msgstr "сообщение" + +#: documents/models.py:325 +msgid "level" +msgstr "уровень" + +#: documents/models.py:332 +msgid "log" +msgstr "журнал" + +#: documents/models.py:333 +msgid "logs" +msgstr "логи" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "сохранённое представление" + +#: documents/models.py:345 +msgid "saved views" +msgstr "сохраненные представления" + +#: documents/models.py:348 +msgid "user" +msgstr "пользователь" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "показать на панели" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "показать в боковой панели" + +#: documents/models.py:361 +msgid "sort field" +msgstr "Поле сортировки" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "обратная сортировка" + +#: documents/models.py:373 +msgid "title contains" +msgstr "заголовок содержит" + +#: documents/models.py:374 +msgid "content contains" +msgstr "содержимое содержит" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "АН" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "корреспондент" + +#: documents/models.py:377 +msgid "document type is" +msgstr "тип документа" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "во входящих" + +#: documents/models.py:379 +msgid "has tag" +msgstr "есть тег" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "есть любой тег" + +#: documents/models.py:381 +msgid "created before" +msgstr "создан до" + +#: documents/models.py:382 +msgid "created after" +msgstr "создан после" + +#: documents/models.py:383 +msgid "created year is" +msgstr "год создания" + +#: documents/models.py:384 +msgid "created month is" +msgstr "месяц создания" + +#: documents/models.py:385 +msgid "created day is" +msgstr "день создания" + +#: documents/models.py:386 +msgid "added before" +msgstr "добавлен до" + +#: documents/models.py:387 +msgid "added after" +msgstr "добавлен после" + +#: documents/models.py:388 +msgid "modified before" +msgstr "изменен до" + +#: documents/models.py:389 +msgid "modified after" +msgstr "изменен после" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "не имеет тега" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "не имеет архивного номера" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "Название или содержимое включает" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "полнотекстовый запрос" + +#: documents/models.py:394 +msgid "more like this" +msgstr "больше похожих" + +#: documents/models.py:405 +msgid "rule type" +msgstr "Тип правила" + +#: documents/models.py:409 +msgid "value" +msgstr "значение" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "Правило фильтрации" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "правила фильтрации" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "неверное регулярное выражение: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Неверный цвет." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Тип файла %(type)s не поддерживается" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng загружается..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Выполнен выход из Paperless-ng" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Вы успешно вышли из системы. До свидания!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Войти снова" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Выполнен выход в Paperless-ng" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Пожалуйста, войдите." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Неправильные имя пользователя или пароль! Попробуйте еще раз." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Имя пользователя" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Пароль" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Вход" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Английский (США)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Английский (Великобритании)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Немецкий" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Датский" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Французский" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portuguese (Brazil)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Португальский" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italian" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Romanian" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Русский" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Испанский" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Администрирование Paperless-ng" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Фильтр" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless будет обрабатывать только те письма, которые соответствуют всем фильтрам, указанным ниже." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Действия" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Действие применено к письму. Это действие применяется только при обработке документов из почты. Сообщения без вложений не обрабатываются." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Метаданные" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Автоматически назначать метаданные документам, полученным из этого правила. Если вы не назначаете здесь теги, типы или корреспонденты, paperless все равно будут обрабатывать все соответствующие правила, которые вы определили." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Безбумажная почта" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "почтовый ящик" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "Почтовые ящики" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Без шифрования" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Использовать SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Использовать STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "Сервер IMAP" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "Порт IMAP" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Обычно это 143 для нешифрованных и STARTTLS соединений и 993 для SSL-соединений." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "Безопасность IMAP" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "Имя пользователя" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "пароль" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "правило почты" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "правила почты" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Обрабатывать только вложения." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Обрабатывать все файлы, включая 'встроенные' вложения." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Пометить как прочитанное, не обрабатывать прочитанные письма" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Пометить почту, не обрабатывать помеченные письма" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Переместить в указанную папку" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Удалить" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Тема в качестве заголовка" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Использовать имя вложенного файла как заголовок" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Не назначать корреспондента" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Использовать email адрес" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Использовать имя (или адрес электронной почты, если недоступно)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Использовать корреспондента, выбранного ниже" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "порядок" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "Учётная запись" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "каталог" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "фильтр по отправителю" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "фильтр по теме" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "фильтр по тексту сообщения" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "фильтр по имени вложения" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Обрабатывать только документы, которые полностью совпадают с именем файла (если оно указано). Маски, например *.pdf или *счет*, разрешены. Без учёта регистра." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "Максимальный возраст" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Указывается в днях." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "Тип вложения" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Вложенные вложения включая встраиваемые изображения. Лучше совместить эту опцию с фильтром по имени вложения." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "действие" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "параметр действия" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "назначить заголовок из" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "назначить этот тег" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "назначить этот тип документа" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "назначить корреспондента из" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "назначить этого корреспондента" + diff --git a/src/locale/sv_SE/LC_MESSAGES/django.po b/src/locale/sv_SE/LC_MESSAGES/django.po new file mode 100644 index 000000000..05219f931 --- /dev/null +++ b/src/locale/sv_SE/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-07-30 18:02\n" +"Last-Translator: \n" +"Language-Team: Swedish\n" +"Language: sv_SE\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: sv-SE\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "Dokument" + +#: documents/models.py:32 +msgid "Any word" +msgstr "Valfritt ord" + +#: documents/models.py:33 +msgid "All words" +msgstr "Alla ord" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "Exakt matchning" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "Reguljära uttryck" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "Ungefärligt ord" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "Automatisk" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "namn" + +#: documents/models.py:45 +msgid "match" +msgstr "träff" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "matchande algoritm" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "är ej skiftlägeskänsligt" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "korrespondent" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "korrespondenter" + +#: documents/models.py:81 +msgid "color" +msgstr "färg" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "är inkorgsetikett" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "Markerar denna etikett som en inkorgsetikett: Alla nyligen konsumerade dokument kommer att märkas med inkorgsetiketter." + +#: documents/models.py:94 +msgid "tag" +msgstr "etikett" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "etiketter" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "dokumenttyp" + +#: documents/models.py:102 +msgid "document types" +msgstr "dokumenttyper" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "Okrypterad" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "Krypterad med GNU Privacy Guard" + +#: documents/models.py:124 +msgid "title" +msgstr "titel" + +#: documents/models.py:137 +msgid "content" +msgstr "innehåll" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "Dokumentets obearbetade textdata. Detta fält används främst för sökning." + +#: documents/models.py:144 +msgid "mime type" +msgstr "MIME-typ" + +#: documents/models.py:155 +msgid "checksum" +msgstr "kontrollsumma" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "Kontrollsumman för originaldokumentet." + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "arkivera kontrollsumma" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "Kontrollsumman för det arkiverade dokumentet." + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "skapad" + +#: documents/models.py:176 +msgid "modified" +msgstr "ändrad" + +#: documents/models.py:180 +msgid "storage type" +msgstr "lagringstyp" + +#: documents/models.py:188 +msgid "added" +msgstr "tillagd" + +#: documents/models.py:192 +msgid "filename" +msgstr "filnamn" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "Nuvarande filnamn i lagringsutrymmet" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "arkivfilnamn" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "Nuvarande arkivfilnamn i lagringsutrymmet" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "serienummer (arkivering)" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "Placeringen av detta dokument i ditt fysiska dokumentarkiv." + +#: documents/models.py:223 +msgid "document" +msgstr "dokument" + +#: documents/models.py:224 +msgid "documents" +msgstr "dokument" + +#: documents/models.py:311 +msgid "debug" +msgstr "felsök" + +#: documents/models.py:312 +msgid "information" +msgstr "information" + +#: documents/models.py:313 +msgid "warning" +msgstr "varning" + +#: documents/models.py:314 +msgid "error" +msgstr "fel" + +#: documents/models.py:315 +msgid "critical" +msgstr "kritisk" + +#: documents/models.py:319 +msgid "group" +msgstr "grupp" + +#: documents/models.py:322 +msgid "message" +msgstr "meddelande" + +#: documents/models.py:325 +msgid "level" +msgstr "nivå" + +#: documents/models.py:332 +msgid "log" +msgstr "logg" + +#: documents/models.py:333 +msgid "logs" +msgstr "loggar" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "sparad vy" + +#: documents/models.py:345 +msgid "saved views" +msgstr "sparade vyer" + +#: documents/models.py:348 +msgid "user" +msgstr "användare" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "visa på kontrollpanelen" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "visa i sidofältet" + +#: documents/models.py:361 +msgid "sort field" +msgstr "sortera fält" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "sortera omvänt" + +#: documents/models.py:373 +msgid "title contains" +msgstr "titel innehåller" + +#: documents/models.py:374 +msgid "content contains" +msgstr "innehåll innehåller" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "ASN är" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "korrespondent är" + +#: documents/models.py:377 +msgid "document type is" +msgstr "dokumenttyp är" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "är i inkorgen" + +#: documents/models.py:379 +msgid "has tag" +msgstr "har etikett" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "har någon etikett" + +#: documents/models.py:381 +msgid "created before" +msgstr "skapad före" + +#: documents/models.py:382 +msgid "created after" +msgstr "skapad efter" + +#: documents/models.py:383 +msgid "created year is" +msgstr "skapat år är" + +#: documents/models.py:384 +msgid "created month is" +msgstr "skapad månad är" + +#: documents/models.py:385 +msgid "created day is" +msgstr "skapad dag är" + +#: documents/models.py:386 +msgid "added before" +msgstr "tillagd före" + +#: documents/models.py:387 +msgid "added after" +msgstr "tillagd efter" + +#: documents/models.py:388 +msgid "modified before" +msgstr "ändrad före" + +#: documents/models.py:389 +msgid "modified after" +msgstr "ändrad efter" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "har inte etikett" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "har inte ASN" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "titel eller innehåll innehåller" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "fulltextfråga" + +#: documents/models.py:394 +msgid "more like this" +msgstr "mer som detta" + +#: documents/models.py:405 +msgid "rule type" +msgstr "regeltyp" + +#: documents/models.py:409 +msgid "value" +msgstr "värde" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "filtrera regel" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "filtrera regler" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "Ogiltigt reguljärt uttryck: %(error)s" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "Ogiltig färg." + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "Filtypen %(type)s stöds inte" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "Paperless-ng laddas..." + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "Paperless-ng utloggad" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "Du är nu utloggad. Hejdå!" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "Logga in igen" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "Paperless-ng inloggning" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "Vänligen logga in." + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "Ditt användarnamn och lösenord matchade inte. Vänligen försök igen." + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "Användarnamn" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "Lösenord" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "Logga in" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "Engelska (USA)" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "Engelska (GB)" + +#: paperless/settings.py:305 +msgid "German" +msgstr "Tyska" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "Holländska" + +#: paperless/settings.py:307 +msgid "French" +msgstr "Franska" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "Portugisiska (Brasilien)" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "Portugisiska" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "Italienska" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "Rumänska" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "Ryska" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "Spanska" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "Polska" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "Svenska" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "Paperless-ng administration" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "Autentisering" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "Avancerade inställningar" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "Filter" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "Paperless kommer endast att behandla e-postmeddelanden som matchar ALLA filter som anges nedan." + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "Åtgärder" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "Åtgärden tillämpas på e-postmeddelandet. Denna åtgärd utförs endast när dokument konsumerades från e-postmeddelandet. E-post utan bilagor kommer att förbli helt orörda." + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "Metadata" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "Tilldela metadata till dokument som konsumeras från denna regel automatiskt. Om du inte tilldelar etiketter, typer eller korrespondenter här kommer paperless fortfarande att behandla alla matchande regler som du har definierat." + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "Paperless e-post" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "e-postkonto" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "e-postkonton" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "Ingen kryptering" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "Använd SSL" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "Använd STARTTLS" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "IMAP-server" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "IMAP-port" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "Detta är vanligtvis 143 för okrypterade och STARTTLS-anslutningar, och 993 för SSL-anslutningar." + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "IMAP-säkerhet" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "användarnamn" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "lösenord" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "Teckenuppsättning" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "Teckenuppsättningen som är tänkt att användas vid kommunikation med mailservern, exempelvis ’UTF-8’ eller ’US-ASCII’." + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "e-postregel" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "e-postregler" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "Behandla endast bilagor." + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "Behandla alla filer, inklusive infogade bilagor." + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "Markera som läst, bearbeta inte lästa meddelanden" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "Flagga meddelandet, bearbeta inte flaggade meddelanden" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "Flytta till angiven mapp" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "Radera" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "Använd ämne som titel" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "Använd bilagans filnamn som titel" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "Tilldela inte en korrespondent" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "Använd e-postadress" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "Använd namn (eller e-postadress om inte tillgängligt)" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "Använd korrespondent som valts nedan" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "ordning" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "konto" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "mapp" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "Undermappar måste vara separerade med punkter." + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "filtrera från" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "filtrera ämne" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "filtrera kropp" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "filtrera filnamn för bilaga" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "Konsumera endast dokument som matchar exakt detta filnamn, om det är angivet. Jokertecken som *.pdf eller *faktura* är tillåtna. Ej skiftlägeskänsligt." + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "högsta ålder" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "Anges i dagar." + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "typ av bilaga" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "Infogade bilagor inkluderar inbäddade bilder, så det är bäst att kombinera detta alternativ med ett filnamnsfilter." + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "åtgärd" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "åtgärdsparameter" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "Ytterligare parametrar för åtgärden som valts ovan, d.v.s. målmappen för åtgärden \"flytta till angiven mapp\". Undermappar måste vara separerade med punkter." + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "tilldela titel från" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "tilldela denna etikett" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "tilldela den här dokumenttypen" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "tilldela korrespondent från" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "tilldela denna korrespondent" + diff --git a/src/locale/th_TH/LC_MESSAGES/django.po b/src/locale/th_TH/LC_MESSAGES/django.po new file mode 100644 index 000000000..20c70dd69 --- /dev/null +++ b/src/locale/th_TH/LC_MESSAGES/django.po @@ -0,0 +1,642 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-02-28 12:40+0100\n" +"PO-Revision-Date: 2021-03-06 21:39\n" +"Last-Translator: \n" +"Language-Team: Thai\n" +"Language: th_TH\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=1; plural=0;\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: th\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "" + +#: documents/models.py:32 +msgid "Any word" +msgstr "" + +#: documents/models.py:33 +msgid "All words" +msgstr "" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:109 +msgid "name" +msgstr "" + +#: documents/models.py:45 +msgid "match" +msgstr "" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "" + +#: documents/models.py:81 +msgid "color" +msgstr "" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "" + +#: documents/models.py:94 +msgid "tag" +msgstr "" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "" + +#: documents/models.py:102 +msgid "document types" +msgstr "" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "" + +#: documents/models.py:124 +msgid "title" +msgstr "" + +#: documents/models.py:137 +msgid "content" +msgstr "" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "" + +#: documents/models.py:144 +msgid "mime type" +msgstr "" + +#: documents/models.py:155 +msgid "checksum" +msgstr "" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "" + +#: documents/models.py:176 +msgid "modified" +msgstr "" + +#: documents/models.py:180 +msgid "storage type" +msgstr "" + +#: documents/models.py:188 +msgid "added" +msgstr "" + +#: documents/models.py:192 +msgid "filename" +msgstr "" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "" + +#: documents/models.py:223 +msgid "document" +msgstr "" + +#: documents/models.py:224 +msgid "documents" +msgstr "" + +#: documents/models.py:311 +msgid "debug" +msgstr "" + +#: documents/models.py:312 +msgid "information" +msgstr "" + +#: documents/models.py:313 +msgid "warning" +msgstr "" + +#: documents/models.py:314 +msgid "error" +msgstr "" + +#: documents/models.py:315 +msgid "critical" +msgstr "" + +#: documents/models.py:319 +msgid "group" +msgstr "" + +#: documents/models.py:322 +msgid "message" +msgstr "" + +#: documents/models.py:325 +msgid "level" +msgstr "" + +#: documents/models.py:332 +msgid "log" +msgstr "" + +#: documents/models.py:333 +msgid "logs" +msgstr "" + +#: documents/models.py:344 documents/models.py:394 +msgid "saved view" +msgstr "" + +#: documents/models.py:345 +msgid "saved views" +msgstr "" + +#: documents/models.py:348 +msgid "user" +msgstr "" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "" + +#: documents/models.py:361 +msgid "sort field" +msgstr "" + +#: documents/models.py:364 +msgid "sort reverse" +msgstr "" + +#: documents/models.py:370 +msgid "title contains" +msgstr "" + +#: documents/models.py:371 +msgid "content contains" +msgstr "" + +#: documents/models.py:372 +msgid "ASN is" +msgstr "" + +#: documents/models.py:373 +msgid "correspondent is" +msgstr "" + +#: documents/models.py:374 +msgid "document type is" +msgstr "" + +#: documents/models.py:375 +msgid "is in inbox" +msgstr "" + +#: documents/models.py:376 +msgid "has tag" +msgstr "" + +#: documents/models.py:377 +msgid "has any tag" +msgstr "" + +#: documents/models.py:378 +msgid "created before" +msgstr "" + +#: documents/models.py:379 +msgid "created after" +msgstr "" + +#: documents/models.py:380 +msgid "created year is" +msgstr "" + +#: documents/models.py:381 +msgid "created month is" +msgstr "" + +#: documents/models.py:382 +msgid "created day is" +msgstr "" + +#: documents/models.py:383 +msgid "added before" +msgstr "" + +#: documents/models.py:384 +msgid "added after" +msgstr "" + +#: documents/models.py:385 +msgid "modified before" +msgstr "" + +#: documents/models.py:386 +msgid "modified after" +msgstr "" + +#: documents/models.py:387 +msgid "does not have tag" +msgstr "" + +#: documents/models.py:398 +msgid "rule type" +msgstr "" + +#: documents/models.py:402 +msgid "value" +msgstr "" + +#: documents/models.py:408 +msgid "filter rule" +msgstr "" + +#: documents/models.py:409 +msgid "filter rules" +msgstr "" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expresssion: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "" + +#: documents/templates/index.html:21 +msgid "Paperless-ng is loading..." +msgstr "" + +#: documents/templates/registration/logged_out.html:13 +msgid "Paperless-ng signed out" +msgstr "" + +#: documents/templates/registration/logged_out.html:41 +msgid "You have been successfully logged out. Bye!" +msgstr "" + +#: documents/templates/registration/logged_out.html:42 +msgid "Sign in again" +msgstr "" + +#: documents/templates/registration/login.html:13 +msgid "Paperless-ng sign in" +msgstr "" + +#: documents/templates/registration/login.html:42 +msgid "Please sign in." +msgstr "" + +#: documents/templates/registration/login.html:45 +msgid "Your username and password didn't match. Please try again." +msgstr "" + +#: documents/templates/registration/login.html:48 +msgid "Username" +msgstr "" + +#: documents/templates/registration/login.html:49 +msgid "Password" +msgstr "" + +#: documents/templates/registration/login.html:54 +msgid "Sign in" +msgstr "" + +#: paperless/settings.py:297 +msgid "English (US)" +msgstr "" + +#: paperless/settings.py:298 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:299 +msgid "German" +msgstr "" + +#: paperless/settings.py:300 +msgid "Dutch" +msgstr "" + +#: paperless/settings.py:301 +msgid "French" +msgstr "" + +#: paperless/settings.py:302 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:303 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:304 +msgid "Romanian" +msgstr "" + +#: paperless/urls.py:118 +msgid "Paperless-ng administration" +msgstr "" + +#: paperless_mail/admin.py:25 +msgid "Filter" +msgstr "" + +#: paperless_mail/admin.py:27 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Actions" +msgstr "" + +#: paperless_mail/admin.py:39 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "" + +#: paperless_mail/admin.py:46 +msgid "Metadata" +msgstr "" + +#: paperless_mail/admin.py:48 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "" + +#: paperless_mail/models.py:60 +msgid "mail rule" +msgstr "" + +#: paperless_mail/models.py:61 +msgid "mail rules" +msgstr "" + +#: paperless_mail/models.py:67 +msgid "Only process attachments." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "Process all files, including 'inline' attachments." +msgstr "" + +#: paperless_mail/models.py:78 +msgid "Mark as read, don't process read mails" +msgstr "" + +#: paperless_mail/models.py:79 +msgid "Flag the mail, don't process flagged mails" +msgstr "" + +#: paperless_mail/models.py:80 +msgid "Move to specified folder" +msgstr "" + +#: paperless_mail/models.py:81 +msgid "Delete" +msgstr "" + +#: paperless_mail/models.py:88 +msgid "Use subject as title" +msgstr "" + +#: paperless_mail/models.py:89 +msgid "Use attachment filename as title" +msgstr "" + +#: paperless_mail/models.py:99 +msgid "Do not assign a correspondent" +msgstr "" + +#: paperless_mail/models.py:101 +msgid "Use mail address" +msgstr "" + +#: paperless_mail/models.py:103 +msgid "Use name (or mail address if not available)" +msgstr "" + +#: paperless_mail/models.py:105 +msgid "Use correspondent selected below" +msgstr "" + +#: paperless_mail/models.py:113 +msgid "order" +msgstr "" + +#: paperless_mail/models.py:120 +msgid "account" +msgstr "" + +#: paperless_mail/models.py:124 +msgid "folder" +msgstr "" + +#: paperless_mail/models.py:128 +msgid "filter from" +msgstr "" + +#: paperless_mail/models.py:131 +msgid "filter subject" +msgstr "" + +#: paperless_mail/models.py:134 +msgid "filter body" +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter attachment filename" +msgstr "" + +#: paperless_mail/models.py:140 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "" + +#: paperless_mail/models.py:146 +msgid "maximum age" +msgstr "" + +#: paperless_mail/models.py:148 +msgid "Specified in days." +msgstr "" + +#: paperless_mail/models.py:151 +msgid "attachment type" +msgstr "" + +#: paperless_mail/models.py:154 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "" + +#: paperless_mail/models.py:159 +msgid "action" +msgstr "" + +#: paperless_mail/models.py:165 +msgid "action parameter" +msgstr "" + +#: paperless_mail/models.py:167 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action." +msgstr "" + +#: paperless_mail/models.py:173 +msgid "assign title from" +msgstr "" + +#: paperless_mail/models.py:183 +msgid "assign this tag" +msgstr "" + +#: paperless_mail/models.py:191 +msgid "assign this document type" +msgstr "" + +#: paperless_mail/models.py:195 +msgid "assign correspondent from" +msgstr "" + +#: paperless_mail/models.py:205 +msgid "assign this correspondent" +msgstr "" + diff --git a/src/locale/xh_ZA/LC_MESSAGES/django.po b/src/locale/xh_ZA/LC_MESSAGES/django.po new file mode 100644 index 000000000..fc6837620 --- /dev/null +++ b/src/locale/xh_ZA/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Xhosa\n" +"Language: xh_ZA\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: xh\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "crwdns2528:0crwdne2528:0" + +#: documents/models.py:32 +msgid "Any word" +msgstr "crwdns2530:0crwdne2530:0" + +#: documents/models.py:33 +msgid "All words" +msgstr "crwdns2532:0crwdne2532:0" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "crwdns2534:0crwdne2534:0" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "crwdns2536:0crwdne2536:0" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "crwdns2538:0crwdne2538:0" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "crwdns2540:0crwdne2540:0" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "crwdns2542:0crwdne2542:0" + +#: documents/models.py:45 +msgid "match" +msgstr "crwdns2544:0crwdne2544:0" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "crwdns2546:0crwdne2546:0" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "crwdns2548:0crwdne2548:0" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "crwdns2550:0crwdne2550:0" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "crwdns2552:0crwdne2552:0" + +#: documents/models.py:81 +msgid "color" +msgstr "crwdns2554:0crwdne2554:0" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "crwdns2556:0crwdne2556:0" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "crwdns2558:0crwdne2558:0" + +#: documents/models.py:94 +msgid "tag" +msgstr "crwdns2560:0crwdne2560:0" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "crwdns2562:0crwdne2562:0" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "crwdns2564:0crwdne2564:0" + +#: documents/models.py:102 +msgid "document types" +msgstr "crwdns2566:0crwdne2566:0" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "crwdns2568:0crwdne2568:0" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "crwdns2570:0crwdne2570:0" + +#: documents/models.py:124 +msgid "title" +msgstr "crwdns2572:0crwdne2572:0" + +#: documents/models.py:137 +msgid "content" +msgstr "crwdns2574:0crwdne2574:0" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "crwdns2576:0crwdne2576:0" + +#: documents/models.py:144 +msgid "mime type" +msgstr "crwdns2578:0crwdne2578:0" + +#: documents/models.py:155 +msgid "checksum" +msgstr "crwdns2580:0crwdne2580:0" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "crwdns2582:0crwdne2582:0" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "crwdns2584:0crwdne2584:0" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "crwdns2586:0crwdne2586:0" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "crwdns2588:0crwdne2588:0" + +#: documents/models.py:176 +msgid "modified" +msgstr "crwdns2590:0crwdne2590:0" + +#: documents/models.py:180 +msgid "storage type" +msgstr "crwdns2592:0crwdne2592:0" + +#: documents/models.py:188 +msgid "added" +msgstr "crwdns2594:0crwdne2594:0" + +#: documents/models.py:192 +msgid "filename" +msgstr "crwdns2596:0crwdne2596:0" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "crwdns2598:0crwdne2598:0" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "crwdns2600:0crwdne2600:0" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "crwdns2602:0crwdne2602:0" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "crwdns2604:0crwdne2604:0" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "crwdns2606:0crwdne2606:0" + +#: documents/models.py:223 +msgid "document" +msgstr "crwdns2608:0crwdne2608:0" + +#: documents/models.py:224 +msgid "documents" +msgstr "crwdns2610:0crwdne2610:0" + +#: documents/models.py:311 +msgid "debug" +msgstr "crwdns2612:0crwdne2612:0" + +#: documents/models.py:312 +msgid "information" +msgstr "crwdns2614:0crwdne2614:0" + +#: documents/models.py:313 +msgid "warning" +msgstr "crwdns2616:0crwdne2616:0" + +#: documents/models.py:314 +msgid "error" +msgstr "crwdns2618:0crwdne2618:0" + +#: documents/models.py:315 +msgid "critical" +msgstr "crwdns2620:0crwdne2620:0" + +#: documents/models.py:319 +msgid "group" +msgstr "crwdns2622:0crwdne2622:0" + +#: documents/models.py:322 +msgid "message" +msgstr "crwdns2624:0crwdne2624:0" + +#: documents/models.py:325 +msgid "level" +msgstr "crwdns2626:0crwdne2626:0" + +#: documents/models.py:332 +msgid "log" +msgstr "crwdns2628:0crwdne2628:0" + +#: documents/models.py:333 +msgid "logs" +msgstr "crwdns2630:0crwdne2630:0" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "crwdns2632:0crwdne2632:0" + +#: documents/models.py:345 +msgid "saved views" +msgstr "crwdns2634:0crwdne2634:0" + +#: documents/models.py:348 +msgid "user" +msgstr "crwdns2636:0crwdne2636:0" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "crwdns2638:0crwdne2638:0" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "crwdns2640:0crwdne2640:0" + +#: documents/models.py:361 +msgid "sort field" +msgstr "crwdns2642:0crwdne2642:0" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "crwdns2644:0crwdne2644:0" + +#: documents/models.py:373 +msgid "title contains" +msgstr "crwdns2646:0crwdne2646:0" + +#: documents/models.py:374 +msgid "content contains" +msgstr "crwdns2648:0crwdne2648:0" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "crwdns2650:0crwdne2650:0" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "crwdns2652:0crwdne2652:0" + +#: documents/models.py:377 +msgid "document type is" +msgstr "crwdns2654:0crwdne2654:0" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "crwdns2656:0crwdne2656:0" + +#: documents/models.py:379 +msgid "has tag" +msgstr "crwdns2658:0crwdne2658:0" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "crwdns2660:0crwdne2660:0" + +#: documents/models.py:381 +msgid "created before" +msgstr "crwdns2662:0crwdne2662:0" + +#: documents/models.py:382 +msgid "created after" +msgstr "crwdns2664:0crwdne2664:0" + +#: documents/models.py:383 +msgid "created year is" +msgstr "crwdns2666:0crwdne2666:0" + +#: documents/models.py:384 +msgid "created month is" +msgstr "crwdns2668:0crwdne2668:0" + +#: documents/models.py:385 +msgid "created day is" +msgstr "crwdns2670:0crwdne2670:0" + +#: documents/models.py:386 +msgid "added before" +msgstr "crwdns2672:0crwdne2672:0" + +#: documents/models.py:387 +msgid "added after" +msgstr "crwdns2674:0crwdne2674:0" + +#: documents/models.py:388 +msgid "modified before" +msgstr "crwdns2676:0crwdne2676:0" + +#: documents/models.py:389 +msgid "modified after" +msgstr "crwdns2678:0crwdne2678:0" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "crwdns2680:0crwdne2680:0" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "crwdns3408:0crwdne3408:0" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "crwdns3410:0crwdne3410:0" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "crwdns3438:0crwdne3438:0" + +#: documents/models.py:394 +msgid "more like this" +msgstr "crwdns3440:0crwdne3440:0" + +#: documents/models.py:405 +msgid "rule type" +msgstr "crwdns2682:0crwdne2682:0" + +#: documents/models.py:409 +msgid "value" +msgstr "crwdns2684:0crwdne2684:0" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "crwdns2686:0crwdne2686:0" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "crwdns2688:0crwdne2688:0" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "crwdns3412:0%(error)scrwdne3412:0" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "crwdns2692:0crwdne2692:0" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "crwdns2694:0%(type)scrwdne2694:0" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "crwdns2696:0crwdne2696:0" + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "crwdns2698:0crwdne2698:0" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "crwdns2700:0crwdne2700:0" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "crwdns2702:0crwdne2702:0" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "crwdns2704:0crwdne2704:0" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "crwdns2706:0crwdne2706:0" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "crwdns2708:0crwdne2708:0" + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "crwdns2710:0crwdne2710:0" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "crwdns2712:0crwdne2712:0" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "crwdns2714:0crwdne2714:0" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "crwdns2716:0crwdne2716:0" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "crwdns2718:0crwdne2718:0" + +#: paperless/settings.py:305 +msgid "German" +msgstr "crwdns2720:0crwdne2720:0" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "crwdns2722:0crwdne2722:0" + +#: paperless/settings.py:307 +msgid "French" +msgstr "crwdns2724:0crwdne2724:0" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "crwdns2726:0crwdne2726:0" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "crwdns3424:0crwdne3424:0" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "crwdns2728:0crwdne2728:0" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "crwdns2730:0crwdne2730:0" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "crwdns3414:0crwdne3414:0" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "crwdns3420:0crwdne3420:0" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "crwdns3444:0crwdne3444:0" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "crwdns3448:0crwdne3448:0" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "crwdns2732:0crwdne2732:0" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "crwdns3456:0crwdne3456:0" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "crwdns3458:0crwdne3458:0" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "crwdns2734:0crwdne2734:0" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "crwdns2736:0crwdne2736:0" + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "crwdns2738:0crwdne2738:0" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "crwdns2740:0crwdne2740:0" + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "crwdns2742:0crwdne2742:0" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "crwdns2744:0crwdne2744:0" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "crwdns2746:0crwdne2746:0" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "crwdns2748:0crwdne2748:0" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "crwdns2750:0crwdne2750:0" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "crwdns2752:0crwdne2752:0" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "crwdns2754:0crwdne2754:0" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "crwdns2756:0crwdne2756:0" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "crwdns2758:0crwdne2758:0" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "crwdns2760:0crwdne2760:0" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "crwdns2762:0crwdne2762:0" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "crwdns2764:0crwdne2764:0" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "crwdns2766:0crwdne2766:0" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "crwdns2768:0crwdne2768:0" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "crwdns3460:0crwdne3460:0" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "crwdns3462:0crwdne3462:0" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "crwdns2770:0crwdne2770:0" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "crwdns2772:0crwdne2772:0" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "crwdns2774:0crwdne2774:0" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "crwdns2776:0crwdne2776:0" + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "crwdns2778:0crwdne2778:0" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "crwdns2780:0crwdne2780:0" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "crwdns2782:0crwdne2782:0" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "crwdns2784:0crwdne2784:0" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "crwdns2786:0crwdne2786:0" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "crwdns2788:0crwdne2788:0" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "crwdns2790:0crwdne2790:0" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "crwdns2792:0crwdne2792:0" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "crwdns2794:0crwdne2794:0" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "crwdns2796:0crwdne2796:0" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "crwdns2798:0crwdne2798:0" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "crwdns2800:0crwdne2800:0" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "crwdns2802:0crwdne2802:0" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "crwdns3464:0crwdne3464:0" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "crwdns2804:0crwdne2804:0" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "crwdns2806:0crwdne2806:0" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "crwdns2808:0crwdne2808:0" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "crwdns2810:0crwdne2810:0" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "crwdns2812:0crwdne2812:0" + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "crwdns2814:0crwdne2814:0" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "crwdns2816:0crwdne2816:0" + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "crwdns2818:0crwdne2818:0" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "crwdns2820:0crwdne2820:0" + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "crwdns2822:0crwdne2822:0" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "crwdns2824:0crwdne2824:0" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "crwdns3466:0crwdne3466:0" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "crwdns2828:0crwdne2828:0" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "crwdns2830:0crwdne2830:0" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "crwdns2832:0crwdne2832:0" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "crwdns2834:0crwdne2834:0" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "crwdns2836:0crwdne2836:0" + diff --git a/src/locale/zh_CN/LC_MESSAGES/django.po b/src/locale/zh_CN/LC_MESSAGES/django.po new file mode 100644 index 000000000..ea3b5b03d --- /dev/null +++ b/src/locale/zh_CN/LC_MESSAGES/django.po @@ -0,0 +1,698 @@ +msgid "" +msgstr "" +"Project-Id-Version: paperless-ng\n" +"Report-Msgid-Bugs-To: \n" +"POT-Creation-Date: 2021-05-16 09:38+0000\n" +"PO-Revision-Date: 2021-05-16 10:09\n" +"Last-Translator: \n" +"Language-Team: Chinese Simplified\n" +"Language: zh_CN\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Plural-Forms: nplurals=1; plural=0;\n" +"X-Crowdin-Project: paperless-ng\n" +"X-Crowdin-Project-ID: 434940\n" +"X-Crowdin-Language: zh-CN\n" +"X-Crowdin-File: /dev/src/locale/en_US/LC_MESSAGES/django.po\n" +"X-Crowdin-File-ID: 54\n" + +#: documents/apps.py:10 +msgid "Documents" +msgstr "文件" + +#: documents/models.py:32 +msgid "Any word" +msgstr "任何词" + +#: documents/models.py:33 +msgid "All words" +msgstr "所有词" + +#: documents/models.py:34 +msgid "Exact match" +msgstr "完全符合" + +#: documents/models.py:35 +msgid "Regular expression" +msgstr "正则表达式" + +#: documents/models.py:36 +msgid "Fuzzy word" +msgstr "模糊词汇" + +#: documents/models.py:37 +msgid "Automatic" +msgstr "自动" + +#: documents/models.py:41 documents/models.py:350 paperless_mail/models.py:25 +#: paperless_mail/models.py:117 +msgid "name" +msgstr "名字" + +#: documents/models.py:45 +msgid "match" +msgstr "配对" + +#: documents/models.py:49 +msgid "matching algorithm" +msgstr "配对算法" + +#: documents/models.py:55 +msgid "is insensitive" +msgstr "" + +#: documents/models.py:74 documents/models.py:120 +msgid "correspondent" +msgstr "" + +#: documents/models.py:75 +msgid "correspondents" +msgstr "" + +#: documents/models.py:81 +msgid "color" +msgstr "" + +#: documents/models.py:87 +msgid "is inbox tag" +msgstr "" + +#: documents/models.py:89 +msgid "Marks this tag as an inbox tag: All newly consumed documents will be tagged with inbox tags." +msgstr "" + +#: documents/models.py:94 +msgid "tag" +msgstr "" + +#: documents/models.py:95 documents/models.py:151 +msgid "tags" +msgstr "" + +#: documents/models.py:101 documents/models.py:133 +msgid "document type" +msgstr "" + +#: documents/models.py:102 +msgid "document types" +msgstr "" + +#: documents/models.py:110 +msgid "Unencrypted" +msgstr "" + +#: documents/models.py:111 +msgid "Encrypted with GNU Privacy Guard" +msgstr "" + +#: documents/models.py:124 +msgid "title" +msgstr "" + +#: documents/models.py:137 +msgid "content" +msgstr "" + +#: documents/models.py:139 +msgid "The raw, text-only data of the document. This field is primarily used for searching." +msgstr "" + +#: documents/models.py:144 +msgid "mime type" +msgstr "" + +#: documents/models.py:155 +msgid "checksum" +msgstr "" + +#: documents/models.py:159 +msgid "The checksum of the original document." +msgstr "" + +#: documents/models.py:163 +msgid "archive checksum" +msgstr "" + +#: documents/models.py:168 +msgid "The checksum of the archived document." +msgstr "" + +#: documents/models.py:172 documents/models.py:328 +msgid "created" +msgstr "" + +#: documents/models.py:176 +msgid "modified" +msgstr "" + +#: documents/models.py:180 +msgid "storage type" +msgstr "" + +#: documents/models.py:188 +msgid "added" +msgstr "" + +#: documents/models.py:192 +msgid "filename" +msgstr "" + +#: documents/models.py:198 +msgid "Current filename in storage" +msgstr "" + +#: documents/models.py:202 +msgid "archive filename" +msgstr "" + +#: documents/models.py:208 +msgid "Current archive filename in storage" +msgstr "" + +#: documents/models.py:212 +msgid "archive serial number" +msgstr "" + +#: documents/models.py:217 +msgid "The position of this document in your physical document archive." +msgstr "" + +#: documents/models.py:223 +msgid "document" +msgstr "" + +#: documents/models.py:224 +msgid "documents" +msgstr "" + +#: documents/models.py:311 +msgid "debug" +msgstr "" + +#: documents/models.py:312 +msgid "information" +msgstr "" + +#: documents/models.py:313 +msgid "warning" +msgstr "" + +#: documents/models.py:314 +msgid "error" +msgstr "" + +#: documents/models.py:315 +msgid "critical" +msgstr "" + +#: documents/models.py:319 +msgid "group" +msgstr "" + +#: documents/models.py:322 +msgid "message" +msgstr "" + +#: documents/models.py:325 +msgid "level" +msgstr "" + +#: documents/models.py:332 +msgid "log" +msgstr "" + +#: documents/models.py:333 +msgid "logs" +msgstr "" + +#: documents/models.py:344 documents/models.py:401 +msgid "saved view" +msgstr "" + +#: documents/models.py:345 +msgid "saved views" +msgstr "" + +#: documents/models.py:348 +msgid "user" +msgstr "" + +#: documents/models.py:354 +msgid "show on dashboard" +msgstr "" + +#: documents/models.py:357 +msgid "show in sidebar" +msgstr "" + +#: documents/models.py:361 +msgid "sort field" +msgstr "" + +#: documents/models.py:367 +msgid "sort reverse" +msgstr "" + +#: documents/models.py:373 +msgid "title contains" +msgstr "" + +#: documents/models.py:374 +msgid "content contains" +msgstr "" + +#: documents/models.py:375 +msgid "ASN is" +msgstr "" + +#: documents/models.py:376 +msgid "correspondent is" +msgstr "" + +#: documents/models.py:377 +msgid "document type is" +msgstr "" + +#: documents/models.py:378 +msgid "is in inbox" +msgstr "" + +#: documents/models.py:379 +msgid "has tag" +msgstr "" + +#: documents/models.py:380 +msgid "has any tag" +msgstr "" + +#: documents/models.py:381 +msgid "created before" +msgstr "" + +#: documents/models.py:382 +msgid "created after" +msgstr "" + +#: documents/models.py:383 +msgid "created year is" +msgstr "" + +#: documents/models.py:384 +msgid "created month is" +msgstr "" + +#: documents/models.py:385 +msgid "created day is" +msgstr "" + +#: documents/models.py:386 +msgid "added before" +msgstr "" + +#: documents/models.py:387 +msgid "added after" +msgstr "" + +#: documents/models.py:388 +msgid "modified before" +msgstr "" + +#: documents/models.py:389 +msgid "modified after" +msgstr "" + +#: documents/models.py:390 +msgid "does not have tag" +msgstr "" + +#: documents/models.py:391 +msgid "does not have ASN" +msgstr "" + +#: documents/models.py:392 +msgid "title or content contains" +msgstr "" + +#: documents/models.py:393 +msgid "fulltext query" +msgstr "" + +#: documents/models.py:394 +msgid "more like this" +msgstr "" + +#: documents/models.py:405 +msgid "rule type" +msgstr "" + +#: documents/models.py:409 +msgid "value" +msgstr "" + +#: documents/models.py:415 +msgid "filter rule" +msgstr "" + +#: documents/models.py:416 +msgid "filter rules" +msgstr "" + +#: documents/serialisers.py:53 +#, python-format +msgid "Invalid regular expression: %(error)s" +msgstr "" + +#: documents/serialisers.py:177 +msgid "Invalid color." +msgstr "" + +#: documents/serialisers.py:451 +#, python-format +msgid "File type %(type)s not supported" +msgstr "" + +#: documents/templates/index.html:22 +msgid "Paperless-ng is loading..." +msgstr "" + +#: documents/templates/registration/logged_out.html:14 +msgid "Paperless-ng signed out" +msgstr "" + +#: documents/templates/registration/logged_out.html:45 +msgid "You have been successfully logged out. Bye!" +msgstr "" + +#: documents/templates/registration/logged_out.html:46 +msgid "Sign in again" +msgstr "" + +#: documents/templates/registration/login.html:15 +msgid "Paperless-ng sign in" +msgstr "" + +#: documents/templates/registration/login.html:47 +msgid "Please sign in." +msgstr "" + +#: documents/templates/registration/login.html:50 +msgid "Your username and password didn't match. Please try again." +msgstr "" + +#: documents/templates/registration/login.html:53 +msgid "Username" +msgstr "" + +#: documents/templates/registration/login.html:54 +msgid "Password" +msgstr "" + +#: documents/templates/registration/login.html:59 +msgid "Sign in" +msgstr "" + +#: paperless/settings.py:303 +msgid "English (US)" +msgstr "" + +#: paperless/settings.py:304 +msgid "English (GB)" +msgstr "" + +#: paperless/settings.py:305 +msgid "German" +msgstr "" + +#: paperless/settings.py:306 +msgid "Dutch" +msgstr "" + +#: paperless/settings.py:307 +msgid "French" +msgstr "" + +#: paperless/settings.py:308 +msgid "Portuguese (Brazil)" +msgstr "" + +#: paperless/settings.py:309 +msgid "Portuguese" +msgstr "" + +#: paperless/settings.py:310 +msgid "Italian" +msgstr "" + +#: paperless/settings.py:311 +msgid "Romanian" +msgstr "" + +#: paperless/settings.py:312 +msgid "Russian" +msgstr "" + +#: paperless/settings.py:313 +msgid "Spanish" +msgstr "" + +#: paperless/settings.py:314 +msgid "Polish" +msgstr "" + +#: paperless/settings.py:315 +msgid "Swedish" +msgstr "" + +#: paperless/urls.py:120 +msgid "Paperless-ng administration" +msgstr "" + +#: paperless_mail/admin.py:15 +msgid "Authentication" +msgstr "" + +#: paperless_mail/admin.py:18 +msgid "Advanced settings" +msgstr "" + +#: paperless_mail/admin.py:37 +msgid "Filter" +msgstr "" + +#: paperless_mail/admin.py:39 +msgid "Paperless will only process mails that match ALL of the filters given below." +msgstr "" + +#: paperless_mail/admin.py:49 +msgid "Actions" +msgstr "" + +#: paperless_mail/admin.py:51 +msgid "The action applied to the mail. This action is only performed when documents were consumed from the mail. Mails without attachments will remain entirely untouched." +msgstr "" + +#: paperless_mail/admin.py:58 +msgid "Metadata" +msgstr "" + +#: paperless_mail/admin.py:60 +msgid "Assign metadata to documents consumed from this rule automatically. If you do not assign tags, types or correspondents here, paperless will still process all matching rules that you have defined." +msgstr "" + +#: paperless_mail/apps.py:9 +msgid "Paperless mail" +msgstr "" + +#: paperless_mail/models.py:11 +msgid "mail account" +msgstr "" + +#: paperless_mail/models.py:12 +msgid "mail accounts" +msgstr "" + +#: paperless_mail/models.py:19 +msgid "No encryption" +msgstr "" + +#: paperless_mail/models.py:20 +msgid "Use SSL" +msgstr "" + +#: paperless_mail/models.py:21 +msgid "Use STARTTLS" +msgstr "" + +#: paperless_mail/models.py:29 +msgid "IMAP server" +msgstr "" + +#: paperless_mail/models.py:33 +msgid "IMAP port" +msgstr "" + +#: paperless_mail/models.py:36 +msgid "This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections." +msgstr "" + +#: paperless_mail/models.py:40 +msgid "IMAP security" +msgstr "" + +#: paperless_mail/models.py:46 +msgid "username" +msgstr "" + +#: paperless_mail/models.py:50 +msgid "password" +msgstr "" + +#: paperless_mail/models.py:54 +msgid "character set" +msgstr "" + +#: paperless_mail/models.py:57 +msgid "The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'." +msgstr "" + +#: paperless_mail/models.py:68 +msgid "mail rule" +msgstr "" + +#: paperless_mail/models.py:69 +msgid "mail rules" +msgstr "" + +#: paperless_mail/models.py:75 +msgid "Only process attachments." +msgstr "" + +#: paperless_mail/models.py:76 +msgid "Process all files, including 'inline' attachments." +msgstr "" + +#: paperless_mail/models.py:86 +msgid "Mark as read, don't process read mails" +msgstr "" + +#: paperless_mail/models.py:87 +msgid "Flag the mail, don't process flagged mails" +msgstr "" + +#: paperless_mail/models.py:88 +msgid "Move to specified folder" +msgstr "" + +#: paperless_mail/models.py:89 +msgid "Delete" +msgstr "" + +#: paperless_mail/models.py:96 +msgid "Use subject as title" +msgstr "" + +#: paperless_mail/models.py:97 +msgid "Use attachment filename as title" +msgstr "" + +#: paperless_mail/models.py:107 +msgid "Do not assign a correspondent" +msgstr "" + +#: paperless_mail/models.py:109 +msgid "Use mail address" +msgstr "" + +#: paperless_mail/models.py:111 +msgid "Use name (or mail address if not available)" +msgstr "" + +#: paperless_mail/models.py:113 +msgid "Use correspondent selected below" +msgstr "" + +#: paperless_mail/models.py:121 +msgid "order" +msgstr "" + +#: paperless_mail/models.py:128 +msgid "account" +msgstr "" + +#: paperless_mail/models.py:132 +msgid "folder" +msgstr "" + +#: paperless_mail/models.py:134 +msgid "Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:138 +msgid "filter from" +msgstr "" + +#: paperless_mail/models.py:141 +msgid "filter subject" +msgstr "" + +#: paperless_mail/models.py:144 +msgid "filter body" +msgstr "" + +#: paperless_mail/models.py:148 +msgid "filter attachment filename" +msgstr "" + +#: paperless_mail/models.py:150 +msgid "Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." +msgstr "" + +#: paperless_mail/models.py:156 +msgid "maximum age" +msgstr "" + +#: paperless_mail/models.py:158 +msgid "Specified in days." +msgstr "" + +#: paperless_mail/models.py:161 +msgid "attachment type" +msgstr "" + +#: paperless_mail/models.py:164 +msgid "Inline attachments include embedded images, so it's best to combine this option with a filename filter." +msgstr "" + +#: paperless_mail/models.py:169 +msgid "action" +msgstr "" + +#: paperless_mail/models.py:175 +msgid "action parameter" +msgstr "" + +#: paperless_mail/models.py:177 +msgid "Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots." +msgstr "" + +#: paperless_mail/models.py:184 +msgid "assign title from" +msgstr "" + +#: paperless_mail/models.py:194 +msgid "assign this tag" +msgstr "" + +#: paperless_mail/models.py:202 +msgid "assign this document type" +msgstr "" + +#: paperless_mail/models.py:206 +msgid "assign correspondent from" +msgstr "" + +#: paperless_mail/models.py:216 +msgid "assign this correspondent" +msgstr "" + diff --git a/src/paperless/asgi.py b/src/paperless/asgi.py new file mode 100644 index 000000000..2f6cc2d5f --- /dev/null +++ b/src/paperless/asgi.py @@ -0,0 +1,23 @@ +import os + +from django.core.asgi import get_asgi_application +# Fetch Django ASGI application early to ensure AppRegistry is populated +# before importing consumers and AuthMiddlewareStack that may import ORM +# models. + +os.environ.setdefault("DJANGO_SETTINGS_MODULE", "paperless.settings") +django_asgi_app = get_asgi_application() + +from channels.auth import AuthMiddlewareStack # NOQA: E402 +from channels.routing import ProtocolTypeRouter, URLRouter # NOQA: E402 + +from paperless.urls import websocket_urlpatterns # NOQA: E402 + +application = ProtocolTypeRouter({ + "http": get_asgi_application(), + "websocket": AuthMiddlewareStack( + URLRouter( + websocket_urlpatterns + ) + ), +}) diff --git a/src/paperless/auth.py b/src/paperless/auth.py index ece5d0eba..79deb5c37 100644 --- a/src/paperless/auth.py +++ b/src/paperless/auth.py @@ -1,7 +1,9 @@ from django.conf import settings +from django.contrib import auth from django.contrib.auth.models import User from django.utils.deprecation import MiddlewareMixin from rest_framework import authentication +from django.contrib.auth.middleware import RemoteUserMiddleware class AutoLoginMiddleware(MiddlewareMixin): @@ -10,6 +12,7 @@ class AutoLoginMiddleware(MiddlewareMixin): try: request.user = User.objects.get( username=settings.AUTO_LOGIN_USERNAME) + auth.login(request, request.user) except User.DoesNotExist: pass @@ -26,3 +29,10 @@ class AngularApiAuthenticationOverride(authentication.BaseAuthentication): return (user, None) else: return None + + +class HttpRemoteUserMiddleware(RemoteUserMiddleware): + """ This class allows authentication via HTTP_REMOTE_USER which is set for + example by certain SSO applications. + """ + header = settings.HTTP_REMOTE_USER_HEADER_NAME diff --git a/src/paperless/checks.py b/src/paperless/checks.py index 819582ffc..1b19f1a72 100644 --- a/src/paperless/checks.py +++ b/src/paperless/checks.py @@ -1,5 +1,6 @@ import os import shutil +import stat from django.conf import settings from django.core.checks import Error, Warning, register @@ -13,20 +14,32 @@ writeable_hint = ( ) -def path_check(env_var): +def path_check(var, directory): messages = [] - directory = os.getenv(env_var) if directory: - if not os.path.exists(directory): + if not os.path.isdir(directory): messages.append(Error( - exists_message.format(env_var), + exists_message.format(var), exists_hint.format(directory) )) - elif not os.access(directory, os.W_OK | os.X_OK): - messages.append(Error( - writeable_message.format(env_var), - writeable_hint.format(directory) - )) + else: + test_file = os.path.join( + directory, f'__paperless_write_test_{os.getpid()}__' + ) + try: + with open(test_file, 'w'): + pass + except PermissionError: + messages.append(Error( + writeable_message.format(var), + writeable_hint.format( + f'\n{stat.filemode(os.stat(directory).st_mode)} ' + f'{directory}\n') + )) + finally: + if os.path.isfile(test_file): + os.remove(test_file) + return messages @@ -36,12 +49,9 @@ def paths_check(app_configs, **kwargs): Check the various paths for existence, readability and writeability """ - check_messages = path_check("PAPERLESS_DATA_DIR") + \ - path_check("PAPERLESS_MEDIA_ROOT") + \ - path_check("PAPERLESS_CONSUMPTION_DIR") + \ - path_check("PAPERLESS_STATICDIR") - - return check_messages + return path_check("PAPERLESS_DATA_DIR", settings.DATA_DIR) + \ + path_check("PAPERLESS_MEDIA_ROOT", settings.MEDIA_ROOT) + \ + path_check("PAPERLESS_CONSUMPTION_DIR", settings.CONSUMPTION_DIR) @register() diff --git a/src/paperless/consumers.py b/src/paperless/consumers.py new file mode 100644 index 000000000..45f6ad9c5 --- /dev/null +++ b/src/paperless/consumers.py @@ -0,0 +1,29 @@ +import json + +from asgiref.sync import async_to_sync +from channels.exceptions import DenyConnection, AcceptConnection +from channels.generic.websocket import WebsocketConsumer + + +class StatusConsumer(WebsocketConsumer): + + def _authenticated(self): + return 'user' in self.scope and self.scope['user'].is_authenticated + + def connect(self): + if not self._authenticated(): + raise DenyConnection() + else: + async_to_sync(self.channel_layer.group_add)( + 'status_updates', self.channel_name) + raise AcceptConnection() + + def disconnect(self, close_code): + async_to_sync(self.channel_layer.group_discard)( + 'status_updates', self.channel_name) + + def status_update(self, event): + if not self._authenticated(): + self.close() + else: + self.send(json.dumps(event['data'])) diff --git a/src/paperless/middleware.py b/src/paperless/middleware.py new file mode 100644 index 000000000..2e9d2793c --- /dev/null +++ b/src/paperless/middleware.py @@ -0,0 +1,20 @@ +from django.conf import settings + +from paperless import version + + +class ApiVersionMiddleware: + + def __init__(self, get_response): + self.get_response = get_response + + def __call__(self, request): + response = self.get_response(request) + if request.user.is_authenticated: + versions = settings.REST_FRAMEWORK['ALLOWED_VERSIONS'] + response['X-Api-Version'] = versions[len(versions)-1] + response['X-Version'] = ".".join( + [str(_) for _ in version.__version__] + ) + + return response diff --git a/src/paperless/settings.py b/src/paperless/settings.py index 1a6b80a0c..805b1aec0 100644 --- a/src/paperless/settings.py +++ b/src/paperless/settings.py @@ -4,8 +4,11 @@ import multiprocessing import os import re +from concurrent_log_handler.queue import setup_logging_queues from dotenv import load_dotenv +from django.utils.translation import gettext_lazy as _ + # Tap paperless.conf if it's available if os.path.exists("../paperless.conf"): load_dotenv("../paperless.conf") @@ -60,6 +63,8 @@ MEDIA_LOCK = os.path.join(MEDIA_ROOT, "media.lock") INDEX_DIR = os.path.join(DATA_DIR, "index") MODEL_FILE = os.path.join(DATA_DIR, "classification_model.pickle") +LOGGING_DIR = os.getenv('PAPERLESS_LOGGING_DIR', os.path.join(DATA_DIR, "log")) + CONSUMPTION_DIR = os.getenv("PAPERLESS_CONSUMPTION_DIR", os.path.join(BASE_DIR, "..", "consume")) # This will be created if it doesn't exist @@ -69,6 +74,8 @@ SCRATCH_DIR = os.getenv("PAPERLESS_SCRATCH_DIR", "/tmp/paperless") # Application Definition # ############################################################################### +env_apps = os.getenv("PAPERLESS_APPS").split(",") if os.getenv("PAPERLESS_APPS") else [] + INSTALLED_APPS = [ "whitenoise.runserver_nostatic", @@ -95,14 +102,22 @@ INSTALLED_APPS = [ "django_q", -] +] + env_apps + +if DEBUG: + INSTALLED_APPS.append("channels") REST_FRAMEWORK = { 'DEFAULT_AUTHENTICATION_CLASSES': [ 'rest_framework.authentication.BasicAuthentication', 'rest_framework.authentication.SessionAuthentication', 'rest_framework.authentication.TokenAuthentication' - ] + ], + 'DEFAULT_VERSIONING_CLASS': 'rest_framework.versioning.AcceptHeaderVersioning', + 'DEFAULT_VERSION': '1', + # Make sure these are ordered and that the most recent version appears + # last + 'ALLOWED_VERSIONS': ['1', '2'] } if DEBUG: @@ -115,8 +130,10 @@ MIDDLEWARE = [ 'whitenoise.middleware.WhiteNoiseMiddleware', 'django.contrib.sessions.middleware.SessionMiddleware', 'corsheaders.middleware.CorsMiddleware', + 'django.middleware.locale.LocaleMiddleware', 'django.middleware.common.CommonMiddleware', 'django.middleware.csrf.CsrfViewMiddleware', + 'paperless.middleware.ApiVersionMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', 'django.middleware.clickjacking.XFrameOptionsMiddleware', @@ -125,12 +142,17 @@ MIDDLEWARE = [ ROOT_URLCONF = 'paperless.urls' FORCE_SCRIPT_NAME = os.getenv("PAPERLESS_FORCE_SCRIPT_NAME") +BASE_URL = (FORCE_SCRIPT_NAME or "") + "/" +LOGIN_URL = BASE_URL + "accounts/login/" +LOGOUT_REDIRECT_URL = os.getenv("PAPERLESS_LOGOUT_REDIRECT_URL") WSGI_APPLICATION = 'paperless.wsgi.application' +ASGI_APPLICATION = "paperless.asgi.application" -STATIC_URL = os.getenv("PAPERLESS_STATIC_URL", "/static/") +STATIC_URL = os.getenv("PAPERLESS_STATIC_URL", BASE_URL + "static/") +WHITENOISE_STATIC_PREFIX = "/static/" -# what is this used for? +# TODO: what is this used for? TEMPLATES = [ { 'BACKEND': 'django.template.backends.django.DjangoTemplates', @@ -147,6 +169,17 @@ TEMPLATES = [ }, ] +CHANNEL_LAYERS = { + "default": { + "BACKEND": "channels_redis.core.RedisChannelLayer", + "CONFIG": { + "hosts": [os.getenv("PAPERLESS_REDIS", "redis://localhost:6379")], + "capacity": 2000, # default 100 + "expiry": 15, # default 60 + }, + }, +} + ############################################################################### # Security # ############################################################################### @@ -159,11 +192,24 @@ if AUTO_LOGIN_USERNAME: # regular login in case the provided user does not exist. MIDDLEWARE.insert(_index+1, 'paperless.auth.AutoLoginMiddleware') +ENABLE_HTTP_REMOTE_USER = __get_boolean("PAPERLESS_ENABLE_HTTP_REMOTE_USER") +HTTP_REMOTE_USER_HEADER_NAME = os.getenv("PAPERLESS_HTTP_REMOTE_USER_HEADER_NAME", "HTTP_REMOTE_USER") +if ENABLE_HTTP_REMOTE_USER: + MIDDLEWARE.append( + 'paperless.auth.HttpRemoteUserMiddleware' + ) + AUTHENTICATION_BACKENDS = [ + 'django.contrib.auth.backends.RemoteUserBackend', + 'django.contrib.auth.backends.ModelBackend' + ] + REST_FRAMEWORK['DEFAULT_AUTHENTICATION_CLASSES'].append( + 'rest_framework.authentication.RemoteUserAuthentication' + ) + +# X-Frame options for embedded PDF display: if DEBUG: - X_FRAME_OPTIONS = '' - # this should really be 'allow-from uri' but its not supported in any mayor - # browser. + X_FRAME_OPTIONS = 'ANY' else: X_FRAME_OPTIONS = 'SAMEORIGIN' @@ -241,16 +287,40 @@ if os.getenv("PAPERLESS_DBHOST"): "NAME": os.getenv("PAPERLESS_DBNAME", "paperless"), "USER": os.getenv("PAPERLESS_DBUSER", "paperless"), "PASSWORD": os.getenv("PAPERLESS_DBPASS", "paperless"), + 'OPTIONS': {'sslmode': os.getenv("PAPERLESS_DBSSLMODE", "prefer")}, } if os.getenv("PAPERLESS_DBPORT"): DATABASES["default"]["PORT"] = os.getenv("PAPERLESS_DBPORT") +DEFAULT_AUTO_FIELD = 'django.db.models.AutoField' + ############################################################################### # Internationalization # ############################################################################### LANGUAGE_CODE = 'en-us' +LANGUAGES = [ + ("en-us", _("English (US)")), + ("en-gb", _("English (GB)")), + ("de-de", _("German")), + ("nl-nl", _("Dutch")), + ("fr-fr", _("French")), + ("pt-br", _("Portuguese (Brazil)")), + ("pt-pt", _("Portuguese")), + ("it-it", _("Italian")), + ("ro-ro", _("Romanian")), + ("ru-ru", _("Russian")), + ("es-es", _("Spanish")), + ("pl-pl", _("Polish")), + ("sv-se", _("Swedish")), + ("lb-lu", _("Luxembourgish")), +] + +LOCALE_PATHS = [ + os.path.join(BASE_DIR, "locale") +] + TIME_ZONE = os.getenv("PAPERLESS_TIME_ZONE", "UTC") USE_I18N = True @@ -263,14 +333,19 @@ USE_TZ = True # Logging # ############################################################################### -DISABLE_DBHANDLER = __get_boolean("PAPERLESS_DISABLE_DBHANDLER") +setup_logging_queues() + +os.makedirs(LOGGING_DIR, exist_ok=True) + +LOGROTATE_MAX_SIZE = os.getenv("PAPERLESS_LOGROTATE_MAX_SIZE", 1024*1024) +LOGROTATE_MAX_BACKUPS = os.getenv("PAPERLESS_LOGROTATE_MAX_BACKUPS", 20) LOGGING = { "version": 1, "disable_existing_loggers": False, 'formatters': { 'verbose': { - 'format': '{levelname} {asctime} {module} {message}', + 'format': '[{asctime}] [{levelname}] [{name}] {message}', 'style': '{', }, 'simple': { @@ -279,34 +354,39 @@ LOGGING = { }, }, "handlers": { - "db": { - "level": "DEBUG", - "class": "documents.loggers.PaperlessHandler", - }, "console": { - "level": "INFO", + "level": "DEBUG" if DEBUG else "INFO", "class": "logging.StreamHandler", "formatter": "verbose", + }, + "file_paperless": { + "class": "concurrent_log_handler.ConcurrentRotatingFileHandler", + "formatter": "verbose", + "filename": os.path.join(LOGGING_DIR, "paperless.log"), + "maxBytes": LOGROTATE_MAX_SIZE, + "backupCount": LOGROTATE_MAX_BACKUPS + }, + "file_mail": { + "class": "concurrent_log_handler.ConcurrentRotatingFileHandler", + "formatter": "verbose", + "filename": os.path.join(LOGGING_DIR, "mail.log"), + "maxBytes": LOGROTATE_MAX_SIZE, + "backupCount": LOGROTATE_MAX_BACKUPS } }, "root": { - "handlers": ["console"], - "level": "DEBUG", + "handlers": ["console"] }, "loggers": { - "documents": { - "handlers": ["db"], - "propagate": True, + "paperless": { + "handlers": ["file_paperless"], + "level": "DEBUG" }, "paperless_mail": { - "handlers": ["db"], - "propagate": True, - }, - "paperless_tesseract": { - "handlers": ["db"], - "propagate": True, - }, - }, + "handlers": ["file_mail"], + "level": "DEBUG" + } + } } ############################################################################### @@ -321,10 +401,15 @@ LOGGING = { # Favors threads per worker on smaller systems and never exceeds cpu_count() # in total. + def default_task_workers(): + # always leave one core open + available_cores = max(multiprocessing.cpu_count(), 1) try: + if available_cores < 4: + return available_cores return max( - math.floor(math.sqrt(multiprocessing.cpu_count())), + math.floor(math.sqrt(available_cores)), 1 ) except NotImplementedError: @@ -336,22 +421,27 @@ TASK_WORKERS = int(os.getenv("PAPERLESS_TASK_WORKERS", default_task_workers())) Q_CLUSTER = { 'name': 'paperless', 'catch_up': False, + 'recycle': 1, + 'retry': 1800, + 'timeout': 1800, 'workers': TASK_WORKERS, 'redis': os.getenv("PAPERLESS_REDIS", "redis://localhost:6379") } -def default_threads_per_worker(): +def default_threads_per_worker(task_workers): + # always leave one core open + available_cores = max(multiprocessing.cpu_count(), 1) try: return max( - math.floor(multiprocessing.cpu_count() / TASK_WORKERS), + math.floor(available_cores / task_workers), 1 ) except NotImplementedError: return 1 -THREADS_PER_WORKER = os.getenv("PAPERLESS_THREADS_PER_WORKER", default_threads_per_worker()) +THREADS_PER_WORKER = os.getenv("PAPERLESS_THREADS_PER_WORKER", default_threads_per_worker(TASK_WORKERS)) ############################################################################### # Paperless Specific Settings # @@ -359,10 +449,22 @@ THREADS_PER_WORKER = os.getenv("PAPERLESS_THREADS_PER_WORKER", default_threads_p CONSUMER_POLLING = int(os.getenv("PAPERLESS_CONSUMER_POLLING", 0)) +CONSUMER_POLLING_DELAY = int(os.getenv("PAPERLESS_CONSUMER_POLLING_DELAY", 5)) + +CONSUMER_POLLING_RETRY_COUNT = int( + os.getenv("PAPERLESS_CONSUMER_POLLING_RETRY_COUNT", 5) +) + CONSUMER_DELETE_DUPLICATES = __get_boolean("PAPERLESS_CONSUMER_DELETE_DUPLICATES") CONSUMER_RECURSIVE = __get_boolean("PAPERLESS_CONSUMER_RECURSIVE") +# Ignore glob patterns, relative to PAPERLESS_CONSUMPTION_DIR +CONSUMER_IGNORE_PATTERNS = list( + json.loads( + os.getenv("PAPERLESS_CONSUMER_IGNORE_PATTERNS", + '[".DS_STORE/*", "._*", ".stfolder/*"]'))) + CONSUMER_SUBDIRS_AS_TAGS = __get_boolean("PAPERLESS_CONSUMER_SUBDIRS_AS_TAGS") OPTIMIZE_THUMBNAILS = __get_boolean("PAPERLESS_OPTIMIZE_THUMBNAILS", "true") @@ -383,6 +485,14 @@ OCR_MODE = os.getenv("PAPERLESS_OCR_MODE", "skip") OCR_IMAGE_DPI = os.getenv("PAPERLESS_OCR_IMAGE_DPI") +OCR_CLEAN = os.getenv("PAPERLESS_OCR_CLEAN", "clean") + +OCR_DESKEW = __get_boolean("PAPERLESS_OCR_DESKEW", "true") + +OCR_ROTATE_PAGES = __get_boolean("PAPERLESS_OCR_ROTATE_PAGES", "true") + +OCR_ROTATE_PAGES_THRESHOLD = float(os.getenv("PAPERLESS_OCR_ROTATE_PAGES_THRESHOLD", 12.0)) + OCR_USER_ARGS = os.getenv("PAPERLESS_OCR_USER_ARGS", "{}") # GNUPG needs a home directory for some reason @@ -427,3 +537,26 @@ for t in json.loads(os.getenv("PAPERLESS_FILENAME_PARSE_TRANSFORMS", "[]")): # TODO: this should not have a prefix. # Specify the filename format for out files PAPERLESS_FILENAME_FORMAT = os.getenv("PAPERLESS_FILENAME_FORMAT") + +THUMBNAIL_FONT_NAME = os.getenv("PAPERLESS_THUMBNAIL_FONT_NAME", "/usr/share/fonts/liberation/LiberationSerif-Regular.ttf") + +# Tika settings +PAPERLESS_TIKA_ENABLED = __get_boolean("PAPERLESS_TIKA_ENABLED", "NO") +PAPERLESS_TIKA_ENDPOINT = os.getenv("PAPERLESS_TIKA_ENDPOINT", "http://localhost:9998") +PAPERLESS_TIKA_GOTENBERG_ENDPOINT = os.getenv( + "PAPERLESS_TIKA_GOTENBERG_ENDPOINT", "http://localhost:3000" +) + +if PAPERLESS_TIKA_ENABLED: + INSTALLED_APPS.append("paperless_tika.apps.PaperlessTikaConfig") + +# List dates that should be ignored when trying to parse date from document text +IGNORE_DATES = set() + +if os.getenv("PAPERLESS_IGNORE_DATES", ""): + import dateparser + + for s in os.getenv("PAPERLESS_IGNORE_DATES", "").split(","): + d = dateparser.parse(s) + if d: + IGNORE_DATES.add(d.date()) diff --git a/src/paperless/tests/test_checks.py b/src/paperless/tests/test_checks.py new file mode 100644 index 000000000..e1525cab8 --- /dev/null +++ b/src/paperless/tests/test_checks.py @@ -0,0 +1,54 @@ +import os +import shutil + +from django.test import TestCase, override_settings + +from documents.tests.utils import DirectoriesMixin +from paperless import binaries_check, paths_check +from paperless.checks import debug_mode_check + + +class TestChecks(DirectoriesMixin, TestCase): + + def test_binaries(self): + self.assertEqual(binaries_check(None), []) + + @override_settings(CONVERT_BINARY="uuuhh", OPTIPNG_BINARY="forgot") + def test_binaries_fail(self): + self.assertEqual(len(binaries_check(None)), 2) + + def test_paths_check(self): + self.assertEqual(paths_check(None), []) + + @override_settings(MEDIA_ROOT="uuh", + DATA_DIR="whatever", + CONSUMPTION_DIR="idontcare") + def test_paths_check_dont_exist(self): + msgs = paths_check(None) + self.assertEqual(len(msgs), 3, str(msgs)) + + for msg in msgs: + self.assertTrue(msg.msg.endswith("is set but doesn't exist.")) + + def test_paths_check_no_access(self): + os.chmod(self.dirs.data_dir, 0o000) + os.chmod(self.dirs.media_dir, 0o000) + os.chmod(self.dirs.consumption_dir, 0o000) + + self.addCleanup(os.chmod, self.dirs.data_dir, 0o777) + self.addCleanup(os.chmod, self.dirs.media_dir, 0o777) + self.addCleanup(os.chmod, self.dirs.consumption_dir, 0o777) + + msgs = paths_check(None) + self.assertEqual(len(msgs), 3) + + for msg in msgs: + self.assertTrue(msg.msg.endswith("is not writeable")) + + @override_settings(DEBUG=False) + def test_debug_disabled(self): + self.assertEqual(debug_mode_check(None), []) + + @override_settings(DEBUG=True) + def test_debug_enabled(self): + self.assertEqual(len(debug_mode_check(None)), 1) diff --git a/src/paperless/tests/test_websockets.py b/src/paperless/tests/test_websockets.py new file mode 100644 index 000000000..e1b948edf --- /dev/null +++ b/src/paperless/tests/test_websockets.py @@ -0,0 +1,60 @@ +from unittest import mock + +from channels.layers import get_channel_layer +from channels.testing import WebsocketCommunicator +from django.test import TestCase, override_settings + +from paperless.asgi import application + + +TEST_CHANNEL_LAYERS = { + 'default': { + 'BACKEND': 'channels.layers.InMemoryChannelLayer', + }, +} + + +class TestWebSockets(TestCase): + + @override_settings(CHANNEL_LAYERS=TEST_CHANNEL_LAYERS) + async def test_no_auth(self): + communicator = WebsocketCommunicator(application, "/ws/status/") + connected, subprotocol = await communicator.connect() + self.assertFalse(connected) + await communicator.disconnect() + + @override_settings(CHANNEL_LAYERS=TEST_CHANNEL_LAYERS) + @mock.patch("paperless.consumers.StatusConsumer._authenticated") + async def test_auth(self, _authenticated): + _authenticated.return_value = True + + communicator = WebsocketCommunicator(application, "/ws/status/") + connected, subprotocol = await communicator.connect() + self.assertTrue(connected) + + await communicator.disconnect() + + @override_settings(CHANNEL_LAYERS=TEST_CHANNEL_LAYERS) + @mock.patch("paperless.consumers.StatusConsumer._authenticated") + async def test_receive(self, _authenticated): + _authenticated.return_value = True + + communicator = WebsocketCommunicator(application, "/ws/status/") + connected, subprotocol = await communicator.connect() + self.assertTrue(connected) + + message = { + "task_id": "test" + } + + channel_layer = get_channel_layer() + await channel_layer.group_send("status_updates", { + "type": "status_update", + "data": message + }) + + response = await communicator.receive_json_from() + + self.assertEqual(response, message) + + await communicator.disconnect() diff --git a/src/paperless/urls.py b/src/paperless/urls.py old mode 100755 new mode 100644 index 9b390b139..eee04af9f --- a/src/paperless/urls.py +++ b/src/paperless/urls.py @@ -7,26 +7,35 @@ from django.views.generic import RedirectView from rest_framework.authtoken import views from rest_framework.routers import DefaultRouter +from django.utils.translation import gettext_lazy as _ + +from django.conf import settings + +from paperless.consumers import StatusConsumer from documents.views import ( CorrespondentViewSet, - DocumentViewSet, + UnifiedSearchViewSet, LogViewSet, TagViewSet, DocumentTypeViewSet, - SearchView, IndexView, SearchAutoCompleteView, StatisticsView, - PostDocumentView + PostDocumentView, + SavedViewViewSet, + BulkEditView, + SelectionDataView, + BulkDownloadView ) from paperless.views import FaviconView api_router = DefaultRouter() api_router.register(r"correspondents", CorrespondentViewSet) api_router.register(r"document_types", DocumentTypeViewSet) -api_router.register(r"documents", DocumentViewSet) -api_router.register(r"logs", LogViewSet) +api_router.register(r"documents", UnifiedSearchViewSet) +api_router.register(r"logs", LogViewSet, basename="logs") api_router.register(r"tags", TagViewSet) +api_router.register(r"saved_views", SavedViewViewSet) urlpatterns = [ @@ -39,10 +48,6 @@ urlpatterns = [ SearchAutoCompleteView.as_view(), name="autocomplete"), - re_path(r"^search/", - SearchView.as_view(), - name="search"), - re_path(r"^statistics/", StatisticsView.as_view(), name="statistics"), @@ -50,6 +55,15 @@ urlpatterns = [ re_path(r"^documents/post_document/", PostDocumentView.as_view(), name="post_document"), + re_path(r"^documents/bulk_edit/", BulkEditView.as_view(), + name="bulk_edit"), + + re_path(r"^documents/selection_data/", SelectionDataView.as_view(), + name="selection_data"), + + re_path(r"^documents/bulk_download/", BulkDownloadView.as_view(), + name="bulk_download"), + path('token/', views.obtain_auth_token) ] + api_router.urls)), @@ -61,30 +75,41 @@ urlpatterns = [ re_path(r"^fetch/", include([ re_path( r"^doc/(?P\d+)$", - RedirectView.as_view(url='/api/documents/%(pk)s/download/'), + RedirectView.as_view(url=settings.BASE_URL + + 'api/documents/%(pk)s/download/'), ), re_path( r"^thumb/(?P\d+)$", - RedirectView.as_view(url='/api/documents/%(pk)s/thumb/'), + RedirectView.as_view(url=settings.BASE_URL + + 'api/documents/%(pk)s/thumb/'), ), re_path( r"^preview/(?P\d+)$", - RedirectView.as_view(url='/api/documents/%(pk)s/preview/'), + RedirectView.as_view(url=settings.BASE_URL + + 'api/documents/%(pk)s/preview/'), ), ])), re_path(r"^push$", csrf_exempt( - RedirectView.as_view(url='/api/documents/post_document/'))), + RedirectView.as_view(url=settings.BASE_URL + + 'api/documents/post_document/'))), # Frontend assets TODO: this is pretty bad, but it works. path('assets/', - RedirectView.as_view(url='/static/frontend/assets/%(path)s')), + RedirectView.as_view(url=settings.STATIC_URL + + 'frontend/en-US/assets/%(path)s')), + # TODO: with localization, this is even worse! :/ # login, logout path('accounts/', include('django.contrib.auth.urls')), # Root of the Frontent - re_path(r".*", login_required(IndexView.as_view())), + re_path(r".*", login_required(IndexView.as_view()), name='base'), +] + + +websocket_urlpatterns = [ + re_path(r'ws/status/$', StatusConsumer.as_asgi()), ] # Text in each page's

(and above login form). @@ -92,4 +117,4 @@ admin.site.site_header = 'Paperless-ng' # Text at the end of each page's . admin.site.site_title = 'Paperless-ng' # Text at the top of the admin index page. -admin.site.index_title = 'Paperless-ng administration' +admin.site.index_title = _('Paperless-ng administration') diff --git a/src/paperless/version.py b/src/paperless/version.py index 527e0668d..e75384eea 100644 --- a/src/paperless/version.py +++ b/src/paperless/version.py @@ -1 +1 @@ -__version__ = (0, 9, 6) +__version__ = (1, 5, 0) diff --git a/src/paperless/workers.py b/src/paperless/workers.py new file mode 100644 index 000000000..4f2f0802a --- /dev/null +++ b/src/paperless/workers.py @@ -0,0 +1,11 @@ +import os +from uvicorn.workers import UvicornWorker +from django.conf import settings + +os.environ.setdefault("DJANGO_SETTINGS_MODULE", "paperless.settings") + + +class ConfigurableWorker(UvicornWorker): + CONFIG_KWARGS = { + "root_path": settings.FORCE_SCRIPT_NAME or "", + } diff --git a/src/paperless_mail/admin.py b/src/paperless_mail/admin.py index 0440234b7..0c7e01ab8 100644 --- a/src/paperless_mail/admin.py +++ b/src/paperless_mail/admin.py @@ -1,15 +1,30 @@ from django.contrib import admin from paperless_mail.models import MailAccount, MailRule +from django.utils.translation import gettext_lazy as _ + class MailAccountAdmin(admin.ModelAdmin): list_display = ("name", "imap_server", "username") + fieldsets = [ + (None, { + 'fields': ['name', 'imap_server', 'imap_port'] + }), + (_("Authentication"), { + 'fields': ['imap_security', 'username', 'password'] + }), + (_("Advanced settings"), { + 'fields': ['character_set'] + }) + ] + class MailRuleAdmin(admin.ModelAdmin): radio_fields = { + "attachment_type": admin.VERTICAL, "action": admin.VERTICAL, "assign_title_from": admin.VERTICAL, "assign_correspondent_from": admin.VERTICAL @@ -19,31 +34,33 @@ class MailRuleAdmin(admin.ModelAdmin): (None, { 'fields': ('name', 'order', 'account', 'folder') }), - ("Filter", { + (_("Filter"), { 'description': - "Paperless will only process mails that match ALL of the " - "filters given below.", + _("Paperless will only process mails that match ALL of the " + "filters given below."), 'fields': ('filter_from', 'filter_subject', 'filter_body', - 'maximum_age') + 'filter_attachment_filename', + 'maximum_age', + 'attachment_type') }), - ("Actions", { + (_("Actions"), { 'description': - "The action applied to the mail. This action is only " - "performed when documents were consumed from the mail. Mails " - "without attachments will remain entirely untouched.", + _("The action applied to the mail. This action is only " + "performed when documents were consumed from the mail. " + "Mails without attachments will remain entirely untouched."), 'fields': ( 'action', 'action_parameter') }), - ("Metadata", { + (_("Metadata"), { 'description': - "Assign metadata to documents consumed from this rule " - "automatically. If you do not assign tags, types or " - "correspondents here, paperless will still process all " - "matching rules that you have defined.", + _("Assign metadata to documents consumed from this rule " + "automatically. If you do not assign tags, types or " + "correspondents here, paperless will still process all " + "matching rules that you have defined."), "fields": ( 'assign_title_from', 'assign_tag', diff --git a/src/paperless_mail/apps.py b/src/paperless_mail/apps.py index 18c15a4f4..efbea745b 100644 --- a/src/paperless_mail/apps.py +++ b/src/paperless_mail/apps.py @@ -1,7 +1,9 @@ from django.apps import AppConfig +from django.utils.translation import gettext_lazy as _ + class PaperlessMailConfig(AppConfig): name = 'paperless_mail' - verbose_name = 'Paperless Mail' + verbose_name = _('Paperless mail') diff --git a/src/paperless_mail/mail.py b/src/paperless_mail/mail.py index a82c34f15..327731a84 100644 --- a/src/paperless_mail/mail.py +++ b/src/paperless_mail/mail.py @@ -1,11 +1,12 @@ import os import tempfile from datetime import timedelta, date +from fnmatch import fnmatch import magic +import pathvalidate from django.conf import settings from django.db import DatabaseError -from django.utils.text import slugify from django_q.tasks import async_task from imap_tools import MailBox, MailBoxUnencrypted, AND, MailMessageFlags, \ MailboxFolderSelectError @@ -26,7 +27,7 @@ class BaseMailAction: return {} def post_consume(self, M, message_uids, parameter): - pass + pass # pragma: nocover class DeleteMailAction(BaseMailAction): @@ -69,14 +70,14 @@ def get_rule_action(rule): elif rule.action == MailRule.ACTION_MARK_READ: return MarkReadMailAction() else: - raise ValueError("Unknown action.") + raise NotImplementedError("Unknown action.") # pragma: nocover def make_criterias(rule): maximum_age = date.today() - timedelta(days=rule.maximum_age) - criterias = { - "date_gte": maximum_age - } + criterias = {} + if rule.maximum_age > 0: + criterias["date_gte"] = maximum_age if rule.filter_from: criterias["from_"] = rule.filter_from if rule.filter_subject: @@ -95,12 +96,14 @@ def get_mailbox(server, port, security): elif security == MailAccount.IMAP_SECURITY_SSL: mailbox = MailBox(server, port) else: - raise ValueError("Unknown IMAP security") + raise NotImplementedError("Unknown IMAP security") # pragma: nocover return mailbox class MailAccountHandler(LoggingMixin): + logging_name = "paperless_mail" + def _correspondent_from_name(self, name): try: return Correspondent.objects.get_or_create(name=name)[0] @@ -119,7 +122,7 @@ class MailAccountHandler(LoggingMixin): return os.path.splitext(os.path.basename(att.filename))[0] else: - raise ValueError("Unknown title selector.") + raise NotImplementedError("Unknown title selector.") # pragma: nocover # NOQA: E501 def get_correspondent(self, message, rule): c_from = rule.assign_correspondent_from @@ -141,7 +144,7 @@ class MailAccountHandler(LoggingMixin): return rule.assign_correspondent else: - raise ValueError("Unknwown correspondent selector") + raise NotImplementedError("Unknwown correspondent selector") # pragma: nocover # NOQA: E501 def handle_mail_account(self, account): @@ -197,8 +200,10 @@ class MailAccountHandler(LoggingMixin): f"{str(AND(**criterias))}") try: - messages = M.fetch(criteria=AND(**criterias), - mark_seen=False) + messages = M.fetch( + criteria=AND(**criterias), + mark_seen=False, + charset=rule.account.character_set) except Exception: raise MailError( f"Rule {rule}: Error while fetching folder {rule.folder}") @@ -263,7 +268,7 @@ class MailAccountHandler(LoggingMixin): for att in message.attachments: - if not att.content_disposition == "attachment": + if not att.content_disposition == "attachment" and rule.attachment_type == MailRule.ATTACHMENT_TYPE_ATTACHMENTS_ONLY: # NOQA: E501 self.log( 'debug', f"Rule {rule}: " @@ -271,6 +276,10 @@ class MailAccountHandler(LoggingMixin): f"with content disposition {att.content_disposition}") continue + if rule.filter_attachment_filename: + if not fnmatch(att.filename, rule.filter_attachment_filename): + continue + title = self.get_title(message, att, rule) # don't trust the content type of the attachment. Could be @@ -294,7 +303,7 @@ class MailAccountHandler(LoggingMixin): async_task( "documents.tasks.consume_file", path=temp_filename, - override_filename=att.filename, + override_filename=pathvalidate.sanitize_filename(att.filename), # NOQA: E501 override_title=title, override_correspondent_id=correspondent.id if correspondent else None, # NOQA: E501 override_document_type_id=doc_type.id if doc_type else None, # NOQA: E501 diff --git a/src/paperless_mail/migrations/0006_auto_20210101_2340.py b/src/paperless_mail/migrations/0006_auto_20210101_2340.py new file mode 100644 index 000000000..c03ee627d --- /dev/null +++ b/src/paperless_mail/migrations/0006_auto_20210101_2340.py @@ -0,0 +1,128 @@ +# Generated by Django 3.1.4 on 2021-01-01 23:40 + +from django.db import migrations, models +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + ('documents', '1011_auto_20210101_2340'), + ('paperless_mail', '0005_help_texts'), + ] + + operations = [ + migrations.AlterModelOptions( + name='mailaccount', + options={'verbose_name': 'mail account', 'verbose_name_plural': 'mail accounts'}, + ), + migrations.AlterModelOptions( + name='mailrule', + options={'verbose_name': 'mail rule', 'verbose_name_plural': 'mail rules'}, + ), + migrations.AlterField( + model_name='mailaccount', + name='imap_port', + field=models.IntegerField(blank=True, help_text='This is usually 143 for unencrypted and STARTTLS connections, and 993 for SSL connections.', null=True, verbose_name='IMAP port'), + ), + migrations.AlterField( + model_name='mailaccount', + name='imap_security', + field=models.PositiveIntegerField(choices=[(1, 'No encryption'), (2, 'Use SSL'), (3, 'Use STARTTLS')], default=2, verbose_name='IMAP security'), + ), + migrations.AlterField( + model_name='mailaccount', + name='imap_server', + field=models.CharField(max_length=256, verbose_name='IMAP server'), + ), + migrations.AlterField( + model_name='mailaccount', + name='name', + field=models.CharField(max_length=256, unique=True, verbose_name='name'), + ), + migrations.AlterField( + model_name='mailaccount', + name='password', + field=models.CharField(max_length=256, verbose_name='password'), + ), + migrations.AlterField( + model_name='mailaccount', + name='username', + field=models.CharField(max_length=256, verbose_name='username'), + ), + migrations.AlterField( + model_name='mailrule', + name='account', + field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='rules', to='paperless_mail.mailaccount', verbose_name='account'), + ), + migrations.AlterField( + model_name='mailrule', + name='action', + field=models.PositiveIntegerField(choices=[(3, "Mark as read, don't process read mails"), (4, "Flag the mail, don't process flagged mails"), (2, 'Move to specified folder'), (1, 'Delete')], default=3, verbose_name='action'), + ), + migrations.AlterField( + model_name='mailrule', + name='action_parameter', + field=models.CharField(blank=True, help_text='Additional parameter for the action selected above, i.e., the target folder of the move to folder action.', max_length=256, null=True, verbose_name='action parameter'), + ), + migrations.AlterField( + model_name='mailrule', + name='assign_correspondent', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='documents.correspondent', verbose_name='assign this correspondent'), + ), + migrations.AlterField( + model_name='mailrule', + name='assign_correspondent_from', + field=models.PositiveIntegerField(choices=[(1, 'Do not assign a correspondent'), (2, 'Use mail address'), (3, 'Use name (or mail address if not available)'), (4, 'Use correspondent selected below')], default=1, verbose_name='assign correspondent from'), + ), + migrations.AlterField( + model_name='mailrule', + name='assign_document_type', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='documents.documenttype', verbose_name='assign this document type'), + ), + migrations.AlterField( + model_name='mailrule', + name='assign_tag', + field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, to='documents.tag', verbose_name='assign this tag'), + ), + migrations.AlterField( + model_name='mailrule', + name='assign_title_from', + field=models.PositiveIntegerField(choices=[(1, 'Use subject as title'), (2, 'Use attachment filename as title')], default=1, verbose_name='assign title from'), + ), + migrations.AlterField( + model_name='mailrule', + name='filter_body', + field=models.CharField(blank=True, max_length=256, null=True, verbose_name='filter body'), + ), + migrations.AlterField( + model_name='mailrule', + name='filter_from', + field=models.CharField(blank=True, max_length=256, null=True, verbose_name='filter from'), + ), + migrations.AlterField( + model_name='mailrule', + name='filter_subject', + field=models.CharField(blank=True, max_length=256, null=True, verbose_name='filter subject'), + ), + migrations.AlterField( + model_name='mailrule', + name='folder', + field=models.CharField(default='INBOX', max_length=256, verbose_name='folder'), + ), + migrations.AlterField( + model_name='mailrule', + name='maximum_age', + field=models.PositiveIntegerField(default=30, help_text='Specified in days.', verbose_name='maximum age'), + ), + migrations.AlterField( + model_name='mailrule', + name='name', + field=models.CharField(max_length=256, unique=True, verbose_name='name'), + ), + migrations.AlterField( + model_name='mailrule', + name='order', + field=models.IntegerField(default=0, verbose_name='order'), + ), + ] diff --git a/src/paperless_mail/migrations/0007_auto_20210106_0138.py b/src/paperless_mail/migrations/0007_auto_20210106_0138.py new file mode 100644 index 000000000..2da9eecc9 --- /dev/null +++ b/src/paperless_mail/migrations/0007_auto_20210106_0138.py @@ -0,0 +1,23 @@ +# Generated by Django 3.1.5 on 2021-01-06 01:38 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('paperless_mail', '0006_auto_20210101_2340'), + ] + + operations = [ + migrations.AddField( + model_name='mailrule', + name='attachment_type', + field=models.PositiveIntegerField(choices=[(1, 'Only process attachments.'), (2, "Process all files, including 'inline' attachments.")], default=1, help_text="Inline attachments include embedded images, so it's best to combine this option with a filename filter.", verbose_name='attachment type'), + ), + migrations.AddField( + model_name='mailrule', + name='filter_attachment_filename', + field=models.CharField(blank=True, help_text='Only consume documents which entirely match this filename if specified. Wildcards such as *.pdf or *invoice* are allowed. Case insensitive.', max_length=256, null=True, verbose_name='filter attachment filename'), + ), + ] diff --git a/src/paperless_mail/migrations/0008_auto_20210516_0940.py b/src/paperless_mail/migrations/0008_auto_20210516_0940.py new file mode 100644 index 000000000..3cb98752c --- /dev/null +++ b/src/paperless_mail/migrations/0008_auto_20210516_0940.py @@ -0,0 +1,28 @@ +# Generated by Django 3.2.3 on 2021-05-16 09:40 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('paperless_mail', '0007_auto_20210106_0138'), + ] + + operations = [ + migrations.AddField( + model_name='mailaccount', + name='character_set', + field=models.CharField(default='UTF-8', help_text="The character set to use when communicating with the mail server, such as 'UTF-8' or 'US-ASCII'.", max_length=256, verbose_name='character set'), + ), + migrations.AlterField( + model_name='mailrule', + name='action_parameter', + field=models.CharField(blank=True, help_text='Additional parameter for the action selected above, i.e., the target folder of the move to folder action. Subfolders must be separated by dots.', max_length=256, null=True, verbose_name='action parameter'), + ), + migrations.AlterField( + model_name='mailrule', + name='folder', + field=models.CharField(default='INBOX', help_text='Subfolders must be separated by dots.', max_length=256, verbose_name='folder'), + ), + ] diff --git a/src/paperless_mail/models.py b/src/paperless_mail/models.py index aa1ac5684..47921800e 100644 --- a/src/paperless_mail/models.py +++ b/src/paperless_mail/models.py @@ -2,37 +2,61 @@ from django.db import models import documents.models as document_models +from django.utils.translation import gettext_lazy as _ + class MailAccount(models.Model): + class Meta: + verbose_name = _("mail account") + verbose_name_plural = _("mail accounts") + IMAP_SECURITY_NONE = 1 IMAP_SECURITY_SSL = 2 IMAP_SECURITY_STARTTLS = 3 IMAP_SECURITY_OPTIONS = ( - (IMAP_SECURITY_NONE, "No encryption"), - (IMAP_SECURITY_SSL, "Use SSL"), - (IMAP_SECURITY_STARTTLS, "Use STARTTLS"), + (IMAP_SECURITY_NONE, _("No encryption")), + (IMAP_SECURITY_SSL, _("Use SSL")), + (IMAP_SECURITY_STARTTLS, _("Use STARTTLS")), ) - name = models.CharField(max_length=256, unique=True) + name = models.CharField( + _("name"), + max_length=256, unique=True) - imap_server = models.CharField(max_length=256) + imap_server = models.CharField( + _("IMAP server"), + max_length=256) imap_port = models.IntegerField( + _("IMAP port"), blank=True, null=True, - help_text="This is usually 143 for unencrypted and STARTTLS " - "connections, and 993 for SSL connections.") + help_text=_("This is usually 143 for unencrypted and STARTTLS " + "connections, and 993 for SSL connections.")) imap_security = models.PositiveIntegerField( + _("IMAP security"), choices=IMAP_SECURITY_OPTIONS, default=IMAP_SECURITY_SSL ) - username = models.CharField(max_length=256) + username = models.CharField( + _("username"), + max_length=256) - password = models.CharField(max_length=256) + password = models.CharField( + _("password"), + max_length=256) + + character_set = models.CharField( + _("character set"), + max_length=256, + default="UTF-8", + help_text=_("The character set to use when communicating with the " + "mail server, such as 'UTF-8' or 'US-ASCII'.") + ) def __str__(self): return self.name @@ -40,24 +64,37 @@ class MailAccount(models.Model): class MailRule(models.Model): + class Meta: + verbose_name = _("mail rule") + verbose_name_plural = _("mail rules") + + ATTACHMENT_TYPE_ATTACHMENTS_ONLY = 1 + ATTACHMENT_TYPE_EVERYTHING = 2 + + ATTACHMENT_TYPES = ( + (ATTACHMENT_TYPE_ATTACHMENTS_ONLY, _("Only process attachments.")), + (ATTACHMENT_TYPE_EVERYTHING, _("Process all files, including 'inline' " + "attachments.")) + ) + ACTION_DELETE = 1 ACTION_MOVE = 2 ACTION_MARK_READ = 3 ACTION_FLAG = 4 ACTIONS = ( - (ACTION_MARK_READ, "Mark as read, don't process read mails"), - (ACTION_FLAG, "Flag the mail, don't process flagged mails"), - (ACTION_MOVE, "Move to specified folder"), - (ACTION_DELETE, "Delete"), + (ACTION_MARK_READ, _("Mark as read, don't process read mails")), + (ACTION_FLAG, _("Flag the mail, don't process flagged mails")), + (ACTION_MOVE, _("Move to specified folder")), + (ACTION_DELETE, _("Delete")), ) TITLE_FROM_SUBJECT = 1 TITLE_FROM_FILENAME = 2 TITLE_SELECTOR = ( - (TITLE_FROM_SUBJECT, "Use subject as title"), - (TITLE_FROM_FILENAME, "Use attachment filename as title") + (TITLE_FROM_SUBJECT, _("Use subject as title")), + (TITLE_FROM_FILENAME, _("Use attachment filename as title")) ) CORRESPONDENT_FROM_NOTHING = 1 @@ -67,47 +104,84 @@ class MailRule(models.Model): CORRESPONDENT_SELECTOR = ( (CORRESPONDENT_FROM_NOTHING, - "Do not assign a correspondent"), + _("Do not assign a correspondent")), (CORRESPONDENT_FROM_EMAIL, - "Use mail address"), + _("Use mail address")), (CORRESPONDENT_FROM_NAME, - "Use name (or mail address if not available)"), + _("Use name (or mail address if not available)")), (CORRESPONDENT_FROM_CUSTOM, - "Use correspondent selected below") + _("Use correspondent selected below")) ) - name = models.CharField(max_length=256, unique=True) + name = models.CharField( + _("name"), + max_length=256, unique=True) - order = models.IntegerField(default=0) + order = models.IntegerField( + _("order"), + default=0) account = models.ForeignKey( MailAccount, related_name="rules", - on_delete=models.CASCADE + on_delete=models.CASCADE, + verbose_name=_("account") ) - folder = models.CharField(default='INBOX', max_length=256) + folder = models.CharField( + _("folder"), + default='INBOX', max_length=256, + help_text=_("Subfolders must be separated by dots.") + ) - filter_from = models.CharField(max_length=256, null=True, blank=True) - filter_subject = models.CharField(max_length=256, null=True, blank=True) - filter_body = models.CharField(max_length=256, null=True, blank=True) + filter_from = models.CharField( + _("filter from"), + max_length=256, null=True, blank=True) + filter_subject = models.CharField( + _("filter subject"), + max_length=256, null=True, blank=True) + filter_body = models.CharField( + _("filter body"), + max_length=256, null=True, blank=True) + + filter_attachment_filename = models.CharField( + _("filter attachment filename"), + max_length=256, null=True, blank=True, + help_text=_("Only consume documents which entirely match this " + "filename if specified. Wildcards such as *.pdf or " + "*invoice* are allowed. Case insensitive.") + ) maximum_age = models.PositiveIntegerField( + _("maximum age"), default=30, - help_text="Specified in days.") + help_text=_("Specified in days.")) + + attachment_type = models.PositiveIntegerField( + _("attachment type"), + choices=ATTACHMENT_TYPES, + default=ATTACHMENT_TYPE_ATTACHMENTS_ONLY, + help_text=_("Inline attachments include embedded images, so it's best " + "to combine this option with a filename filter.") + ) action = models.PositiveIntegerField( + _("action"), choices=ACTIONS, default=ACTION_MARK_READ, ) action_parameter = models.CharField( + _("action parameter"), max_length=256, blank=True, null=True, - help_text="Additional parameter for the action selected above, i.e., " - "the target folder of the move to folder action." + help_text=_("Additional parameter for the action selected above, " + "i.e., " + "the target folder of the move to folder action. " + "Subfolders must be separated by dots.") ) assign_title_from = models.PositiveIntegerField( + _("assign title from"), choices=TITLE_SELECTOR, default=TITLE_FROM_SUBJECT ) @@ -116,17 +190,20 @@ class MailRule(models.Model): document_models.Tag, null=True, blank=True, - on_delete=models.SET_NULL + on_delete=models.SET_NULL, + verbose_name=_("assign this tag"), ) assign_document_type = models.ForeignKey( document_models.DocumentType, null=True, blank=True, - on_delete=models.SET_NULL + on_delete=models.SET_NULL, + verbose_name=_("assign this document type"), ) assign_correspondent_from = models.PositiveIntegerField( + _("assign correspondent from"), choices=CORRESPONDENT_SELECTOR, default=CORRESPONDENT_FROM_NOTHING ) @@ -135,7 +212,8 @@ class MailRule(models.Model): document_models.Correspondent, null=True, blank=True, - on_delete=models.SET_NULL + on_delete=models.SET_NULL, + verbose_name=_("assign this correspondent") ) def __str__(self): diff --git a/src/paperless_mail/tasks.py b/src/paperless_mail/tasks.py index 68fb859a4..c591f04b9 100644 --- a/src/paperless_mail/tasks.py +++ b/src/paperless_mail/tasks.py @@ -4,17 +4,17 @@ from paperless_mail.mail import MailAccountHandler, MailError from paperless_mail.models import MailAccount +logger = logging.getLogger("paperless.mail.tasks") + + def process_mail_accounts(): total_new_documents = 0 for account in MailAccount.objects.all(): try: total_new_documents += MailAccountHandler().handle_mail_account( account) - except MailError as e: - logging.getLogger(__name__).error( - f"Error while processing mail account {account}: {e}", - exc_info=True - ) + except MailError: + logger.exception(f"Error while processing mail account {account}") if total_new_documents > 0: return f"Added {total_new_documents} document(s)." @@ -27,4 +27,4 @@ def process_mail_account(name): account = MailAccount.objects.get(name=name) MailAccountHandler().handle_mail_account(account) except MailAccount.DoesNotExist: - logging.getLogger(__name__).error(f"Unknown mail acccount: {name}") + logger.error(f"Unknown mail acccount: {name}") diff --git a/src/paperless_mail/tests/test_mail.py b/src/paperless_mail/tests/test_mail.py index 2a391a268..bec0ff4b4 100644 --- a/src/paperless_mail/tests/test_mail.py +++ b/src/paperless_mail/tests/test_mail.py @@ -1,3 +1,4 @@ +import os import uuid from collections import namedtuple from typing import ContextManager @@ -9,6 +10,7 @@ from django.test import TestCase from imap_tools import MailMessageFlags, MailboxFolderSelectError from documents.models import Correspondent +from documents.tests.utils import DirectoriesMixin from paperless_mail import tasks from paperless_mail.mail import MailError, MailAccountHandler from paperless_mail.models import MailRule, MailAccount @@ -41,7 +43,7 @@ class BogusMailBox(ContextManager): folder = BogusFolderManager() - def fetch(self, criteria, mark_seen): + def fetch(self, criteria, mark_seen, charset=""): msg = self.messages criteria = str(criteria).strip('()').split(" ") @@ -130,7 +132,7 @@ def fake_magic_from_buffer(buffer, mime=False): @mock.patch('paperless_mail.mail.magic.from_buffer', fake_magic_from_buffer) -class TestMail(TestCase): +class TestMail(DirectoriesMixin, TestCase): def setUp(self): patcher = mock.patch('paperless_mail.mail.MailBox') @@ -146,6 +148,7 @@ class TestMail(TestCase): self.reset_bogus_mailbox() self.mail_account_handler = MailAccountHandler() + super(TestMail, self).setUp() def reset_bogus_mailbox(self): self.bogus_mailbox.messages = [] @@ -220,9 +223,13 @@ class TestMail(TestCase): args1, kwargs1 = self.async_task.call_args_list[0] args2, kwargs2 = self.async_task.call_args_list[1] + self.assertTrue(os.path.isfile(kwargs1['path']), kwargs1['path']) + self.assertEqual(kwargs1['override_title'], "file_0") self.assertEqual(kwargs1['override_filename'], "file_0.pdf") + self.assertTrue(os.path.isfile(kwargs2['path']), kwargs1['path']) + self.assertEqual(kwargs2['override_title'], "file_1") self.assertEqual(kwargs2['override_filename'], "file_1.pdf") @@ -253,6 +260,7 @@ class TestMail(TestCase): self.assertEqual(self.async_task.call_count, 1) args, kwargs = self.async_task.call_args + self.assertTrue(os.path.isfile(kwargs['path']), kwargs['path']) self.assertEqual(kwargs['override_filename'], "f1.pdf") def test_handle_disposition(self): @@ -273,6 +281,49 @@ class TestMail(TestCase): args, kwargs = self.async_task.call_args self.assertEqual(kwargs['override_filename'], "f2.pdf") + def test_handle_inline_files(self): + message = create_message() + message.attachments = [ + create_attachment(filename="f1.pdf", content_disposition='inline'), + create_attachment(filename="f2.pdf", content_disposition='attachment') + ] + + account = MailAccount() + rule = MailRule(assign_title_from=MailRule.TITLE_FROM_FILENAME, account=account, attachment_type=MailRule.ATTACHMENT_TYPE_EVERYTHING) + + result = self.mail_account_handler.handle_message(message, rule) + + self.assertEqual(result, 2) + self.assertEqual(self.async_task.call_count, 2) + + def test_filename_filter(self): + message = create_message() + message.attachments = [ + create_attachment(filename="f1.pdf"), + create_attachment(filename="f2.pdf"), + create_attachment(filename="f3.pdf"), + create_attachment(filename="f2.png"), + ] + + tests = [ + ("*.pdf", ["f1.pdf", "f2.pdf", "f3.pdf"]), + ("f1.pdf", ["f1.pdf"]), + ("f1", []), + ("*", ["f1.pdf", "f2.pdf", "f3.pdf", "f2.png"]), + ("*.png", ["f2.png"]), + ] + + for (pattern, matches) in tests: + self.async_task.reset_mock() + account = MailAccount() + rule = MailRule(assign_title_from=MailRule.TITLE_FROM_FILENAME, account=account, filter_attachment_filename=pattern) + + result = self.mail_account_handler.handle_message(message, rule) + + self.assertEqual(result, len(matches)) + filenames = [a[1]['override_filename'] for a in self.async_task.call_args_list] + self.assertCountEqual(filenames, matches) + def test_handle_mail_account_mark_read(self): account = MailAccount.objects.create(name="test", imap_server="", username="admin", password="secret") @@ -399,7 +450,7 @@ class TestMail(TestCase): c = Correspondent.objects.get(name="amazon@amazon.de") # should work - self.assertEquals(kwargs['override_correspondent_id'], c.id) + self.assertEqual(kwargs['override_correspondent_id'], c.id) self.async_task.reset_mock() self.reset_bogus_mailbox() @@ -411,7 +462,7 @@ class TestMail(TestCase): args, kwargs = self.async_task.call_args self.async_task.assert_called_once() - self.assertEquals(kwargs['override_correspondent_id'], None) + self.assertEqual(kwargs['override_correspondent_id'], None) def test_filters(self): diff --git a/src/paperless_tesseract/checks.py b/src/paperless_tesseract/checks.py index 41ea3c9b5..d58b7ac6d 100644 --- a/src/paperless_tesseract/checks.py +++ b/src/paperless_tesseract/checks.py @@ -1,7 +1,7 @@ import subprocess from django.conf import settings -from django.core.checks import Error, register +from django.core.checks import Error, Warning, register def get_tesseract_langs(): diff --git a/src/paperless_tesseract/languages.py b/src/paperless_tesseract/languages.py deleted file mode 100644 index 5ea560654..000000000 --- a/src/paperless_tesseract/languages.py +++ /dev/null @@ -1,194 +0,0 @@ -# Thanks to the Library of Congress and some creative use of sed and awk: -# http://www.loc.gov/standards/iso639-2/php/English_list.php - -ISO639 = { - - "aa": "aar", - "ab": "abk", - "ae": "ave", - "af": "afr", - "ak": "aka", - "am": "amh", - "an": "arg", - "ar": "ara", - "as": "asm", - "av": "ava", - "ay": "aym", - "az": "aze", - "ba": "bak", - "be": "bel", - "bg": "bul", - "bh": "bih", - "bi": "bis", - "bm": "bam", - "bn": "ben", - "bo": "bod", - "br": "bre", - "bs": "bos", - "ca": "cat", - "ce": "che", - "ch": "cha", - "co": "cos", - "cr": "cre", - "cs": "ces", - "cu": "chu", - "cv": "chv", - "cy": "cym", - "da": "dan", - "de": "deu", - "dv": "div", - "dz": "dzo", - "ee": "ewe", - "el": "ell", - "en": "eng", - "eo": "epo", - "es": "spa", - "et": "est", - "eu": "eus", - "fa": "fas", - "ff": "ful", - "fi": "fin", - "fj": "fij", - "fo": "fao", - "fr": "fra", - "fy": "fry", - "ga": "gle", - "gd": "gla", - "gl": "glg", - "gn": "grn", - "gu": "guj", - "gv": "glv", - "ha": "hau", - "he": "heb", - "hi": "hin", - "ho": "hmo", - "hr": "hrv", - "ht": "hat", - "hu": "hun", - "hy": "hye", - "hz": "her", - "ia": "ina", - "id": "ind", - "ie": "ile", - "ig": "ibo", - "ii": "iii", - "ik": "ipk", - "io": "ido", - "is": "isl", - "it": "ita", - "iu": "iku", - "ja": "jpn", - "jv": "jav", - "ka": "kat", - "kg": "kon", - "ki": "kik", - "kj": "kua", - "kk": "kaz", - "kl": "kal", - "km": "khm", - "kn": "kan", - "ko": "kor", - "kr": "kau", - "ks": "kas", - "ku": "kur", - "kv": "kom", - "kw": "cor", - "ky": "kir", - "la": "lat", - "lb": "ltz", - "lg": "lug", - "li": "lim", - "ln": "lin", - "lo": "lao", - "lt": "lit", - "lu": "lub", - "lv": "lav", - "mg": "mlg", - "mh": "mah", - "mi": "mri", - "mk": "mkd", - "ml": "mal", - "mn": "mon", - "mr": "mar", - "ms": "msa", - "mt": "mlt", - "my": "mya", - "na": "nau", - "nb": "nob", - "nd": "nde", - "ne": "nep", - "ng": "ndo", - "nl": "nld", - "no": "nor", - "nr": "nbl", - "nv": "nav", - "ny": "nya", - "oc": "oci", - "oj": "oji", - "om": "orm", - "or": "ori", - "os": "oss", - "pa": "pan", - "pi": "pli", - "pl": "pol", - "ps": "pus", - "pt": "por", - "qu": "que", - "rm": "roh", - "rn": "run", - "ro": "ron", - "ru": "rus", - "rw": "kin", - "sa": "san", - "sc": "srd", - "sd": "snd", - "se": "sme", - "sg": "sag", - "si": "sin", - "sk": "slk", - "sl": "slv", - "sm": "smo", - "sn": "sna", - "so": "som", - "sq": "sqi", - "sr": "srp", - "ss": "ssw", - "st": "sot", - "su": "sun", - "sv": "swe", - "sw": "swa", - "ta": "tam", - "te": "tel", - "tg": "tgk", - "th": "tha", - "ti": "tir", - "tk": "tuk", - "tl": "tgl", - "tn": "tsn", - "to": "ton", - "tr": "tur", - "ts": "tso", - "tt": "tat", - "tw": "twi", - "ty": "tah", - "ug": "uig", - "uk": "ukr", - "ur": "urd", - "uz": "uzb", - "ve": "ven", - "vi": "vie", - "vo": "vol", - "wa": "wln", - "wo": "wol", - "xh": "xho", - "yi": "yid", - "yo": "yor", - "za": "zha", - - # Tessdata contains two values for Chinese, "chi_sim" and "chi_tra". I - # have no idea which one is better, so I just picked the bigger file. - "zh": "chi_tra", - - "zu": "zul" - -} diff --git a/src/paperless_tesseract/parsers.py b/src/paperless_tesseract/parsers.py index 80e200f27..d53879542 100644 --- a/src/paperless_tesseract/parsers.py +++ b/src/paperless_tesseract/parsers.py @@ -1,16 +1,16 @@ import json import os import re -import subprocess -import ocrmypdf -import pdftotext -import pikepdf from PIL import Image from django.conf import settings -from ocrmypdf import InputFileError, EncryptedPdfError -from documents.parsers import DocumentParser, ParseError, run_convert +from documents.parsers import DocumentParser, ParseError, \ + make_thumbnail_from_pdf + + +class NoTextFoundException(Exception): + pass class RasterisedDocumentParser(DocumentParser): @@ -19,11 +19,16 @@ class RasterisedDocumentParser(DocumentParser): image, whether it's a PDF, or other graphical format (JPEG, TIFF, etc.) """ + logging_name = "paperless.parsing.tesseract" + def extract_metadata(self, document_path, mime_type): - namespace_pattern = re.compile(r"\{(.*)\}(.*)") result = [] if mime_type == 'application/pdf': + import pikepdf + + namespace_pattern = re.compile(r"\{(.*)\}(.*)") + pdf = pikepdf.open(document_path) meta = pdf.open_metadata() for key, value in meta.items(): @@ -46,49 +51,11 @@ class RasterisedDocumentParser(DocumentParser): ) return result - def get_thumbnail(self, document_path, mime_type): - """ - The thumbnail of a PDF is just a 500px wide image of the first page. - """ - - out_path = os.path.join(self.tempdir, "convert.png") - - # Run convert to get a decent thumbnail - try: - run_convert(density=300, - scale="500x5000>", - alpha="remove", - strip=True, - trim=False, - input_file="{}[0]".format(document_path), - output_file=out_path, - logging_group=self.logging_group) - except ParseError: - # if convert fails, fall back to extracting - # the first PDF page as a PNG using Ghostscript - self.log( - 'warning', - "Thumbnail generation with ImageMagick failed, falling back " - "to ghostscript. Check your /etc/ImageMagick-x/policy.xml!") - gs_out_path = os.path.join(self.tempdir, "gs_out.png") - cmd = [settings.GS_BINARY, - "-q", - "-sDEVICE=pngalpha", - "-o", gs_out_path, - document_path] - if not subprocess.Popen(cmd).wait() == 0: - raise ParseError("Thumbnail (gs) failed at {}".format(cmd)) - # then run convert on the output from gs - run_convert(density=300, - scale="500x5000>", - alpha="remove", - strip=True, - trim=False, - input_file=gs_out_path, - output_file=out_path, - logging_group=self.logging_group) - - return out_path + def get_thumbnail(self, document_path, mime_type, file_name=None): + return make_thumbnail_from_pdf( + self.archive_path or document_path, + self.tempdir, + self.logging_group) def is_image(self, mime_type): return mime_type in [ @@ -103,7 +70,7 @@ class RasterisedDocumentParser(DocumentParser): try: with Image.open(image) as im: x, y = im.info['dpi'] - return x + return round(x) except Exception as e: self.log( 'warning', @@ -128,125 +95,215 @@ class RasterisedDocumentParser(DocumentParser): f"Error while calculating DPI for image {image}: {e}") return None - def parse(self, document_path, mime_type): - mode = settings.OCR_MODE + def extract_text(self, sidecar_file, pdf_file): + if sidecar_file and os.path.isfile(sidecar_file): + with open(sidecar_file, "r") as f: + text = f.read() - text_original = get_text_from_pdf(document_path) - has_text = text_original and len(text_original) > 50 + if "[OCR skipped on page" not in text: + # This happens when there's already text in the input file. + # The sidecar file will only contain text for OCR'ed pages. + self.log("debug", "Using text from sidecar file") + return post_process_text(text) + else: + self.log("debug", "Incomplete sidecar file: discarding.") - if mode == "skip_noarchive" and has_text: + # no success with the sidecar file, try PDF + + if not os.path.isfile(pdf_file): + return None + + from pdfminer.high_level import extract_text as pdfminer_extract_text + from pdfminer.pdftypes import PDFException + + try: + stripped = post_process_text(pdfminer_extract_text(pdf_file)) + + self.log("debug", f"Extracted text from PDF file {pdf_file}") + return stripped + except Exception: + # TODO catch all for various issues with PDFminer.six. + # If PDFminer fails, fall back to OCR. + self.log("warn", + "Error while getting text from PDF document with " + "pdfminer.six", exc_info=True) + # probably not a PDF file. + return None + + def construct_ocrmypdf_parameters(self, + input_file, + mime_type, + output_file, + sidecar_file, + safe_fallback=False): + ocrmypdf_args = { + 'input_file': input_file, + 'output_file': output_file, + # need to use threads, since this will be run in daemonized + # processes by django-q. + 'use_threads': True, + 'jobs': settings.THREADS_PER_WORKER, + 'language': settings.OCR_LANGUAGE, + 'output_type': settings.OCR_OUTPUT_TYPE, + 'progress_bar': False + } + + if settings.OCR_MODE == 'force' or safe_fallback: + ocrmypdf_args['force_ocr'] = True + elif settings.OCR_MODE in ['skip', 'skip_noarchive']: + ocrmypdf_args['skip_text'] = True + elif settings.OCR_MODE == 'redo': + ocrmypdf_args['redo_ocr'] = True + else: + raise ParseError( + f"Invalid ocr mode: {settings.OCR_MODE}") + + if settings.OCR_CLEAN == 'clean': + ocrmypdf_args['clean'] = True + elif settings.OCR_CLEAN == 'clean-final': + if settings.OCR_MODE == 'redo': + ocrmypdf_args['clean'] = True + else: + ocrmypdf_args['clean_final'] = True + + if settings.OCR_DESKEW and not settings.OCR_MODE == 'redo': + ocrmypdf_args['deskew'] = True + + if settings.OCR_ROTATE_PAGES: + ocrmypdf_args['rotate_pages'] = True + ocrmypdf_args['rotate_pages_threshold'] = settings.OCR_ROTATE_PAGES_THRESHOLD # NOQA: E501 + + if settings.OCR_PAGES > 0: + ocrmypdf_args['pages'] = f"1-{settings.OCR_PAGES}" + else: + # sidecar is incompatible with pages + ocrmypdf_args['sidecar'] = sidecar_file + + if self.is_image(mime_type): + dpi = self.get_dpi(input_file) + a4_dpi = self.calculate_a4_dpi(input_file) + if dpi: + self.log( + "debug", + f"Detected DPI for image {input_file}: {dpi}" + ) + ocrmypdf_args['image_dpi'] = dpi + elif settings.OCR_IMAGE_DPI: + ocrmypdf_args['image_dpi'] = settings.OCR_IMAGE_DPI + elif a4_dpi: + ocrmypdf_args['image_dpi'] = a4_dpi + else: + raise ParseError( + f"Cannot produce archive PDF for image {input_file}, " + f"no DPI information is present in this image and " + f"OCR_IMAGE_DPI is not set.") + + if settings.OCR_USER_ARGS and not safe_fallback: + try: + user_args = json.loads(settings.OCR_USER_ARGS) + ocrmypdf_args = {**ocrmypdf_args, **user_args} + except Exception as e: + self.log( + "warning", + f"There is an issue with PAPERLESS_OCR_USER_ARGS, so " + f"they will not be used. Error: {e}") + + return ocrmypdf_args + + def parse(self, document_path, mime_type, file_name=None): + # This forces tesseract to use one core per page. + os.environ['OMP_THREAD_LIMIT'] = "1" + + if mime_type == "application/pdf": + text_original = self.extract_text(None, document_path) + original_has_text = text_original and len(text_original) > 50 + else: + text_original = None + original_has_text = False + + if settings.OCR_MODE == "skip_noarchive" and original_has_text: self.log("debug", "Document has text, skipping OCRmyPDF entirely.") self.text = text_original return - if mode in ['skip', 'skip_noarchive'] and not has_text: - # upgrade to redo, since there appears to be no text in the - # document. This happens to some weird encrypted documents or - # documents with failed OCR attempts for which OCRmyPDF will - # still report that there actually is text in them. - self.log("debug", - "No text was found in the document and skip is " - "specified. Upgrading OCR mode to redo.") - mode = "redo" + import ocrmypdf + from ocrmypdf import InputFileError, EncryptedPdfError archive_path = os.path.join(self.tempdir, "archive.pdf") + sidecar_file = os.path.join(self.tempdir, "sidecar.txt") - ocr_args = { - 'input_file': document_path, - 'output_file': archive_path, - 'use_threads': True, - 'jobs': settings.THREADS_PER_WORKER, - 'language': settings.OCR_LANGUAGE, - 'output_type': settings.OCR_OUTPUT_TYPE, - 'progress_bar': False, - 'clean': True - } - - if settings.OCR_PAGES > 0: - ocr_args['pages'] = f"1-{settings.OCR_PAGES}" - - # Mode selection. - - if mode in ['skip', 'skip_noarchive']: - ocr_args['skip_text'] = True - elif mode == 'redo': - ocr_args['redo_ocr'] = True - elif mode == 'force': - ocr_args['force_ocr'] = True - else: - raise ParseError( - f"Invalid ocr mode: {mode}") - - if self.is_image(mime_type): - dpi = self.get_dpi(document_path) - a4_dpi = self.calculate_a4_dpi(document_path) - if dpi: - self.log( - "debug", - f"Detected DPI for image {document_path}: {dpi}" - ) - ocr_args['image_dpi'] = dpi - elif settings.OCR_IMAGE_DPI: - ocr_args['image_dpi'] = settings.OCR_IMAGE_DPI - elif a4_dpi: - ocr_args['image_dpi'] = a4_dpi - else: - raise ParseError( - f"Cannot produce archive PDF for image {document_path}, " - f"no DPI information is present in this image and " - f"OCR_IMAGE_DPI is not set.") - - if settings.OCR_USER_ARGS: - try: - user_args = json.loads(settings.OCR_USER_ARGS) - ocr_args = {**ocr_args, **user_args} - except Exception as e: - self.log( - "warning", - f"There is an issue with PAPERLESS_OCR_USER_ARGS, so " - f"they will not be used: {e}") - - # This forces tesseract to use one core per page. - os.environ['OMP_THREAD_LIMIT'] = "1" + args = self.construct_ocrmypdf_parameters( + document_path, mime_type, archive_path, sidecar_file) try: - self.log("debug", - f"Calling OCRmyPDF with {str(ocr_args)}") - ocrmypdf.ocr(**ocr_args) - # success! announce results + self.log("debug", f"Calling OCRmyPDF with args: {args}") + ocrmypdf.ocr(**args) + self.archive_path = archive_path - self.text = get_text_from_pdf(archive_path) + self.text = self.extract_text(sidecar_file, archive_path) - except (InputFileError, EncryptedPdfError) as e: - - self.log("debug", - f"Encountered an error: {e}. Trying to use text from " - f"original.") - # This happens with some PDFs when used with the redo_ocr option. - # This is not the end of the world, we'll just use what we already - # have in the document. - self.text = text_original - # Also, no archived file. if not self.text: - # However, if we don't have anything, fail: - raise ParseError(e) + raise NoTextFoundException( + "No text was found in the original document") + except EncryptedPdfError: + self.log("warning", + "This file is encrypted, OCR is impossible. Using " + "any text present in the original file.") + if original_has_text: + self.text = text_original + except (NoTextFoundException, InputFileError) as e: + self.log("warning", + f"Encountered an error while running OCR: {str(e)}. " + f"Attempting force OCR to get the text.") + + archive_path_fallback = os.path.join( + self.tempdir, "archive-fallback.pdf") + sidecar_file_fallback = os.path.join( + self.tempdir, "sidecar-fallback.txt") + + # Attempt to run OCR with safe settings. + + args = self.construct_ocrmypdf_parameters( + document_path, mime_type, + archive_path_fallback, sidecar_file_fallback, + safe_fallback=True + ) + + try: + self.log("debug", + f"Fallback: Calling OCRmyPDF with args: {args}") + ocrmypdf.ocr(**args) + + # Don't return the archived file here, since this file + # is bigger and blurry due to --force-ocr. + + self.text = self.extract_text( + sidecar_file_fallback, archive_path_fallback) + + except Exception as e: + # If this fails, we have a serious issue at hand. + raise ParseError(f"{e.__class__.__name__}: {str(e)}") except Exception as e: # Anything else is probably serious. - raise ParseError(e) + raise ParseError(f"{e.__class__.__name__}: {str(e)}") + # As a last resort, if we still don't have any text for any reason, + # try to extract the text from the original document. if not self.text: - # This may happen for files that don't have any text. - self.log( - 'warning', - f"Document {document_path} does not have any text." - f"This is probably an error or you tried to add an image " - f"without text, or something is wrong with this document.") - self.text = "" + if original_has_text: + self.text = text_original + else: + self.log( + "warning", + f"No text was found in {document_path}, the content will " + f"be empty." + ) + self.text = "" -def strip_excess_whitespace(text): +def post_process_text(text): if not text: return None @@ -257,21 +314,6 @@ def strip_excess_whitespace(text): r"([^\S\n\r]+)$", '', no_leading_whitespace) # TODO: this needs a rework - return no_trailing_whitespace.strip() - - -def get_text_from_pdf(pdf_file): - - if not os.path.isfile(pdf_file): - return None - - with open(pdf_file, "rb") as f: - try: - pdf = pdftotext.PDF(f) - except pdftotext.Error: - # might not be a PDF file - return None - - text = "\n".join(pdf) - - return strip_excess_whitespace(text) + # replace \0 prevents issues with saving to postgres. + # text may contain \0 when this character is present in PDF files. + return no_trailing_whitespace.strip().replace("\0", " ") diff --git a/src/paperless_tesseract/signals.py b/src/paperless_tesseract/signals.py index 1e1cd1e1a..fedd08a92 100644 --- a/src/paperless_tesseract/signals.py +++ b/src/paperless_tesseract/signals.py @@ -1,9 +1,13 @@ -from .parsers import RasterisedDocumentParser + +def get_parser(*args, **kwargs): + from .parsers import RasterisedDocumentParser + + return RasterisedDocumentParser(*args, **kwargs) def tesseract_consumer_declaration(sender, **kwargs): return { - "parser": RasterisedDocumentParser, + "parser": get_parser, "weight": 0, "mime_types": { "application/pdf": ".pdf", diff --git a/src/paperless_tesseract/tests/samples/encrypted.pdf b/src/paperless_tesseract/tests/samples/encrypted.pdf new file mode 100644 index 000000000..12c5a3070 Binary files /dev/null and b/src/paperless_tesseract/tests/samples/encrypted.pdf differ diff --git a/src/paperless_tesseract/tests/samples/multi-page-mixed.pdf b/src/paperless_tesseract/tests/samples/multi-page-mixed.pdf new file mode 100644 index 000000000..cca8db419 Binary files /dev/null and b/src/paperless_tesseract/tests/samples/multi-page-mixed.pdf differ diff --git a/src/paperless_tesseract/tests/samples/rotated.pdf b/src/paperless_tesseract/tests/samples/rotated.pdf new file mode 100644 index 000000000..8b79a9082 Binary files /dev/null and b/src/paperless_tesseract/tests/samples/rotated.pdf differ diff --git a/src/paperless_tesseract/tests/samples/signed.pdf b/src/paperless_tesseract/tests/samples/signed.pdf new file mode 100644 index 000000000..5996edeea Binary files /dev/null and b/src/paperless_tesseract/tests/samples/signed.pdf differ diff --git a/src/paperless_tesseract/tests/test_checks.py b/src/paperless_tesseract/tests/test_checks.py new file mode 100644 index 000000000..c4f15764e --- /dev/null +++ b/src/paperless_tesseract/tests/test_checks.py @@ -0,0 +1,26 @@ +from unittest import mock + +from django.core.checks import ERROR +from django.test import TestCase, override_settings + +from paperless_tesseract import check_default_language_available + + +class TestChecks(TestCase): + + def test_default_language(self): + msgs = check_default_language_available(None) + + @override_settings(OCR_LANGUAGE="") + def test_no_language(self): + msgs = check_default_language_available(None) + self.assertEqual(len(msgs), 1) + self.assertTrue(msgs[0].msg.startswith("No OCR language has been specified with PAPERLESS_OCR_LANGUAGE")) + + @override_settings(OCR_LANGUAGE="ita") + @mock.patch("paperless_tesseract.checks.get_tesseract_langs") + def test_invalid_language(self, m): + m.return_value = ["deu", "eng"] + msgs = check_default_language_available(None) + self.assertEqual(len(msgs), 1) + self.assertEqual(msgs[0].level, ERROR) diff --git a/src/paperless_tesseract/tests/test_parser.py b/src/paperless_tesseract/tests/test_parser.py index 7be176663..e39f87017 100644 --- a/src/paperless_tesseract/tests/test_parser.py +++ b/src/paperless_tesseract/tests/test_parser.py @@ -7,7 +7,7 @@ from django.test import TestCase, override_settings from documents.parsers import ParseError, run_convert from documents.tests.utils import DirectoriesMixin -from paperless_tesseract.parsers import RasterisedDocumentParser, get_text_from_pdf, strip_excess_whitespace +from paperless_tesseract.parsers import RasterisedDocumentParser, post_process_text image_to_string_calls = [] @@ -32,13 +32,16 @@ class FakeImageFile(ContextManager): return os.path.basename(self.fname) - - class TestParser(DirectoriesMixin, TestCase): def assertContainsStrings(self, content, strings): # Asserts that all strings appear in content, in the given order. - indices = [content.index(s) for s in strings] + indices = [] + for s in strings: + if s in content: + indices.append(content.index(s)) + else: + self.fail(f"'{s}' is not in '{content}'") self.assertListEqual(indices, sorted(indices)) text_cases = [ @@ -53,9 +56,9 @@ class TestParser(DirectoriesMixin, TestCase): ) ] - def test_strip_excess_whitespace(self): + def test_post_process_text(self): for source, result in self.text_cases: - actual_result = strip_excess_whitespace(source) + actual_result = post_process_text(source) self.assertEqual( result, actual_result, @@ -69,16 +72,17 @@ class TestParser(DirectoriesMixin, TestCase): SAMPLE_FILES = os.path.join(os.path.dirname(__file__), "samples") def test_get_text_from_pdf(self): - text = get_text_from_pdf(os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf')) + parser = RasterisedDocumentParser(uuid.uuid4()) + text = parser.extract_text(None, os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf')) self.assertContainsStrings(text.strip(), ["This is a test document."]) def test_thumbnail(self): parser = RasterisedDocumentParser(uuid.uuid4()) - parser.get_thumbnail(os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf'), "application/pdf") - # dont really know how to test it, just call it and assert that it does not raise anything. + thumb = parser.get_thumbnail(os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf'), "application/pdf") + self.assertTrue(os.path.isfile(thumb)) - @mock.patch("paperless_tesseract.parsers.run_convert") + @mock.patch("documents.parsers.run_convert") def test_thumbnail_fallback(self, m): def call_convert(input_file, output_file, **kwargs): @@ -90,8 +94,13 @@ class TestParser(DirectoriesMixin, TestCase): m.side_effect = call_convert parser = RasterisedDocumentParser(uuid.uuid4()) - parser.get_thumbnail(os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf'), "application/pdf") - # dont really know how to test it, just call it and assert that it does not raise anything. + thumb = parser.get_thumbnail(os.path.join(self.SAMPLE_FILES, 'simple-digital.pdf'), "application/pdf") + self.assertTrue(os.path.isfile(thumb)) + + def test_thumbnail_encrypted(self): + parser = RasterisedDocumentParser(uuid.uuid4()) + thumb = parser.get_thumbnail(os.path.join(self.SAMPLE_FILES, 'encrypted.pdf'), "application/pdf") + self.assertTrue(os.path.isfile(thumb)) def test_get_dpi(self): parser = RasterisedDocumentParser(None) @@ -129,15 +138,31 @@ class TestParser(DirectoriesMixin, TestCase): self.assertIsNone(parser.archive_path) self.assertContainsStrings(parser.get_text(), ["Please enter your name in here:", "This is a PDF document with a form."]) - @override_settings(OCR_MODE="redo") - @mock.patch("paperless_tesseract.parsers.get_text_from_pdf", lambda _: None) - def test_with_form_error_notext(self): + @override_settings(OCR_MODE="skip") + def test_signed(self): parser = RasterisedDocumentParser(None) - def f(): - parser.parse(os.path.join(self.SAMPLE_FILES, "with-form.pdf"), "application/pdf") + parser.parse(os.path.join(self.SAMPLE_FILES, "signed.pdf"), "application/pdf") - self.assertRaises(ParseError, f) + self.assertIsNone(parser.archive_path) + self.assertContainsStrings(parser.get_text(), ["This is a digitally signed PDF, created with Acrobat Pro for the Paperless project to enable", "automated testing of signed/encrypted PDFs"]) + + @override_settings(OCR_MODE="skip") + def test_encrypted(self): + parser = RasterisedDocumentParser(None) + + parser.parse(os.path.join(self.SAMPLE_FILES, "encrypted.pdf"), "application/pdf") + + self.assertIsNone(parser.archive_path) + self.assertEqual(parser.get_text(), "") + + + @override_settings(OCR_MODE="redo") + def test_with_form_error_notext(self): + parser = RasterisedDocumentParser(None) + parser.parse(os.path.join(self.SAMPLE_FILES, "with-form.pdf"), "application/pdf") + + self.assertContainsStrings(parser.get_text(), ["Please enter your name in here:", "This is a PDF document with a form."]) @override_settings(OCR_MODE="force") def test_with_form_force(self): @@ -164,17 +189,12 @@ class TestParser(DirectoriesMixin, TestCase): self.assertRaises(ParseError, f) - @mock.patch("paperless_tesseract.parsers.ocrmypdf.ocr") - def test_image_calc_a4_dpi(self, m): + def test_image_calc_a4_dpi(self): parser = RasterisedDocumentParser(None) - parser.parse(os.path.join(self.SAMPLE_FILES, "simple-no-dpi.png"), "image/png") + dpi = parser.calculate_a4_dpi(os.path.join(self.SAMPLE_FILES, "simple-no-dpi.png")) - m.assert_called_once() - - args, kwargs = m.call_args - - self.assertEqual(kwargs['image_dpi'], 62) + self.assertEqual(dpi, 62) @mock.patch("paperless_tesseract.parsers.RasterisedDocumentParser.calculate_a4_dpi") def test_image_dpi_fail(self, m): @@ -258,9 +278,80 @@ class TestParser(DirectoriesMixin, TestCase): def test_skip_noarchive_notext(self): parser = RasterisedDocumentParser(None) parser.parse(os.path.join(self.SAMPLE_FILES, "multi-page-images.pdf"), "application/pdf") - self.assertTrue(os.path.join(parser.archive_path)) + self.assertTrue(os.path.isfile(parser.archive_path)) self.assertContainsStrings(parser.get_text().lower(), ["page 1", "page 2", "page 3"]) + @override_settings(OCR_MODE="skip") + def test_multi_page_mixed(self): + parser = RasterisedDocumentParser(None) + parser.parse(os.path.join(self.SAMPLE_FILES, "multi-page-mixed.pdf"), "application/pdf") + self.assertTrue(os.path.isfile(parser.archive_path)) + self.assertContainsStrings(parser.get_text().lower(), ["page 1", "page 2", "page 3", "page 4", "page 5", "page 6"]) + + with open(os.path.join(parser.tempdir, "sidecar.txt")) as f: + sidecar = f.read() + + self.assertIn("[OCR skipped on page(s) 4-6]", sidecar) + + @override_settings(OCR_MODE="skip_noarchive") + def test_multi_page_mixed_no_archive(self): + parser = RasterisedDocumentParser(None) + parser.parse(os.path.join(self.SAMPLE_FILES, "multi-page-mixed.pdf"), "application/pdf") + self.assertIsNone(parser.archive_path) + self.assertContainsStrings(parser.get_text().lower(), ["page 4", "page 5", "page 6"]) + + @override_settings(OCR_MODE="skip", OCR_ROTATE_PAGES=True) + def test_rotate(self): + parser = RasterisedDocumentParser(None) + parser.parse(os.path.join(self.SAMPLE_FILES, "rotated.pdf"), "application/pdf") + self.assertContainsStrings(parser.get_text(), [ + "This is the text that appears on the first page. It’s a lot of text.", + "Even if the pages are rotated, OCRmyPDF still gets the job done.", + "This is a really weird file with lots of nonsense text.", + "If you read this, it’s your own fault. Also check your screen orientation." + ]) + + def test_ocrmypdf_parameters(self): + parser = RasterisedDocumentParser(None) + params = parser.construct_ocrmypdf_parameters(input_file="input.pdf", output_file="output.pdf", + sidecar_file="sidecar.txt", mime_type="application/pdf", + safe_fallback=False) + + self.assertEqual(params['input_file'], "input.pdf") + self.assertEqual(params['output_file'], "output.pdf") + self.assertEqual(params['sidecar'], "sidecar.txt") + + with override_settings(OCR_CLEAN="none"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertNotIn("clean", params) + self.assertNotIn("clean_final", params) + + with override_settings(OCR_CLEAN="clean"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertTrue(params['clean']) + self.assertNotIn("clean_final", params) + + with override_settings(OCR_CLEAN="clean-final", OCR_MODE="skip"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertTrue(params['clean_final']) + self.assertNotIn("clean", params) + + with override_settings(OCR_CLEAN="clean-final", OCR_MODE="redo"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertTrue(params['clean']) + self.assertNotIn("clean_final", params) + + with override_settings(OCR_DESKEW=True, OCR_MODE="skip"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertTrue(params['deskew']) + + with override_settings(OCR_DESKEW=True, OCR_MODE="redo"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertNotIn('deskew', params) + + with override_settings(OCR_DESKEW=False, OCR_MODE="skip"): + params = parser.construct_ocrmypdf_parameters("", "", "", "") + self.assertNotIn('deskew', params) class TestParserFileTypes(DirectoriesMixin, TestCase): diff --git a/src/paperless_text/parsers.py b/src/paperless_text/parsers.py index f8f369ab0..837f05c9f 100644 --- a/src/paperless_text/parsers.py +++ b/src/paperless_text/parsers.py @@ -1,9 +1,9 @@ import os -import subprocess +from PIL import ImageDraw, ImageFont, Image from django.conf import settings -from documents.parsers import DocumentParser, ParseError +from documents.parsers import DocumentParser class TextDocumentParser(DocumentParser): @@ -11,79 +11,29 @@ class TextDocumentParser(DocumentParser): This parser directly parses a text document (.txt, .md, or .csv) """ - def get_thumbnail(self, document_path, mime_type): - """ - The thumbnail of a text file is just a 500px wide image of the text - rendered onto a letter-sized page. - """ - # The below is heavily cribbed from https://askubuntu.com/a/590951 + logging_name = "paperless.parsing.text" - bg_color = "white" # bg color - text_color = "black" # text color - psize = [500, 647] # icon size - n_lines = 50 # number of lines to show - out_path = os.path.join(self.tempdir, "convert.png") - - temp_bg = os.path.join(self.tempdir, "bg.png") - temp_txlayer = os.path.join(self.tempdir, "tx.png") - picsize = "x".join([str(n) for n in psize]) - txsize = "x".join([str(n - 8) for n in psize]) - - def create_bg(): - work_size = ",".join([str(n - 1) for n in psize]) - r = str(round(psize[0] / 10)) - rounded = ",".join([r, r]) - run_command( - settings.CONVERT_BINARY, - "-size ", picsize, - ' xc:none -draw ', - '"fill ', bg_color, ' roundrectangle 0,0,', work_size, ",", rounded, '" ', # NOQA: E501 - temp_bg - ) + def get_thumbnail(self, document_path, mime_type, file_name=None): def read_text(): with open(document_path, 'r') as src: lines = [line.strip() for line in src.readlines()] - text = "\n".join([line for line in lines[:n_lines]]) - return text.replace('"', "'") + text = "\n".join(lines[:50]) + return text - def create_txlayer(): - run_command( - settings.CONVERT_BINARY, - "-background none", - "-fill", - text_color, - "-pointsize", "12", - "-border 4 -bordercolor none", - "-size ", txsize, - ' caption:"', read_text(), '" ', - temp_txlayer - ) + img = Image.new("RGB", (500, 700), color="white") + draw = ImageDraw.Draw(img) + font = ImageFont.truetype( + font=settings.THUMBNAIL_FONT_NAME, + size=20, + layout_engine=ImageFont.LAYOUT_BASIC) + draw.text((5, 5), read_text(), font=font, fill="black") - create_txlayer() - create_bg() - run_command( - settings.CONVERT_BINARY, - temp_bg, - temp_txlayer, - "-background None -layers merge ", - out_path - ) + out_path = os.path.join(self.tempdir, "thumb.png") + img.save(out_path) return out_path - def parse(self, document_path, mime_type): + def parse(self, document_path, mime_type, file_name=None): with open(document_path, 'r') as f: self.text = f.read() - - -def run_command(*args): - environment = os.environ.copy() - if settings.CONVERT_MEMORY_LIMIT: - environment["MAGICK_MEMORY_LIMIT"] = settings.CONVERT_MEMORY_LIMIT - if settings.CONVERT_TMPDIR: - environment["MAGICK_TMPDIR"] = settings.CONVERT_TMPDIR - - if not subprocess.Popen(' '.join(args), env=environment, - shell=True).wait() == 0: - raise ParseError("Convert failed at {}".format(args)) diff --git a/src/paperless_text/signals.py b/src/paperless_text/signals.py index 1e0493f4f..833d0be28 100644 --- a/src/paperless_text/signals.py +++ b/src/paperless_text/signals.py @@ -1,9 +1,13 @@ -from .parsers import TextDocumentParser + +def get_parser(*args, **kwargs): + from .parsers import TextDocumentParser + + return TextDocumentParser(*args, **kwargs) def text_consumer_declaration(sender, **kwargs): return { - "parser": TextDocumentParser, + "parser": get_parser, "weight": 10, "mime_types": { "text/plain": ".txt", diff --git a/src/paperless_text/tests/samples/test.txt b/src/paperless_text/tests/samples/test.txt new file mode 100644 index 000000000..6de7b8c69 --- /dev/null +++ b/src/paperless_text/tests/samples/test.txt @@ -0,0 +1 @@ +This is a test file. diff --git a/src/paperless_text/tests/test_parser.py b/src/paperless_text/tests/test_parser.py new file mode 100644 index 000000000..413aa91cf --- /dev/null +++ b/src/paperless_text/tests/test_parser.py @@ -0,0 +1,26 @@ +import os + +from django.test import TestCase + +from documents.tests.utils import DirectoriesMixin +from paperless_text.parsers import TextDocumentParser + + +class TestTextParser(DirectoriesMixin, TestCase): + + def test_thumbnail(self): + + parser = TextDocumentParser(None) + + # just make sure that it does not crash + f = parser.get_thumbnail(os.path.join(os.path.dirname(__file__), "samples", "test.txt"), "text/plain") + self.assertTrue(os.path.isfile(f)) + + def test_parse(self): + + parser = TextDocumentParser(None) + + parser.parse(os.path.join(os.path.dirname(__file__), "samples", "test.txt"), "text/plain") + + self.assertEqual(parser.get_text(), "This is a test file.\n") + self.assertIsNone(parser.get_archive_path()) diff --git a/src/paperless_tika/__init__.py b/src/paperless_tika/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/src/paperless_tika/apps.py b/src/paperless_tika/apps.py new file mode 100644 index 000000000..5cab21427 --- /dev/null +++ b/src/paperless_tika/apps.py @@ -0,0 +1,14 @@ +from django.apps import AppConfig +from django.conf import settings +from paperless_tika.signals import tika_consumer_declaration + + +class PaperlessTikaConfig(AppConfig): + name = "paperless_tika" + + def ready(self): + from documents.signals import document_consumer_declaration + + if settings.PAPERLESS_TIKA_ENABLED: + document_consumer_declaration.connect(tika_consumer_declaration) + AppConfig.ready(self) diff --git a/src/paperless_tika/parsers.py b/src/paperless_tika/parsers.py new file mode 100644 index 000000000..6b0f62ada --- /dev/null +++ b/src/paperless_tika/parsers.py @@ -0,0 +1,89 @@ +import os +import requests +import dateutil.parser + +from django.conf import settings + +from documents.parsers import DocumentParser, ParseError, \ + make_thumbnail_from_pdf +from tika import parser + + +class TikaDocumentParser(DocumentParser): + """ + This parser sends documents to a local tika server + """ + + logging_name = "paperless.parsing.tika" + + def get_thumbnail(self, document_path, mime_type, file_name=None): + if not self.archive_path: + self.archive_path = self.convert_to_pdf(document_path, file_name) + + return make_thumbnail_from_pdf( + self.archive_path, self.tempdir, self.logging_group) + + def extract_metadata(self, document_path, mime_type): + tika_server = settings.PAPERLESS_TIKA_ENDPOINT + try: + parsed = parser.from_file(document_path, tika_server) + except Exception as e: + self.log("warning", f"Error while fetching document metadata for " + f"{document_path}: {e}") + return [] + + return [ + { + "namespace": "", + "prefix": "", + "key": key, + "value": parsed['metadata'][key] + } for key in parsed['metadata'] + ] + + def parse(self, document_path, mime_type, file_name=None): + self.log("info", f"Sending {document_path} to Tika server") + tika_server = settings.PAPERLESS_TIKA_ENDPOINT + + try: + parsed = parser.from_file(document_path, tika_server) + except Exception as err: + raise ParseError( + f"Could not parse {document_path} with tika server at " + f"{tika_server}: {err}" + ) + + self.text = parsed["content"].strip() + + try: + self.date = dateutil.parser.isoparse( + parsed["metadata"]["Creation-Date"]) + except Exception as e: + self.log("warning", f"Unable to extract date for document " + f"{document_path}: {e}") + + self.archive_path = self.convert_to_pdf(document_path, file_name) + + def convert_to_pdf(self, document_path, file_name): + pdf_path = os.path.join(self.tempdir, "convert.pdf") + gotenberg_server = settings.PAPERLESS_TIKA_GOTENBERG_ENDPOINT + url = gotenberg_server + "/convert/office" + + self.log("info", f"Converting {document_path} to PDF as {pdf_path}") + files = {"files": (file_name or os.path.basename(document_path), + open(document_path, "rb"))} + headers = {} + + try: + response = requests.post(url, files=files, headers=headers) + response.raise_for_status() # ensure we notice bad responses + except Exception as err: + raise ParseError( + f"Error while converting document to PDF: {err}" + ) + + file = open(pdf_path, "wb") + file.write(response.content) + file.close() + + return pdf_path diff --git a/src/paperless_tika/signals.py b/src/paperless_tika/signals.py new file mode 100644 index 000000000..6b1698f2d --- /dev/null +++ b/src/paperless_tika/signals.py @@ -0,0 +1,24 @@ +def get_parser(*args, **kwargs): + from .parsers import TikaDocumentParser + + return TikaDocumentParser(*args, **kwargs) + + +def tika_consumer_declaration(sender, **kwargs): + return { + "parser": get_parser, + "weight": 10, + "mime_types": { + "application/msword": ".doc", + "application/vnd.openxmlformats-officedocument.wordprocessingml.document": ".docx", # NOQA: E501 + "application/vnd.ms-excel": ".xls", + "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet": ".xlsx", # NOQA: E501 + "application/vnd.ms-powerpoint": ".ppt", + "application/vnd.openxmlformats-officedocument.presentationml.presentation": ".pptx", # NOQA: E501 + "application/vnd.openxmlformats-officedocument.presentationml.slideshow": ".ppsx", # NOQA: E501 + "application/vnd.oasis.opendocument.presentation": ".odp", + "application/vnd.oasis.opendocument.spreadsheet": ".ods", + "application/vnd.oasis.opendocument.text": ".odt", + "text/rtf": ".rtf", + }, + } diff --git a/src/paperless_tika/tests/test_tika_parser.py b/src/paperless_tika/tests/test_tika_parser.py new file mode 100644 index 000000000..67563e5fe --- /dev/null +++ b/src/paperless_tika/tests/test_tika_parser.py @@ -0,0 +1,60 @@ +import datetime +import os +from pathlib import Path +from unittest import mock + +from django.test import TestCase +from requests import Response + +from paperless_tika.parsers import TikaDocumentParser + + +class TestTikaParser(TestCase): + + def setUp(self) -> None: + self.parser = TikaDocumentParser(logging_group=None) + + def tearDown(self) -> None: + self.parser.cleanup() + + @mock.patch("paperless_tika.parsers.parser.from_file") + @mock.patch("paperless_tika.parsers.requests.post") + def test_parse(self, post, from_file): + from_file.return_value = { + "content": "the content", + "metadata": { + "Creation-Date": "2020-11-21" + } + } + response = Response() + response._content = b"PDF document" + response.status_code = 200 + post.return_value = response + + file = os.path.join(self.parser.tempdir, "input.odt") + Path(file).touch() + self.parser.parse(file, "application/vnd.oasis.opendocument.text") + + self.assertEqual(self.parser.text, "the content") + self.assertIsNotNone(self.parser.archive_path) + with open(self.parser.archive_path, "rb") as f: + self.assertEqual(f.read(), b"PDF document") + + self.assertEqual(self.parser.date, datetime.datetime(2020, 11, 21)) + + @mock.patch("paperless_tika.parsers.parser.from_file") + def test_metadata(self, from_file): + from_file.return_value = { + "metadata": { + "Creation-Date": "2020-11-21", + "Some-key": "value" + } + } + + file = os.path.join(self.parser.tempdir, "input.odt") + Path(file).touch() + + metadata = self.parser.extract_metadata(file, "application/vnd.oasis.opendocument.text") + + self.assertTrue("Creation-Date" in [m['key'] for m in metadata]) + self.assertTrue("Some-key" in [m['key'] for m in metadata]) diff --git a/src/tox.ini b/src/tox.ini deleted file mode 100644 index 9a5bbff90..000000000 --- a/src/tox.ini +++ /dev/null @@ -1,21 +0,0 @@ -# Tox (http://tox.testrun.org/) is a tool for running tests -# in multiple virtualenvs. This configuration file will run the -# test suite on all supported python versions. To use it, "pip install tox" -# and then run "tox" from this directory. - -[tox] -skipsdist = True -envlist = py34, py35, py36, py37, pycodestyle, doc - -[testenv] -commands = pytest -deps = -r{toxinidir}/../requirements.txt - -[testenv:pycodestyle] -commands=pycodestyle -deps=pycodestyle - -[testenv:doc] -deps = - -r {toxinidir}/../requirements.txt -commands=sphinx-build -b html ../docs ../docs/_build -W