emailengine-app 1.14.8 → 2.61.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env.development +49 -0
- package/.env.example +82 -0
- package/.env.production +87 -0
- package/.eslintignore +1 -0
- package/.github/workflows/deploy.yml +104 -0
- package/.github/workflows/release.yaml +107 -0
- package/.github/workflows/test.yml +82 -0
- package/.ncurc.js +19 -5
- package/.prettierignore +44 -0
- package/CHANGELOG.md +1129 -0
- package/DOCKER_DEPLOYMENT.md +495 -0
- package/Dockerfile +53 -6
- package/Dockerfile-legacy +18 -0
- package/Fluid-Attacks-Results.csv +1 -0
- package/Gruntfile.js +46 -5
- package/LICENSE_EMAILENGINE.txt +110 -0
- package/README.md +72 -344
- package/app.json +40 -0
- package/bin/emailengine.js +671 -46
- package/config/default.toml +9 -11
- package/config/test.toml +45 -0
- package/copy-static-files.sh +34 -0
- package/data/google-crawlers.json +797 -0
- package/docker-compose.yml +103 -31
- package/encrypt.js +85 -10
- package/eslint.config.js +110 -0
- package/examples/auth-server.js +121 -69
- package/examples/grafana-dashboard.json +2375 -0
- package/install.sh +426 -0
- package/lib/account.js +2348 -124
- package/lib/add-trackers.js +119 -0
- package/lib/api-routes/bull-board-routes.js +60 -0
- package/lib/api-routes/chat-routes.js +519 -0
- package/lib/api-routes/template-routes.js +490 -0
- package/lib/append-list.js +9 -2
- package/lib/arf-detect.js +202 -0
- package/lib/autodetect-imap-settings.js +781 -0
- package/lib/bounce-detect.js +511 -50
- package/lib/capa.js +97 -0
- package/lib/consts.js +210 -1
- package/lib/db.js +227 -8
- package/lib/document-store.js +54 -0
- package/lib/email-client/base-client.js +3715 -0
- package/lib/email-client/gmail-client.js +2796 -0
- package/lib/email-client/imap/mailbox.js +3760 -0
- package/lib/email-client/imap/subconnection.js +269 -0
- package/lib/email-client/imap-client.js +2628 -0
- package/lib/email-client/outlook-client.js +3805 -0
- package/lib/encrypt.js +85 -14
- package/lib/es.js +784 -0
- package/lib/feature-flags.js +42 -0
- package/lib/gateway.js +271 -0
- package/lib/generate-text-preview.js +56 -0
- package/lib/get-raw-email.js +302 -42
- package/lib/get-secret.js +23 -67
- package/lib/headers-rewriter.js +33 -0
- package/lib/imapproxy/imap-core/index.js +4 -0
- package/lib/imapproxy/imap-core/lib/commands/append.js +187 -0
- package/lib/imapproxy/imap-core/lib/commands/authenticate-plain.js +145 -0
- package/lib/imapproxy/imap-core/lib/commands/capability.js +13 -0
- package/lib/imapproxy/imap-core/lib/commands/check.js +10 -0
- package/lib/imapproxy/imap-core/lib/commands/close.js +44 -0
- package/lib/imapproxy/imap-core/lib/commands/compress.js +102 -0
- package/lib/imapproxy/imap-core/lib/commands/copy.js +109 -0
- package/lib/imapproxy/imap-core/lib/commands/create.js +93 -0
- package/lib/imapproxy/imap-core/lib/commands/delete.js +84 -0
- package/lib/imapproxy/imap-core/lib/commands/enable.js +36 -0
- package/lib/imapproxy/imap-core/lib/commands/expunge.js +68 -0
- package/lib/imapproxy/imap-core/lib/commands/fetch.js +385 -0
- package/lib/imapproxy/imap-core/lib/commands/getquota.js +85 -0
- package/lib/imapproxy/imap-core/lib/commands/getquotaroot.js +111 -0
- package/lib/imapproxy/imap-core/lib/commands/id.js +111 -0
- package/lib/imapproxy/imap-core/lib/commands/idle.js +45 -0
- package/lib/imapproxy/imap-core/lib/commands/list.js +218 -0
- package/lib/imapproxy/imap-core/lib/commands/login.js +135 -0
- package/lib/imapproxy/imap-core/lib/commands/logout.js +26 -0
- package/lib/imapproxy/imap-core/lib/commands/lsub.js +102 -0
- package/lib/imapproxy/imap-core/lib/commands/move.js +106 -0
- package/lib/imapproxy/imap-core/lib/commands/namespace.js +14 -0
- package/lib/imapproxy/imap-core/lib/commands/noop.js +10 -0
- package/lib/imapproxy/imap-core/lib/commands/rename.js +102 -0
- package/lib/imapproxy/imap-core/lib/commands/search.js +306 -0
- package/lib/imapproxy/imap-core/lib/commands/select.js +248 -0
- package/lib/imapproxy/imap-core/lib/commands/setquota.js +24 -0
- package/lib/imapproxy/imap-core/lib/commands/starttls.js +100 -0
- package/lib/imapproxy/imap-core/lib/commands/status.js +149 -0
- package/lib/imapproxy/imap-core/lib/commands/store.js +208 -0
- package/lib/imapproxy/imap-core/lib/commands/subscribe.js +69 -0
- package/lib/imapproxy/imap-core/lib/commands/uid-expunge.js +71 -0
- package/lib/imapproxy/imap-core/lib/commands/uid-store.js +170 -0
- package/lib/imapproxy/imap-core/lib/commands/unselect.js +14 -0
- package/lib/imapproxy/imap-core/lib/commands/unsubscribe.js +69 -0
- package/lib/imapproxy/imap-core/lib/handler/README.md +146 -0
- package/lib/imapproxy/imap-core/lib/handler/imap-compile-stream.js +252 -0
- package/lib/imapproxy/imap-core/lib/handler/imap-compiler.js +134 -0
- package/lib/imapproxy/imap-core/lib/handler/imap-formal-syntax.js +147 -0
- package/lib/imapproxy/imap-core/lib/handler/imap-handler.js +11 -0
- package/lib/imapproxy/imap-core/lib/handler/imap-parser.js +678 -0
- package/lib/imapproxy/imap-core/lib/imap-command.js +381 -0
- package/lib/imapproxy/imap-core/lib/imap-composer.js +71 -0
- package/lib/imapproxy/imap-core/lib/imap-connection.js +929 -0
- package/lib/imapproxy/imap-core/lib/imap-server.js +426 -0
- package/lib/imapproxy/imap-core/lib/imap-stream.js +172 -0
- package/lib/imapproxy/imap-core/lib/imap-tools.js +789 -0
- package/lib/imapproxy/imap-core/lib/indexer/body-structure.js +295 -0
- package/lib/imapproxy/imap-core/lib/indexer/create-envelope.js +103 -0
- package/lib/imapproxy/imap-core/lib/indexer/indexer.js +904 -0
- package/lib/imapproxy/imap-core/lib/indexer/parse-mime-tree.js +340 -0
- package/lib/imapproxy/imap-core/lib/length-limiter.js +76 -0
- package/lib/imapproxy/imap-core/lib/parse-date.js +225 -0
- package/lib/imapproxy/imap-core/lib/search.js +330 -0
- package/lib/imapproxy/imap-core/lib/tls-options.js +69 -0
- package/lib/imapproxy/imap-core/memory-notifier.js +129 -0
- package/lib/imapproxy/imap-core/test/client.js +46 -0
- package/lib/imapproxy/imap-core/test/fixtures/append.eml +1196 -0
- package/lib/imapproxy/imap-core/test/fixtures/chunks.js +44 -0
- package/lib/imapproxy/imap-core/test/fixtures/fix1.eml +6 -0
- package/lib/imapproxy/imap-core/test/fixtures/fix2.eml +599 -0
- package/lib/imapproxy/imap-core/test/fixtures/fix3.eml +32 -0
- package/lib/imapproxy/imap-core/test/fixtures/fix4.eml +6 -0
- package/lib/imapproxy/imap-core/test/fixtures/mimetorture.eml +599 -0
- package/lib/imapproxy/imap-core/test/fixtures/mimetorture.js +2740 -0
- package/lib/imapproxy/imap-core/test/fixtures/mimetorture.json +1411 -0
- package/lib/imapproxy/imap-core/test/fixtures/mimetree.js +85 -0
- package/lib/imapproxy/imap-core/test/fixtures/nodemailer.eml +582 -0
- package/lib/imapproxy/imap-core/test/fixtures/ryan_finnie_mime_torture.eml +599 -0
- package/lib/imapproxy/imap-core/test/fixtures/simple.eml +42 -0
- package/lib/imapproxy/imap-core/test/fixtures/simple.json +164 -0
- package/lib/imapproxy/imap-core/test/imap-compile-stream-test.js +671 -0
- package/lib/imapproxy/imap-core/test/imap-compiler-test.js +272 -0
- package/lib/imapproxy/imap-core/test/imap-indexer-test.js +236 -0
- package/lib/imapproxy/imap-core/test/imap-parser-test.js +922 -0
- package/lib/imapproxy/imap-core/test/memory-notifier.js +129 -0
- package/lib/imapproxy/imap-core/test/prepare.sh +74 -0
- package/lib/imapproxy/imap-core/test/protocol-test.js +1756 -0
- package/lib/imapproxy/imap-core/test/search-test.js +1356 -0
- package/lib/imapproxy/imap-core/test/test-client.js +152 -0
- package/lib/imapproxy/imap-core/test/test-server.js +623 -0
- package/lib/imapproxy/imap-core/test/tools-test.js +22 -0
- package/lib/imapproxy/imap-server.js +577 -0
- package/lib/lists.js +92 -0
- package/lib/llm-pre-process.js +141 -0
- package/lib/logger.js +43 -4
- package/lib/lua/ee-get-idempotency.lua +74 -0
- package/lib/lua/ee-list-add.lua +34 -0
- package/lib/lua/ee-list-remove.lua +37 -0
- package/lib/lua/h-incrby-exists.lua +28 -0
- package/lib/lua/h-push.lua +32 -0
- package/lib/lua/h-set-bigger.lua +40 -0
- package/lib/lua/h-set-exists.lua +29 -0
- package/lib/lua/h-set-new.lua +29 -0
- package/lib/lua/h-update-bigger.lua +45 -0
- package/lib/lua/s-list-accounts.lua +64 -14
- package/lib/lua/z-expunge.lua +86 -10
- package/lib/lua/z-get-by-uid.lua +28 -5
- package/lib/lua/z-get-mailbox-id.lua +24 -2
- package/lib/lua/z-get-mailbox-path.lua +16 -0
- package/lib/lua/z-get.lua +27 -4
- package/lib/lua/z-set.lua +24 -2
- package/lib/metrics-collector.js +209 -0
- package/lib/oauth/gmail.js +663 -0
- package/lib/oauth/mail-ru.js +310 -0
- package/lib/oauth/outlook.js +541 -0
- package/lib/oauth/pubsub/google.js +247 -0
- package/lib/oauth2-apps.js +1420 -0
- package/lib/outbox.js +140 -0
- package/lib/payload-examples-documents.json +404 -0
- package/lib/payload-examples-webhooks.json +266 -0
- package/lib/pre-process.js +193 -0
- package/lib/rate-limit.js +32 -0
- package/lib/reconnection-manager.js +106 -0
- package/lib/redis-scan-delete.js +82 -0
- package/lib/redis-url.js +78 -0
- package/lib/rewrite-text-nodes.js +267 -0
- package/lib/routes-ui.js +10247 -0
- package/lib/schemas.js +1576 -187
- package/lib/settings.js +263 -12
- package/lib/sub-script.js +109 -0
- package/lib/templates.js +240 -0
- package/lib/threads.js +155 -0
- package/lib/tokens.js +353 -0
- package/lib/tools.js +1773 -41
- package/lib/translations.js +33 -0
- package/lib/webhooks.js +605 -0
- package/list-generate.js +96 -0
- package/package.json +129 -54
- package/render.yaml +44 -0
- package/sbom.json +1 -0
- package/scan.js +14 -2
- package/scripts/README.md +50 -0
- package/scripts/refresh-test-tokens.js +180 -0
- package/server.js +2902 -376
- package/setup-production.sh +201 -0
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.css +3 -3
- package/static/bootstrap-4.6.2-dist/css/bootstrap-grid.css.map +1 -0
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.min.css +3 -3
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.min.css.map +1 -1
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-reboot.css +3 -3
- package/static/bootstrap-4.6.2-dist/css/bootstrap-reboot.css.map +1 -0
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-reboot.min.css +3 -3
- package/static/bootstrap-4.6.2-dist/css/bootstrap-reboot.min.css.map +1 -0
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap.css +60 -26
- package/static/bootstrap-4.6.2-dist/css/bootstrap.css.map +1 -0
- package/static/bootstrap-4.6.2-dist/css/bootstrap.min.css +7 -0
- package/static/bootstrap-4.6.2-dist/css/bootstrap.min.css.map +1 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.js +7155 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.js.map +784 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.min.js +7 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.min.js.map +959 -0
- package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/js/bootstrap.js +792 -868
- package/static/bootstrap-4.6.2-dist/js/bootstrap.js.map +1 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.min.js +7 -0
- package/static/bootstrap-4.6.2-dist/js/bootstrap.min.js.map +1 -0
- package/static/css/app.css +146 -0
- package/static/css/arena.css +777 -0
- package/static/css/default.min.css +9 -0
- package/static/css/highlight.min.css +9 -0
- package/static/css/sb-admin-2.min.css +10 -0
- package/static/emailengine.ico +0 -0
- package/static/favicon/android-chrome-192x192.png +0 -0
- package/static/favicon/android-chrome-512x512.png +0 -0
- package/static/favicon/apple-touch-icon.png +0 -0
- package/static/favicon/favicon-16x16.png +0 -0
- package/static/favicon/favicon-32x32.png +0 -0
- package/static/favicon/favicon.ico +0 -0
- package/static/favicon.ico +0 -0
- package/static/fonts/nunito/OFL.txt +93 -0
- package/static/fonts/nunito/XRXV3I6Li01BKofINeaBTMnFcQ.woff2 +0 -0
- package/static/fonts/nunito-font.css +66 -0
- package/static/front/EmailEngine_logo_horiz.png +0 -0
- package/static/front/EmailEngine_logo_vert.png +0 -0
- package/static/front/index.html +57 -0
- package/static/front/logo.png +0 -0
- package/static/imap-capabilities-1.csv +71 -0
- package/static/index.html +30 -713
- package/static/js/ace/README.txt +1 -0
- package/static/js/ace/ace.js +23 -0
- package/static/js/ace/ext-language_tools.js +8 -0
- package/static/js/ace/ext-searchbox.js +8 -0
- package/static/js/ace/mode-handlebars.js +8 -0
- package/static/js/ace/mode-html.js +8 -0
- package/static/js/ace/mode-javascript.js +8 -0
- package/static/js/ace/mode-json.js +8 -0
- package/static/js/ace/mode-markdown.js +8 -0
- package/static/js/ace/snippets/javascript.js +8 -0
- package/static/js/ace/snippets/markdown.js +8 -0
- package/static/js/ace/theme-kuroir.js +8 -0
- package/static/js/ace/theme-xcode.js +8 -0
- package/static/js/ace/worker-html.js +1 -0
- package/static/js/ace/worker-javascript.js +1 -0
- package/static/js/ace/worker-json.js +1 -0
- package/static/js/app.js +526 -0
- package/static/js/bootstrap-autocomplete.min.js +1 -0
- package/static/js/clipboard.min.js +517 -0
- package/static/js/ee-client.js +1977 -0
- package/static/js/evaluation-worker.js +47 -0
- package/static/js/highlight.min.js +1173 -0
- package/static/js/jquery-3.6.0.min.js +2 -0
- package/static/js/sb-admin-2.min.js +7 -0
- package/static/licenses.html +6606 -50
- package/static/logo/EmailEngine_logo_horiz.png +0 -0
- package/static/logo/EmailEngine_logo_vert.png +0 -0
- package/static/logo.png +0 -0
- package/static/logo_transparent.png +0 -0
- package/static/logo_transparent_small.png +0 -0
- package/static/logo_wide.png +0 -0
- package/static/preview/header-template.png +0 -0
- package/static/preview/render.png +0 -0
- package/static/preview/translation.png +0 -0
- package/static/providers/google_dark.png +0 -0
- package/static/providers/google_dark_edited.png +0 -0
- package/static/providers/google_light.png +0 -0
- package/static/providers/ms_dark.svg +1 -0
- package/static/providers/ms_light.svg +1 -0
- package/static/robots.txt +4 -0
- package/static/undraw_profile.svg +38 -0
- package/static/vendor/fontawesome-free/LICENSE.txt +34 -0
- package/static/vendor/fontawesome-free/attribution.js +3 -0
- package/static/vendor/fontawesome-free/css/all.css +4619 -0
- package/static/vendor/fontawesome-free/css/all.min.css +5 -0
- package/static/vendor/fontawesome-free/css/brands.css +15 -0
- package/static/vendor/fontawesome-free/css/brands.min.css +5 -0
- package/static/vendor/fontawesome-free/css/fontawesome.css +4585 -0
- package/static/vendor/fontawesome-free/css/fontawesome.min.css +5 -0
- package/static/vendor/fontawesome-free/css/regular.css +15 -0
- package/static/vendor/fontawesome-free/css/regular.min.css +5 -0
- package/static/vendor/fontawesome-free/css/solid.css +16 -0
- package/static/vendor/fontawesome-free/css/solid.min.css +5 -0
- package/static/vendor/fontawesome-free/css/svg-with-js.css +371 -0
- package/static/vendor/fontawesome-free/css/svg-with-js.min.css +5 -0
- package/static/vendor/fontawesome-free/css/v4-shims.css +2172 -0
- package/static/vendor/fontawesome-free/css/v4-shims.min.css +5 -0
- package/static/vendor/fontawesome-free/js/all.js +4467 -0
- package/static/vendor/fontawesome-free/js/all.min.js +5 -0
- package/static/vendor/fontawesome-free/js/brands.js +586 -0
- package/static/vendor/fontawesome-free/js/brands.min.js +5 -0
- package/static/vendor/fontawesome-free/js/conflict-detection.js +998 -0
- package/static/vendor/fontawesome-free/js/conflict-detection.min.js +5 -0
- package/static/vendor/fontawesome-free/js/fontawesome.js +2483 -0
- package/static/vendor/fontawesome-free/js/fontawesome.min.js +5 -0
- package/static/vendor/fontawesome-free/js/regular.js +280 -0
- package/static/vendor/fontawesome-free/js/regular.min.js +5 -0
- package/static/vendor/fontawesome-free/js/solid.js +1130 -0
- package/static/vendor/fontawesome-free/js/solid.min.js +5 -0
- package/static/vendor/fontawesome-free/js/v4-shims.js +68 -0
- package/static/vendor/fontawesome-free/js/v4-shims.min.js +5 -0
- package/static/vendor/fontawesome-free/less/_animated.less +19 -0
- package/static/vendor/fontawesome-free/less/_bordered-pulled.less +16 -0
- package/static/vendor/fontawesome-free/less/_core.less +12 -0
- package/static/vendor/fontawesome-free/less/_fixed-width.less +6 -0
- package/static/vendor/fontawesome-free/less/_icons.less +1462 -0
- package/static/vendor/fontawesome-free/less/_larger.less +27 -0
- package/static/vendor/fontawesome-free/less/_list.less +18 -0
- package/static/vendor/fontawesome-free/less/_mixins.less +56 -0
- package/static/vendor/fontawesome-free/less/_rotated-flipped.less +24 -0
- package/static/vendor/fontawesome-free/less/_screen-reader.less +5 -0
- package/static/vendor/fontawesome-free/less/_shims.less +2066 -0
- package/static/vendor/fontawesome-free/less/_stacked.less +22 -0
- package/static/vendor/fontawesome-free/less/_variables.less +1474 -0
- package/static/vendor/fontawesome-free/less/brands.less +23 -0
- package/static/vendor/fontawesome-free/less/fontawesome.less +16 -0
- package/static/vendor/fontawesome-free/less/regular.less +23 -0
- package/static/vendor/fontawesome-free/less/solid.less +24 -0
- package/static/vendor/fontawesome-free/less/v4-shims.less +6 -0
- package/static/vendor/fontawesome-free/metadata/categories.yml +2572 -0
- package/static/vendor/fontawesome-free/metadata/icons.yml +21783 -0
- package/static/vendor/fontawesome-free/metadata/shims.yml +298 -0
- package/static/vendor/fontawesome-free/metadata/sponsors.yml +744 -0
- package/static/vendor/fontawesome-free/package.json +58 -0
- package/static/vendor/fontawesome-free/scss/_animated.scss +20 -0
- package/static/vendor/fontawesome-free/scss/_bordered-pulled.scss +20 -0
- package/static/vendor/fontawesome-free/scss/_core.scss +21 -0
- package/static/vendor/fontawesome-free/scss/_fixed-width.scss +6 -0
- package/static/vendor/fontawesome-free/scss/_icons.scss +1462 -0
- package/static/vendor/fontawesome-free/scss/_larger.scss +23 -0
- package/static/vendor/fontawesome-free/scss/_list.scss +18 -0
- package/static/vendor/fontawesome-free/scss/_mixins.scss +56 -0
- package/static/vendor/fontawesome-free/scss/_rotated-flipped.scss +24 -0
- package/static/vendor/fontawesome-free/scss/_screen-reader.scss +5 -0
- package/static/vendor/fontawesome-free/scss/_shims.scss +2066 -0
- package/static/vendor/fontawesome-free/scss/_stacked.scss +31 -0
- package/static/vendor/fontawesome-free/scss/_variables.scss +1479 -0
- package/static/vendor/fontawesome-free/scss/brands.scss +23 -0
- package/static/vendor/fontawesome-free/scss/fontawesome.scss +16 -0
- package/static/vendor/fontawesome-free/scss/regular.scss +23 -0
- package/static/vendor/fontawesome-free/scss/solid.scss +24 -0
- package/static/vendor/fontawesome-free/scss/v4-shims.scss +6 -0
- package/static/vendor/fontawesome-free/sprites/brands.svg +1381 -0
- package/static/vendor/fontawesome-free/sprites/regular.svg +463 -0
- package/static/vendor/fontawesome-free/sprites/solid.svg +3013 -0
- package/static/vendor/fontawesome-free/svgs/brands/500px.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/accessible-icon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/accusoft.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/acquisitions-incorporated.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/adn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/adversal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/affiliatetheme.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/airbnb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/algolia.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/alipay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/amazon-pay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/amazon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/amilia.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/android.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/angellist.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/angrycreative.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/angular.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/app-store-ios.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/app-store.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/apper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/apple-pay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/apple.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/artstation.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/asymmetrik.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/atlassian.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/audible.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/autoprefixer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/avianex.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/aviato.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/aws.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bandcamp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/battle-net.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/behance-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/behance.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bimobject.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bitbucket.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bitcoin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bity.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/black-tie.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/blackberry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/blogger-b.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/blogger.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bluetooth-b.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bluetooth.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/bootstrap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/btc.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/buffer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/buromobelexperte.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/buy-n-large.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/buysellads.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/canadian-maple-leaf.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-amazon-pay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-amex.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-apple-pay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-diners-club.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-discover.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-jcb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-mastercard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-paypal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-stripe.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cc-visa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/centercode.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/centos.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/chrome.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/chromecast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cloudflare.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cloudscale.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cloudsmith.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cloudversify.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/codepen.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/codiepie.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/confluence.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/connectdevelop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/contao.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cotton-bureau.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cpanel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-by.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc-eu.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc-jp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-pd-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-pd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-remix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sampling-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sampling.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-share.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons-zero.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/creative-commons.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/critical-role.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/css3-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/css3.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/cuttlefish.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/d-and-d-beyond.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/d-and-d.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dailymotion.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dashcube.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/deezer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/delicious.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/deploydog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/deskpro.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dev.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/deviantart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dhl.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/diaspora.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/digg.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/digital-ocean.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/discord.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/discourse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dochub.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/docker.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/draft2digital.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dribbble-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dribbble.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dropbox.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/drupal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/dyalog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/earlybirds.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ebay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/edge-legacy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/edge.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/elementor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ello.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ember.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/empire.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/envira.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/erlang.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ethereum.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/etsy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/evernote.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/expeditedssl.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/facebook-f.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/facebook-messenger.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/facebook-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/facebook.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fantasy-flight-games.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fedex.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fedora.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/figma.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/firefox-browser.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/firefox.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/first-order-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/first-order.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/firstdraft.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/flickr.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/flipboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fly.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/font-awesome-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/font-awesome-flag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/font-awesome-logo-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/font-awesome.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fonticons-fi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fonticons.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fort-awesome-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fort-awesome.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/forumbee.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/foursquare.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/free-code-camp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/freebsd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/fulcrum.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/galactic-republic.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/galactic-senate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/get-pocket.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gg-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gg.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/git-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/git-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/git.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/github-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/github-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/github.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gitkraken.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gitlab.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gitter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/glide-g.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/glide.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gofore.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/goodreads-g.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/goodreads.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-drive.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-pay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-play.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-plus-g.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-plus-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google-wallet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/google.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gratipay.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/grav.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gripfire.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/grunt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/guilded.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/gulp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hacker-news-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hacker-news.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hackerrank.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hips.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hire-a-helper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hive.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hooli.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hornbill.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hotjar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/houzz.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/html5.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/hubspot.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ideal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/imdb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/innosoft.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/instagram-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/instagram.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/instalod.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/intercom.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/internet-explorer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/invision.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ioxhost.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/itch-io.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/itunes-note.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/itunes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/java.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/jedi-order.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/jenkins.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/jira.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/joget.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/joomla.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/js-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/js.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/jsfiddle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/kaggle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/keybase.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/keycdn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/kickstarter-k.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/kickstarter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/korvue.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/laravel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/lastfm-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/lastfm.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/leanpub.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/less.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/line.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/linkedin-in.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/linkedin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/linode.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/linux.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/lyft.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/magento.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mailchimp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mandalorian.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/markdown.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mastodon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/maxcdn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mdb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/medapps.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/medium-m.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/medium.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/medrt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/meetup.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/megaport.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mendeley.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/microblog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/microsoft.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mixcloud.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mixer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/mizuni.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/modx.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/monero.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/napster.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/neos.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/nimblr.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/node-js.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/node.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/npm.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ns8.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/nutritionix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/octopus-deploy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/odnoklassniki-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/odnoklassniki.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/old-republic.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/opencart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/openid.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/opera.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/optin-monster.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/orcid.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/osi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/page4.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pagelines.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/palfed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/patreon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/paypal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/penny-arcade.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/perbyte.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/periscope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/phabricator.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/phoenix-framework.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/phoenix-squadron.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/php.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pied-piper-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pied-piper-hat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pied-piper-pp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pied-piper-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pied-piper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pinterest-p.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pinterest-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pinterest.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/playstation.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/product-hunt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/pushed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/python.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/qq.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/quinscape.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/quora.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/r-project.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/raspberry-pi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ravelry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/react.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/reacteurope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/readme.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/rebel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/red-river.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/reddit-alien.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/reddit-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/reddit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/redhat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/renren.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/replyd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/researchgate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/resolving.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/rev.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/rocketchat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/rockrms.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/rust.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/safari.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/salesforce.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/schlix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/scribd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/searchengin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sellcast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sellsy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/servicestack.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/shirtsinbulk.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/shopify.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/shopware.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/simplybuilt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sistrix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sith.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sketch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/skyatlas.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/skype.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/slack-hash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/slack.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/slideshare.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/snapchat-ghost.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/snapchat-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/snapchat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/soundcloud.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sourcetree.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/speakap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/speaker-deck.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/spotify.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/squarespace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stack-exchange.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stack-overflow.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stackpath.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/staylinked.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/steam-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/steam-symbol.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/steam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/sticker-mule.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/strava.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stripe-s.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stripe.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/studiovinari.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stumbleupon-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/stumbleupon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/superpowers.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/supple.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/suse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/swift.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/symfony.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/teamspeak.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/telegram-plane.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/telegram.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/tencent-weibo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/the-red-yeti.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/themeco.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/themeisle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/think-peaks.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/tiktok.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/trade-federation.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/trello.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/tripadvisor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/tumblr-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/tumblr.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/twitch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/twitter-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/twitter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/typo3.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/uber.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ubuntu.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/uikit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/umbraco.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/uncharted.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/uniregistry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/unity.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/unsplash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/untappd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ups.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/usb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/usps.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/ussunnah.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vaadin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/viacoin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/viadeo-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/viadeo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/viber.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vimeo-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vimeo-v.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vimeo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vine.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vk.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vnv.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/vuejs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/watchman-monitoring.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/waze.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/weebly.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/weibo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/weixin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/whatsapp-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/whatsapp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/whmcs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wikipedia-w.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/windows.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wix.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wizards-of-the-coast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wodu.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wolf-pack-battalion.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wordpress-simple.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wordpress.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wpbeginner.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wpexplorer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wpforms.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/wpressr.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/xbox.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/xing-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/xing.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/y-combinator.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yahoo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yammer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yandex-international.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yandex.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yarn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yelp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/yoast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/youtube-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/youtube.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/brands/zhihu.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/address-book.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/address-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/angry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/bell-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/bell.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/bookmark.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/building.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar-minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar-times.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/calendar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/caret-square-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/caret-square-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/caret-square-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/caret-square-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/chart-bar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/check-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/check-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/clipboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/clock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/clone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/closed-captioning.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/comment-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/comment-dots.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/comment.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/comments.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/compass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/copy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/copyright.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/credit-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/dizzy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/dot-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/edit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/envelope-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/envelope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/eye-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/eye.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-archive.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-audio.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-code.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-excel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-image.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-pdf.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-powerpoint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-video.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file-word.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/file.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/flag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/flushed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/folder-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/folder.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/font-awesome-logo-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/frown-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/frown.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/futbol.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/gem.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grimace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-beam-sweat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-hearts.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-squint-tears.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-stars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-tears.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-tongue-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-tongue-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-tongue.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/grin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-lizard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-paper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-peace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-point-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-point-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-point-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-point-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-pointer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-rock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-scissors.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hand-spock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/handshake.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hdd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/heart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hospital.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/hourglass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/id-badge.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/id-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/image.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/images.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/keyboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/kiss-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/kiss-wink-heart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/kiss.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/laugh-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/laugh-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/laugh-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/laugh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/lemon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/life-ring.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/lightbulb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/list-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/map.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/meh-blank.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/meh-rolling-eyes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/meh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/minus-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/money-bill-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/moon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/newspaper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/object-group.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/object-ungroup.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/paper-plane.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/pause-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/play-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/plus-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/question-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/registered.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/sad-cry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/sad-tear.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/save.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/share-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/smile-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/smile-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/smile.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/snowflake.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/star-half.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/star.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/sticky-note.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/stop-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/sun.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/surprise.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/thumbs-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/thumbs-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/times-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/tired.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/trash-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/user-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/user.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/window-close.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/window-maximize.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/window-minimize.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/regular/window-restore.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ad.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/address-book.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/address-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/adjust.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/air-freshener.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/align-center.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/align-justify.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/align-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/align-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/allergies.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ambulance.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/american-sign-language-interpreting.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/anchor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-double-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-double-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-double-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-double-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angle-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/angry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ankh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/apple-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/archive.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/archway.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrow-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrows-alt-h.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrows-alt-v.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/arrows-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/assistive-listening-systems.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/asterisk.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/at.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/atlas.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/atom.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/audio-description.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/award.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/baby-carriage.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/baby.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/backspace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/backward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bacon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bacteria.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bacterium.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bahai.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/balance-scale-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/balance-scale-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/balance-scale.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ban.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/band-aid.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/barcode.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/baseball-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/basketball-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bath.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/battery-empty.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/battery-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/battery-half.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/battery-quarter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/battery-three-quarters.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/beer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bell-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bell.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bezier-curve.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bible.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bicycle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/biking.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/binoculars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/biohazard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/birthday-cake.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/blender-phone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/blender.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/blind.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/blog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bold.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bolt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bomb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bong.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/book-dead.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/book-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/book-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/book-reader.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/book.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bookmark.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/border-all.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/border-none.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/border-style.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bowling-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/box-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/box-tissue.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/box.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/boxes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/braille.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/brain.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bread-slice.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/briefcase-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/briefcase.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/broadcast-tower.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/broom.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/brush.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bug.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/building.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bullhorn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bullseye.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/burn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bus-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/bus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/business-time.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calculator.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-day.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-times.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar-week.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/calendar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/camera-retro.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/camera.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/campground.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/candy-cane.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cannabis.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/capsules.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/car-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/car-battery.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/car-crash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/car-side.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/car.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caravan.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-square-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-square-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-square-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-square-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/caret-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/carrot.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cart-arrow-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cart-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cash-register.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/certificate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chair.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chalkboard-teacher.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chalkboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/charging-station.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chart-area.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chart-bar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chart-line.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chart-pie.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/check-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/check-double.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/check-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cheese.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-bishop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-board.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-king.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-knight.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-pawn.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-queen.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess-rook.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chess.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/chevron-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/child.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/church.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/circle-notch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/city.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clinic-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clipboard-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clipboard-list.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clipboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/clone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/closed-captioning.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-download-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-meatball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-moon-rain.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-moon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-rain.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-showers-heavy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-sun-rain.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-sun.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud-upload-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cloud.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cocktail.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/code-branch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/code.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/coffee.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cogs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/coins.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/columns.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment-dollar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment-dots.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comment.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comments-dollar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/comments.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/compact-disc.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/compass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/compress-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/compress-arrows-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/compress.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/concierge-bell.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cookie-bite.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cookie.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/copy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/copyright.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/couch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/credit-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crop-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cross.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crosshairs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crow.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crown.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/crutch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cube.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cubes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/cut.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/database.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/deaf.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/democrat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/desktop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dharmachakra.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/diagnoses.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-d20.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-d6.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-five.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-four.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-one.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-six.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-three.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice-two.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dice.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/digital-tachograph.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/directions.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/disease.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/divide.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dizzy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dna.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dollar-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dolly-flatbed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dolly.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/donate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/door-closed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/door-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dot-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dove.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/download.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/drafting-compass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dragon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/draw-polygon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/drum-steelpan.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/drum.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/drumstick-bite.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dumbbell.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dumpster-fire.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dumpster.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/dungeon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/edit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/egg.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/eject.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ellipsis-h.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ellipsis-v.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/envelope-open-text.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/envelope-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/envelope-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/envelope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/equals.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/eraser.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ethernet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/euro-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/exchange-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/exclamation-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/exclamation-triangle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/exclamation.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/expand-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/expand-arrows-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/expand.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/external-link-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/external-link-square-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/eye-dropper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/eye-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/eye.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fan.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fast-backward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fast-forward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/faucet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fax.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/feather-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/feather.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/female.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fighter-jet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-archive.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-audio.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-code.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-contract.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-csv.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-download.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-excel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-export.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-image.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-import.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-invoice-dollar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-invoice.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-medical-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-pdf.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-powerpoint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-prescription.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-signature.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-upload.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-video.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file-word.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/file.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fill-drip.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fill.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/film.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/filter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fingerprint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fire-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fire-extinguisher.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fire.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/first-aid.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fish.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/fist-raised.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/flag-checkered.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/flag-usa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/flag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/flask.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/flushed.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/folder-minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/folder-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/folder-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/folder.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/font-awesome-logo-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/font.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/football-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/forward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/frog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/frown-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/frown.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/funnel-dollar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/futbol.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gamepad.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gas-pump.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gavel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gem.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/genderless.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ghost.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gift.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gifts.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/glass-cheers.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/glass-martini-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/glass-martini.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/glass-whiskey.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/glasses.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/globe-africa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/globe-americas.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/globe-asia.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/globe-europe.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/globe.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/golf-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/gopuram.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/graduation-cap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/greater-than-equal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/greater-than.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grimace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-beam-sweat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-hearts.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-squint-tears.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-stars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-tears.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-tongue-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-tongue-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-tongue.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grip-horizontal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grip-lines-vertical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grip-lines.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/grip-vertical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/guitar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/h-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hamburger.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hammer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hamsa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-holding-heart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-holding-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-holding-usd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-holding-water.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-holding.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-lizard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-middle-finger.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-paper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-peace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-point-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-point-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-point-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-point-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-pointer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-rock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-scissors.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-sparkles.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hand-spock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hands-helping.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hands-wash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hands.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/handshake-alt-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/handshake-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/handshake.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hanukiah.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hard-hat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hashtag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hat-cowboy-side.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hat-cowboy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hat-wizard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hdd.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/head-side-cough-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/head-side-cough.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/head-side-mask.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/head-side-virus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/heading.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/headphones-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/headphones.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/headset.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/heart-broken.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/heart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/heartbeat.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/helicopter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/highlighter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hiking.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hippo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/history.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hockey-puck.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/holly-berry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/home.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/horse-head.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/horse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hospital-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hospital-symbol.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hospital-user.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hospital.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hot-tub.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hotdog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hotel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hourglass-end.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hourglass-half.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hourglass-start.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hourglass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/house-damage.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/house-user.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/hryvnia.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/i-cursor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ice-cream.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/icicles.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/icons.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/id-badge.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/id-card-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/id-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/igloo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/image.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/images.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/inbox.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/indent.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/industry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/infinity.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/info-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/info.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/italic.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/jedi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/joint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/journal-whills.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/kaaba.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/key.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/keyboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/khanda.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/kiss-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/kiss-wink-heart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/kiss.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/kiwi-bird.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/landmark.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/language.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laptop-code.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laptop-house.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laptop-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laptop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laugh-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laugh-squint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laugh-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/laugh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/layer-group.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/leaf.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lemon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/less-than-equal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/less-than.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/level-down-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/level-up-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/life-ring.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lightbulb.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/link.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lira-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/list-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/list-ol.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/list-ul.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/list.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/location-arrow.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lock-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/low-vision.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/luggage-cart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lungs-virus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/lungs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/magic.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/magnet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mail-bulk.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/male.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-marked-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-marked.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-marker-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-marker.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-pin.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map-signs.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/map.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/marker.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mars-double.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mars-stroke-h.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mars-stroke-v.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mars-stroke.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mask.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/medal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/medkit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/meh-blank.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/meh-rolling-eyes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/meh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/memory.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/menorah.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mercury.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/meteor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microchip.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microphone-alt-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microphone-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microphone-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microphone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/microscope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/minus-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/minus-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mitten.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mobile-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mobile.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-bill-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-bill-wave-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-bill-wave.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-bill.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-check-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/money-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/monument.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/moon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mortar-pestle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mosque.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/motorcycle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mountain.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mouse-pointer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mouse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/mug-hot.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/music.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/network-wired.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/neuter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/newspaper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/not-equal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/notes-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/object-group.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/object-ungroup.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/oil-can.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/om.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/otter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/outdent.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pager.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paint-brush.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paint-roller.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/palette.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pallet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paper-plane.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paperclip.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/parachute-box.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paragraph.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/parking.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/passport.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pastafarianism.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paste.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pause-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pause.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/paw.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/peace.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pen-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pen-fancy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pen-nib.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pen-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pen.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pencil-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pencil-ruler.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/people-arrows.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/people-carry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pepper-hot.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/percent.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/percentage.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/person-booth.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone-square-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone-volume.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/phone.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/photo-video.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/piggy-bank.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pills.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pizza-slice.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/place-of-worship.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plane-arrival.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plane-departure.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plane-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plane.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/play-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/play.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plug.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plus-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plus-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/podcast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/poll-h.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/poll.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/poo-storm.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/poo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/poop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/portrait.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pound-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/power-off.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pray.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/praying-hands.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/prescription-bottle-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/prescription-bottle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/prescription.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/print.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/procedures.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/project-diagram.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pump-medical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/pump-soap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/puzzle-piece.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/qrcode.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/question-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/question.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/quidditch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/quote-left.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/quote-right.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/quran.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/radiation-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/radiation.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/rainbow.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/random.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/receipt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/record-vinyl.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/recycle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/redo-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/redo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/registered.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/remove-format.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/reply-all.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/reply.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/republican.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/restroom.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/retweet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ribbon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ring.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/road.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/robot.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/rocket.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/route.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/rss-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/rss.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ruble-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ruler-combined.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ruler-horizontal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ruler-vertical.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ruler.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/running.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/rupee-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sad-cry.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sad-tear.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/satellite-dish.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/satellite.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/save.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/school.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/screwdriver.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/scroll.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sd-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/search-dollar.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/search-location.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/search-minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/search-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/search.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/seedling.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/server.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shapes.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/share-alt-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/share-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/share-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/share.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shekel-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shield-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shield-virus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ship.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shipping-fast.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shoe-prints.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shopping-bag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shopping-basket.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shopping-cart.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shower.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/shuttle-van.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sign-in-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sign-language.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sign-out-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/signal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/signature.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sim-card.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sitemap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/skating.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/skiing-nordic.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/skiing.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/skull-crossbones.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/skull.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sleigh.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sliders-h.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smile-beam.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smile-wink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smile.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smoking-ban.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/smoking.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sms.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/snowboarding.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/snowflake.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/snowman.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/snowplow.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/soap.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/socks.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/solar-panel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-down-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-up-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-amount-down-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-amount-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-amount-up-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-amount-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-down-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-up-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sort.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/spa.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/space-shuttle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/spell-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/spider.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/spinner.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/splotch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/spray-can.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/square-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/square-root-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stamp.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star-and-crescent.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star-half-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star-half.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star-of-david.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star-of-life.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/star.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/step-backward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/step-forward.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stethoscope.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sticky-note.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stop-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stop.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stopwatch-20.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stopwatch.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/store-alt-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/store-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/store-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/store.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stream.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/street-view.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/strikethrough.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/stroopwafel.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/subscript.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/subway.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/suitcase-rolling.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/suitcase.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sun.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/superscript.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/surprise.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/swatchbook.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/swimmer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/swimming-pool.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/synagogue.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sync-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/sync.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/syringe.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/table-tennis.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/table.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tablet-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tablet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tablets.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tachometer-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tags.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tape.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tasks.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/taxi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/teeth-open.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/teeth.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/temperature-high.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/temperature-low.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tenge.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/terminal.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/text-height.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/text-width.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/th-large.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/th-list.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/th.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/theater-masks.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer-empty.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer-full.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer-half.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer-quarter.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer-three-quarters.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thermometer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thumbs-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thumbs-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/thumbtack.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/ticket-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/times-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/times.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tint-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tint.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tired.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toggle-off.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toggle-on.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toilet-paper-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toilet-paper.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toilet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/toolbox.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tools.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tooth.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/torah.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/torii-gate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tractor.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trademark.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/traffic-light.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trailer.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/train.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tram.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/transgender-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/transgender.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trash-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trash-restore-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trash-restore.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tree.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/trophy.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/truck-loading.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/truck-monster.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/truck-moving.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/truck-pickup.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/truck.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tshirt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tty.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/tv.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/umbrella-beach.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/umbrella.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/underline.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/undo-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/undo.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/universal-access.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/university.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/unlink.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/unlock-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/unlock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/upload.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-alt-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-astronaut.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-check.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-circle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-clock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-cog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-edit.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-friends.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-graduate.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-injured.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-lock.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-md.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-minus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-ninja.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-nurse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-plus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-secret.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-shield.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-tag.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-tie.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user-times.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/user.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/users-cog.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/users-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/users.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/utensil-spoon.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/utensils.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vector-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/venus-double.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/venus-mars.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/venus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vest-patches.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vest.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vial.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vials.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/video-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/video.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vihara.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/virus-slash.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/virus.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/viruses.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/voicemail.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/volleyball-ball.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/volume-down.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/volume-mute.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/volume-off.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/volume-up.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vote-yea.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/vr-cardboard.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/walking.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wallet.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/warehouse.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/water.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wave-square.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/weight-hanging.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/weight.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wheelchair.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wifi.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wind.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/window-close.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/window-maximize.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/window-minimize.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/window-restore.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wine-bottle.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wine-glass-alt.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wine-glass.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/won-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/wrench.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/x-ray.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/yen-sign.svg +1 -0
- package/static/vendor/fontawesome-free/svgs/solid/yin-yang.svg +1 -0
- package/static/vendor/fontawesome-free/webfonts/fa-brands-400.eot +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-brands-400.svg +3717 -0
- package/static/vendor/fontawesome-free/webfonts/fa-brands-400.ttf +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-brands-400.woff +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-brands-400.woff2 +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-regular-400.eot +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-regular-400.svg +801 -0
- package/static/vendor/fontawesome-free/webfonts/fa-regular-400.ttf +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-regular-400.woff +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-regular-400.woff2 +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-solid-900.eot +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-solid-900.svg +5034 -0
- package/static/vendor/fontawesome-free/webfonts/fa-solid-900.ttf +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-solid-900.woff +0 -0
- package/static/vendor/fontawesome-free/webfonts/fa-solid-900.woff2 +0 -0
- package/static/vendor/handlebars/handlebars.min-v4.7.7.js +29 -0
- package/static/vendor/jquery/jquery.js +10881 -0
- package/static/vendor/jquery/jquery.min.js +2 -0
- package/static/vendor/jquery/jquery.min.map +1 -0
- package/static/vendor/jquery/jquery.slim.js +8782 -0
- package/static/vendor/jquery/jquery.slim.min.js +2 -0
- package/static/vendor/jquery/jquery.slim.min.map +1 -0
- package/static/vendor/jquery-easing/jquery.easing.min.js +1 -0
- package/systemd/emailengine.service +11 -3
- package/systemd/nginx-proxy.conf +1 -1
- package/test/api-test.js +899 -0
- package/test/autoreply-test.js +327 -0
- package/test/bounce-test.js +151 -0
- package/test/complaint-test.js +256 -0
- package/test/fixtures/autoreply/LICENSE +27 -0
- package/test/fixtures/autoreply/rfc3834-01.eml +23 -0
- package/test/fixtures/autoreply/rfc3834-02.eml +24 -0
- package/test/fixtures/autoreply/rfc3834-03.eml +26 -0
- package/test/fixtures/autoreply/rfc3834-04.eml +48 -0
- package/test/fixtures/autoreply/rfc3834-05.eml +19 -0
- package/test/fixtures/autoreply/rfc3834-06.eml +59 -0
- package/test/fixtures/bounces/163.eml +2521 -0
- package/test/fixtures/bounces/fastmail.eml +242 -0
- package/test/fixtures/bounces/gmail.eml +252 -0
- package/test/fixtures/bounces/hotmail.eml +655 -0
- package/test/fixtures/bounces/mailru.eml +121 -0
- package/test/fixtures/bounces/outlook.eml +1107 -0
- package/test/fixtures/bounces/postfix.eml +101 -0
- package/test/fixtures/bounces/rambler.eml +116 -0
- package/test/fixtures/bounces/workmail.eml +142 -0
- package/test/fixtures/bounces/yahoo.eml +139 -0
- package/test/fixtures/bounces/zoho.eml +83 -0
- package/test/fixtures/bounces/zonemta.eml +100 -0
- package/test/fixtures/complaints/LICENSE +27 -0
- package/test/fixtures/complaints/amazonses.eml +72 -0
- package/test/fixtures/complaints/dmarc.eml +59 -0
- package/test/fixtures/complaints/hotmail.eml +49 -0
- package/test/fixtures/complaints/optout.eml +40 -0
- package/test/fixtures/complaints/standard-arf.eml +68 -0
- package/test/fixtures/complaints/yahoo.eml +68 -0
- package/test/oauth2-apps-test.js +301 -0
- package/test/sendonly-test.js +160 -0
- package/test/test-config.js +34 -0
- package/test/webhooks-server.js +39 -0
- package/translations/README.md +16 -0
- package/translations/de.mo +0 -0
- package/translations/de.po +335 -0
- package/translations/en.mo +0 -0
- package/translations/en.po +310 -0
- package/translations/et.mo +0 -0
- package/translations/et.po +331 -0
- package/translations/fr.mo +0 -0
- package/translations/fr.po +333 -0
- package/translations/ja.mo +0 -0
- package/translations/ja.po +322 -0
- package/translations/locales.json +43 -0
- package/translations/messages.pot +323 -0
- package/translations/nl.mo +0 -0
- package/translations/nl.po +325 -0
- package/translations/pl.mo +0 -0
- package/translations/pl.po +328 -0
- package/update-info.sh +10 -0
- package/views/account/login.hbs +54 -0
- package/views/account/password.hbs +88 -0
- package/views/account/security.hbs +269 -0
- package/views/account/totp.hbs +30 -0
- package/views/accounts/account.hbs +1254 -0
- package/views/accounts/browse.hbs +102 -0
- package/views/accounts/edit.hbs +332 -0
- package/views/accounts/index.hbs +143 -0
- package/views/accounts/register/imap-server.hbs +507 -0
- package/views/accounts/register/imap.hbs +56 -0
- package/views/accounts/register/index.hbs +52 -0
- package/views/arena/index.hbs +4 -0
- package/views/config/ai.hbs +820 -0
- package/views/config/document-store/chat.hbs +362 -0
- package/views/config/document-store/index.hbs +231 -0
- package/views/config/document-store/mappings/index.hbs +116 -0
- package/views/config/document-store/mappings/new.hbs +95 -0
- package/views/config/document-store/pre-processing/index.hbs +459 -0
- package/views/config/imap-proxy.hbs +479 -0
- package/views/config/license.hbs +256 -0
- package/views/config/logging.hbs +61 -0
- package/views/config/network.hbs +334 -0
- package/views/config/oauth/app.hbs +309 -0
- package/views/config/oauth/edit.hbs +92 -0
- package/views/config/oauth/index.hbs +150 -0
- package/views/config/oauth/new.hbs +90 -0
- package/views/config/oauth.hbs +354 -0
- package/views/config/service-preview.hbs +14 -0
- package/views/config/service.hbs +718 -0
- package/views/config/smtp.hbs +525 -0
- package/views/config/webhooks.hbs +404 -0
- package/views/dashboard.hbs +315 -0
- package/views/error.hbs +6 -1
- package/views/gateways/edit.hbs +52 -0
- package/views/gateways/gateway.hbs +120 -0
- package/views/gateways/index.hbs +152 -0
- package/views/gateways/new.hbs +61 -0
- package/views/index.hbs +21 -0
- package/views/internals/index.hbs +170 -0
- package/views/internals/thread.hbs +143 -0
- package/views/layout/app.hbs +516 -0
- package/views/layout/login.hbs +78 -0
- package/views/layout/main.hbs +67 -0
- package/views/layout/public.hbs +90 -0
- package/views/legal.hbs +83 -0
- package/views/license.hbs +5 -0
- package/views/partials/accounts_header.hbs +6 -0
- package/views/partials/add_account_modal.hbs +60 -0
- package/views/partials/address_list.hbs +37 -0
- package/views/partials/alerts.hbs +33 -0
- package/views/partials/document_store_header.hbs +52 -0
- package/views/partials/editor_scope_info.hbs +10 -0
- package/views/partials/gateway_form.hbs +65 -0
- package/views/partials/gateway_js.hbs +90 -0
- package/views/partials/gateways_header.hbs +6 -0
- package/views/partials/oauth_config_header.hbs +10 -0
- package/views/partials/oauth_form.hbs +1204 -0
- package/views/partials/scope_info.hbs +134 -0
- package/views/partials/security_header.hbs +11 -0
- package/views/partials/side_menu.hbs +114 -0
- package/views/partials/template_form.hbs +121 -0
- package/views/partials/templates_header.hbs +6 -0
- package/views/partials/test_send.hbs +327 -0
- package/views/partials/tokens_header.hbs +6 -0
- package/views/partials/webhook_form.hbs +151 -0
- package/views/partials/webhooks_editor_functions.hbs +372 -0
- package/views/partials/webhooks_header.hbs +6 -0
- package/views/redirect.hbs +1 -0
- package/views/swagger/index.hbs +76 -0
- package/views/templates/edit.hbs +87 -0
- package/views/templates/index.hbs +208 -0
- package/views/templates/new.hbs +85 -0
- package/views/templates/template.hbs +423 -0
- package/views/tokens/index.hbs +207 -0
- package/views/tokens/new.hbs +230 -0
- package/views/unsubscribe.hbs +93 -0
- package/views/upgrade.hbs +56 -0
- package/views/webhooks/edit.hbs +31 -0
- package/views/webhooks/index.hbs +144 -0
- package/views/webhooks/new.hbs +27 -0
- package/views/webhooks/webhook.hbs +265 -0
- package/winconf.js +93 -0
- package/workers/api.js +8246 -1256
- package/workers/documents.js +1120 -0
- package/workers/imap-proxy.js +91 -0
- package/workers/imap.js +552 -161
- package/workers/smtp.js +355 -82
- package/workers/submit.js +319 -54
- package/workers/webhooks.js +542 -80
- package/.eslintrc +0 -14
- package/.github/FUNDING.yml +0 -4
- package/.github/ISSUE_TEMPLATE/bug_report.md +0 -38
- package/.github/ISSUE_TEMPLATE/feature_request.md +0 -20
- package/LICENSE.txt +0 -661
- package/examples/api.md +0 -137
- package/lib/connection.js +0 -1769
- package/lib/lua/z-push.lua +0 -14
- package/lib/mailbox.js +0 -1546
- package/license-report-config.json +0 -3
- package/licenses.txt +0 -37
- package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.css.map +0 -1
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.css.map +0 -1
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.min.css.map +0 -1
- package/static/bootstrap-4.6.0-dist/css/bootstrap.css.map +0 -1
- package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css +0 -7
- package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css.map +0 -1
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js +0 -7045
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js.map +0 -1
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js +0 -7
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js.map +0 -1
- package/static/bootstrap-4.6.0-dist/js/bootstrap.js.map +0 -1
- package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js +0 -7
- package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js.map +0 -1
- package/static/js/emailengine.js +0 -581
- package/workers/arena.js +0 -89
package/lib/tools.js
CHANGED
|
@@ -1,15 +1,97 @@
|
|
|
1
1
|
/* eslint no-bitwise: 0 */
|
|
2
2
|
|
|
3
|
+
// NB! This file is processed by gettext parser and can not use newer syntax like ?.
|
|
4
|
+
|
|
3
5
|
'use strict';
|
|
4
6
|
|
|
5
7
|
const msgpack = require('msgpack5')();
|
|
6
8
|
const enumMessageFlags = require('./enum-message-flags');
|
|
7
9
|
const Joi = require('joi');
|
|
8
|
-
const fetch = require('node-fetch');
|
|
9
10
|
const he = require('he');
|
|
10
11
|
const packageData = require('../package.json');
|
|
11
12
|
const settings = require('./settings');
|
|
12
|
-
const
|
|
13
|
+
const { brotliDecompress } = require('zlib');
|
|
14
|
+
const util = require('util');
|
|
15
|
+
const brotliDecompressAsync = util.promisify(brotliDecompress);
|
|
16
|
+
const { createPublicKey, createVerify, randomBytes, createHmac } = require('crypto');
|
|
17
|
+
const logger = require('./logger');
|
|
18
|
+
const Boom = require('@hapi/boom');
|
|
19
|
+
const { ImapFlow } = require('imapflow');
|
|
20
|
+
const nodemailer = require('nodemailer');
|
|
21
|
+
const { parentPort } = require('worker_threads');
|
|
22
|
+
const punycode = require('punycode.js');
|
|
23
|
+
const { PassThrough } = require('stream');
|
|
24
|
+
const socks = require('socks');
|
|
25
|
+
const os = require('os');
|
|
26
|
+
const Fs = require('fs');
|
|
27
|
+
const net = require('net');
|
|
28
|
+
const fs = Fs.promises;
|
|
29
|
+
const pathlib = require('path');
|
|
30
|
+
const { randomUUID: uuid } = require('crypto');
|
|
31
|
+
const mimeTypes = require('nodemailer/lib/mime-funcs/mime-types');
|
|
32
|
+
const { v3: murmurhash } = require('murmurhash');
|
|
33
|
+
const { compare: compareVersions, validate: validateVersion } = require('compare-versions');
|
|
34
|
+
const {
|
|
35
|
+
REDIS_PREFIX,
|
|
36
|
+
TLS_DEFAULTS,
|
|
37
|
+
URL_FETCH_TIMEOUT,
|
|
38
|
+
MAX_FORM_TTL,
|
|
39
|
+
FETCH_RETRY_INTERVAL,
|
|
40
|
+
FETCH_RETRY_EXPONENTIAL,
|
|
41
|
+
FETCH_RETRY_MAX,
|
|
42
|
+
URL_FETCH_RETRY_MAX
|
|
43
|
+
} = require('./consts');
|
|
44
|
+
const ipaddr = require('ipaddr.js');
|
|
45
|
+
const bullmqPackage = require('bullmq/package.json');
|
|
46
|
+
const { reverse: dnsReverse } = require('dns').promises;
|
|
47
|
+
const googleCrawlerRanges = require('../data/google-crawlers.json');
|
|
48
|
+
const { resolvePublicInterfaces } = require('pubface');
|
|
49
|
+
|
|
50
|
+
const { fetch: fetchCmd, Agent, RetryAgent } = require('undici');
|
|
51
|
+
|
|
52
|
+
const fetchAgent = new Agent({
|
|
53
|
+
strictContentLength: false,
|
|
54
|
+
connectTimeout: Math.min(30000, URL_FETCH_TIMEOUT), // up to 30s for connection
|
|
55
|
+
headersTimeout: URL_FETCH_TIMEOUT, // Full timeout (90s)
|
|
56
|
+
bodyTimeout: URL_FETCH_TIMEOUT // Full timeout (90s)
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
const retryAgent = new RetryAgent(fetchAgent, {
|
|
60
|
+
maxRetries: URL_FETCH_RETRY_MAX,
|
|
61
|
+
methods: ['GET', 'PUT', 'HEAD', 'OPTIONS', 'DELETE', 'POST'],
|
|
62
|
+
statusCodes: [429] // do not retry 5xx errors
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
const googleCrawlerMap = new Map();
|
|
66
|
+
for (let prefixEntry of googleCrawlerRanges.prefixes) {
|
|
67
|
+
for (let prefixKey of ['ipv6Prefix', 'ipv4Prefix']) {
|
|
68
|
+
if (prefixEntry[prefixKey]) {
|
|
69
|
+
// check if remote address is in range
|
|
70
|
+
let parsed = ipaddr.parseCIDR(prefixEntry[prefixKey]);
|
|
71
|
+
if (!googleCrawlerMap.has(prefixKey)) {
|
|
72
|
+
googleCrawlerMap.set(prefixKey, []);
|
|
73
|
+
}
|
|
74
|
+
googleCrawlerMap.get(prefixKey).push(parsed);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
class LRUCache extends Map {
|
|
80
|
+
constructor(maxSize = 1000) {
|
|
81
|
+
super();
|
|
82
|
+
this.maxSize = maxSize;
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
set(key, value) {
|
|
86
|
+
if (this.size >= this.maxSize) {
|
|
87
|
+
const firstKey = this.keys().next().value;
|
|
88
|
+
this.delete(firstKey);
|
|
89
|
+
}
|
|
90
|
+
super.set(key, value);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
const regexCache = new LRUCache(1000);
|
|
13
95
|
|
|
14
96
|
module.exports = {
|
|
15
97
|
/**
|
|
@@ -157,6 +239,8 @@ module.exports = {
|
|
|
157
239
|
|
|
158
240
|
// detect deleted flags
|
|
159
241
|
if (!keys || keys.includes('flags')) {
|
|
242
|
+
let hasFlagChanges = false;
|
|
243
|
+
|
|
160
244
|
for (let flag of storedMessageEntry.flags.values()) {
|
|
161
245
|
if (!messageData.flags.has(flag)) {
|
|
162
246
|
if (!changes.flags) {
|
|
@@ -167,6 +251,7 @@ module.exports = {
|
|
|
167
251
|
}
|
|
168
252
|
changes.flags.deleted.push(flag);
|
|
169
253
|
hasChanges = true;
|
|
254
|
+
hasFlagChanges = true;
|
|
170
255
|
}
|
|
171
256
|
}
|
|
172
257
|
|
|
@@ -181,11 +266,18 @@ module.exports = {
|
|
|
181
266
|
}
|
|
182
267
|
changes.flags.added.push(flag);
|
|
183
268
|
hasChanges = true;
|
|
269
|
+
hasFlagChanges = true;
|
|
184
270
|
}
|
|
185
271
|
}
|
|
272
|
+
|
|
273
|
+
if (hasFlagChanges) {
|
|
274
|
+
changes.flags.value = Array.from(messageData.flags);
|
|
275
|
+
}
|
|
186
276
|
}
|
|
187
277
|
|
|
188
278
|
if (!keys || keys.includes('labels')) {
|
|
279
|
+
let hasLabelChanges = false;
|
|
280
|
+
|
|
189
281
|
if (storedMessageEntry.labels || messageData.labels) {
|
|
190
282
|
if (storedMessageEntry.labels) {
|
|
191
283
|
// detect deleted labels
|
|
@@ -199,6 +291,7 @@ module.exports = {
|
|
|
199
291
|
}
|
|
200
292
|
changes.labels.deleted.push(flag);
|
|
201
293
|
hasChanges = true;
|
|
294
|
+
hasLabelChanges = true;
|
|
202
295
|
}
|
|
203
296
|
}
|
|
204
297
|
}
|
|
@@ -214,19 +307,29 @@ module.exports = {
|
|
|
214
307
|
}
|
|
215
308
|
changes.labels.added.push(flag);
|
|
216
309
|
hasChanges = true;
|
|
310
|
+
hasLabelChanges = true;
|
|
217
311
|
}
|
|
218
312
|
}
|
|
219
313
|
}
|
|
314
|
+
|
|
315
|
+
if (hasLabelChanges) {
|
|
316
|
+
changes.labels.value = messageData.labels ? Array.from(messageData.labels) : [];
|
|
317
|
+
}
|
|
220
318
|
}
|
|
221
319
|
}
|
|
222
320
|
|
|
223
321
|
return hasChanges ? changes : false;
|
|
224
322
|
},
|
|
225
323
|
|
|
226
|
-
normalizePath(path) {
|
|
324
|
+
normalizePath(path, separator) {
|
|
325
|
+
if (separator) {
|
|
326
|
+
return path.replace(new RegExp(`^INBOX($|${module.exports.escapeRegExp(separator)})`, 'i'), n => n.toUpperCase());
|
|
327
|
+
}
|
|
328
|
+
|
|
227
329
|
if (/^INBOX$/i.test(path)) {
|
|
228
330
|
return 'INBOX';
|
|
229
331
|
}
|
|
332
|
+
|
|
230
333
|
return path;
|
|
231
334
|
},
|
|
232
335
|
|
|
@@ -237,7 +340,9 @@ module.exports = {
|
|
|
237
340
|
throw err;
|
|
238
341
|
}
|
|
239
342
|
|
|
240
|
-
let headers = {
|
|
343
|
+
let headers = {
|
|
344
|
+
'User-Agent': `${packageData.name}/${packageData.version} (+${packageData.homepage})`
|
|
345
|
+
};
|
|
241
346
|
|
|
242
347
|
let parsed = new URL(authServer);
|
|
243
348
|
let username, password;
|
|
@@ -259,7 +364,7 @@ module.exports = {
|
|
|
259
364
|
parsed.searchParams.set('account', account);
|
|
260
365
|
parsed.searchParams.set('proto', proto);
|
|
261
366
|
|
|
262
|
-
let authResponse = await
|
|
367
|
+
let authResponse = await fetchCmd(parsed.toString(), { method: 'GET', headers, dispatcher: retryAgent });
|
|
263
368
|
if (!authResponse.ok) {
|
|
264
369
|
throw new Error(`Invalid response: ${authResponse.status} ${authResponse.statusText}`);
|
|
265
370
|
}
|
|
@@ -271,9 +376,16 @@ module.exports = {
|
|
|
271
376
|
|
|
272
377
|
const schema = Joi.object({
|
|
273
378
|
user: Joi.string().max(256).required(),
|
|
274
|
-
pass: Joi.string()
|
|
275
|
-
|
|
276
|
-
|
|
379
|
+
pass: Joi.string()
|
|
380
|
+
.allow('')
|
|
381
|
+
.max(256)
|
|
382
|
+
.when('accessToken', {
|
|
383
|
+
is: Joi.exist().not(false, null),
|
|
384
|
+
then: Joi.optional().valid(false, null),
|
|
385
|
+
otherwise: Joi.required()
|
|
386
|
+
}),
|
|
387
|
+
accessToken: Joi.string().max(4 * 4096)
|
|
388
|
+
});
|
|
277
389
|
|
|
278
390
|
const { error, value } = schema.validate(authData, {
|
|
279
391
|
abortEarly: false,
|
|
@@ -288,7 +400,23 @@ module.exports = {
|
|
|
288
400
|
return value;
|
|
289
401
|
},
|
|
290
402
|
|
|
291
|
-
|
|
403
|
+
getWorkerCount(processCount) {
|
|
404
|
+
if (/^\s*cpus\s*$/i.test(processCount)) {
|
|
405
|
+
processCount = os.cpus().length;
|
|
406
|
+
}
|
|
407
|
+
|
|
408
|
+
if (typeof processCount !== 'number' && !isNaN(processCount)) {
|
|
409
|
+
processCount = Number(processCount);
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
if (isNaN(processCount)) {
|
|
413
|
+
processCount = 0;
|
|
414
|
+
}
|
|
415
|
+
|
|
416
|
+
return processCount;
|
|
417
|
+
},
|
|
418
|
+
|
|
419
|
+
getDuration(val, opts = {}) {
|
|
292
420
|
val = (val || '').toString().replace(/^([\d.]+)\s*([smhdy][a-z]*)$/i, (o, num, m) => {
|
|
293
421
|
if (!num || isNaN(num)) {
|
|
294
422
|
return false;
|
|
@@ -299,7 +427,15 @@ module.exports = {
|
|
|
299
427
|
return num;
|
|
300
428
|
}
|
|
301
429
|
|
|
302
|
-
|
|
430
|
+
let unit = m.charAt(0).toLowerCase();
|
|
431
|
+
if (unit === 'm' && m.charAt(1).toLowerCase() === 's') {
|
|
432
|
+
unit = 'ms';
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
switch (unit) {
|
|
436
|
+
case 'ms':
|
|
437
|
+
// keep as is
|
|
438
|
+
break;
|
|
303
439
|
case 's':
|
|
304
440
|
num = num * 1000;
|
|
305
441
|
break;
|
|
@@ -330,10 +466,18 @@ module.exports = {
|
|
|
330
466
|
return val;
|
|
331
467
|
}
|
|
332
468
|
|
|
469
|
+
if (opts.seconds) {
|
|
470
|
+
return Math.ceil(Number(val) / 1000);
|
|
471
|
+
}
|
|
472
|
+
|
|
333
473
|
return Number(val);
|
|
334
474
|
},
|
|
335
475
|
|
|
336
476
|
getByteSize(val) {
|
|
477
|
+
if (typeof val === 'number') {
|
|
478
|
+
return val;
|
|
479
|
+
}
|
|
480
|
+
|
|
337
481
|
val = (val || '').toString().replace(/^([\d.]+)\s*([kMGTP])B?$/i, (o, num, m) => {
|
|
338
482
|
if (!num || isNaN(num)) {
|
|
339
483
|
return false;
|
|
@@ -372,6 +516,29 @@ module.exports = {
|
|
|
372
516
|
return Number(val);
|
|
373
517
|
},
|
|
374
518
|
|
|
519
|
+
formatByteSize(val) {
|
|
520
|
+
if (isNaN(val)) {
|
|
521
|
+
return val;
|
|
522
|
+
}
|
|
523
|
+
val = Number(val);
|
|
524
|
+
|
|
525
|
+
let types = new Set([
|
|
526
|
+
['PB', 1024 * 1024 * 1024 * 1024 * 1024],
|
|
527
|
+
['TB', 1024 * 1024 * 1024 * 1024],
|
|
528
|
+
['GB', 1024 * 1024 * 1024],
|
|
529
|
+
['MB', 1024 * 1024],
|
|
530
|
+
['kB', 1024]
|
|
531
|
+
]);
|
|
532
|
+
|
|
533
|
+
for (let [[type, nr]] of types.entries()) {
|
|
534
|
+
if (val % nr === 0) {
|
|
535
|
+
return `${Math.round(val / nr)}${type}`;
|
|
536
|
+
}
|
|
537
|
+
}
|
|
538
|
+
|
|
539
|
+
return val;
|
|
540
|
+
},
|
|
541
|
+
|
|
375
542
|
formatAccountListingResponse(entry) {
|
|
376
543
|
if (Array.isArray(entry)) {
|
|
377
544
|
let obj = {};
|
|
@@ -384,13 +551,11 @@ module.exports = {
|
|
|
384
551
|
return entry;
|
|
385
552
|
},
|
|
386
553
|
|
|
387
|
-
|
|
388
|
-
seconds = Number(seconds) || 3600;
|
|
389
|
-
|
|
554
|
+
getDateBuckets(seconds) {
|
|
390
555
|
let now = new Date();
|
|
391
556
|
let startTime = new Date(now.getTime() - seconds * 1000);
|
|
392
557
|
|
|
393
|
-
let
|
|
558
|
+
let bucketKeys = [];
|
|
394
559
|
|
|
395
560
|
// find out all the date buckets we need to check for
|
|
396
561
|
let endDateStr = `${now
|
|
@@ -403,7 +568,7 @@ module.exports = {
|
|
|
403
568
|
let startTimeStr = `${startTime
|
|
404
569
|
.toISOString()
|
|
405
570
|
// bucket includes 1 minute
|
|
406
|
-
.
|
|
571
|
+
.substring(0, 16)
|
|
407
572
|
.replace(/[^0-9]+/g, '')}`;
|
|
408
573
|
|
|
409
574
|
while (dateStr < endDateStr) {
|
|
@@ -411,12 +576,20 @@ module.exports = {
|
|
|
411
576
|
.toISOString()
|
|
412
577
|
.substr(0, 10)
|
|
413
578
|
.replace(/[^0-9]+/g, '')}`;
|
|
414
|
-
|
|
579
|
+
bucketKeys.push(dateStr);
|
|
415
580
|
hashTime = new Date(hashTime.getTime() + 24 * 3600 * 1000);
|
|
416
581
|
}
|
|
417
582
|
|
|
583
|
+
return { bucketKeys, startTimeStr };
|
|
584
|
+
},
|
|
585
|
+
|
|
586
|
+
async getCounterValues(redis, seconds) {
|
|
587
|
+
seconds = Number(seconds) || 3600;
|
|
588
|
+
|
|
589
|
+
const { bucketKeys: hashKeys, startTimeStr } = module.exports.getDateBuckets(seconds);
|
|
590
|
+
|
|
418
591
|
// list potential counter keys
|
|
419
|
-
let statUpdateKeys = await redis.smembers(
|
|
592
|
+
let statUpdateKeys = await redis.smembers(`${REDIS_PREFIX}stats:keys`);
|
|
420
593
|
|
|
421
594
|
let req = redis.multi();
|
|
422
595
|
let rIndex = [];
|
|
@@ -424,7 +597,7 @@ module.exports = {
|
|
|
424
597
|
for (let statUpdateKey of statUpdateKeys) {
|
|
425
598
|
// load stats for this key
|
|
426
599
|
for (let dateStr of hashKeys) {
|
|
427
|
-
req = req.hgetall(
|
|
600
|
+
req = req.hgetall(`${REDIS_PREFIX}stats:${statUpdateKey}:${dateStr}`);
|
|
428
601
|
rIndex.push(statUpdateKey);
|
|
429
602
|
}
|
|
430
603
|
}
|
|
@@ -454,36 +627,21 @@ module.exports = {
|
|
|
454
627
|
return counters;
|
|
455
628
|
},
|
|
456
629
|
|
|
457
|
-
getAuthSettings(authStr) {
|
|
458
|
-
if (!authStr || typeof authStr !== 'string') {
|
|
459
|
-
return { enabled: false };
|
|
460
|
-
}
|
|
461
|
-
let sep = authStr.indexOf(':');
|
|
462
|
-
let user = authStr.substr(0, sep).trim();
|
|
463
|
-
let pass = authStr.substr(sep + 1);
|
|
464
|
-
if (!user || !pass) {
|
|
465
|
-
return { enabled: false };
|
|
466
|
-
}
|
|
467
|
-
return { enabled: true, user, pass };
|
|
468
|
-
},
|
|
469
|
-
|
|
470
630
|
escapeRegExp(string) {
|
|
471
631
|
return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string
|
|
472
632
|
},
|
|
473
633
|
|
|
474
634
|
getRendezvousScore(key, shardId) {
|
|
475
|
-
|
|
476
|
-
return crypto
|
|
477
|
-
.createHash(alg)
|
|
478
|
-
.update((shardId || '').toString())
|
|
479
|
-
.update('\x00')
|
|
480
|
-
.update(key)
|
|
481
|
-
.digest()
|
|
482
|
-
.readUInt32LE(0, true);
|
|
635
|
+
return murmurhash(`${(shardId || '').toString()}\x00${key}`);
|
|
483
636
|
},
|
|
484
637
|
|
|
485
638
|
selectRendezvousNode(key, workers) {
|
|
486
|
-
return workers.map(worker => ({ worker, score: module.exports.getRendezvousScore(key, worker.threadId) })).sort((a, b) => b - a)[0].worker;
|
|
639
|
+
return workers.map(worker => ({ worker, score: module.exports.getRendezvousScore(key, worker.threadId) })).sort((a, b) => b.score - a.score)[0].worker;
|
|
640
|
+
},
|
|
641
|
+
|
|
642
|
+
selectRendezvousAddress(key, addresses) {
|
|
643
|
+
let scoredAddresses = addresses.map(address => ({ address, score: module.exports.getRendezvousScore(key, address) })).sort((a, b) => b.score - a.score);
|
|
644
|
+
return scoredAddresses[0].address;
|
|
487
645
|
},
|
|
488
646
|
|
|
489
647
|
getBoolean(value) {
|
|
@@ -508,5 +666,1579 @@ module.exports = {
|
|
|
508
666
|
}
|
|
509
667
|
|
|
510
668
|
return false; // ????
|
|
511
|
-
}
|
|
669
|
+
},
|
|
670
|
+
|
|
671
|
+
async checkLicense(license) {
|
|
672
|
+
let verifyKeys = [
|
|
673
|
+
// main-key-1
|
|
674
|
+
`-----BEGIN PUBLIC KEY-----
|
|
675
|
+
MFIwEAYHKoZIzj0CAQYFK4EEAAMDPgAEAaBbeChuyKlNp0MFi4nnRelWA6H/JHWr
|
|
676
|
+
ZdCXj2+HK4j0W0yzPN8VX0P7ox+1YgXNegBNchjVuu6xWSKE
|
|
677
|
+
-----END PUBLIC KEY-----`,
|
|
678
|
+
// trial-key-1
|
|
679
|
+
`-----BEGIN PUBLIC KEY-----
|
|
680
|
+
MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEV3QUiYsp13nD9suD1/ZkEXnuMoSg
|
|
681
|
+
8ZRXhDSmZQgW92fmNTsPs3tB6fQ3eAqO4JECE5Y2sI2EX/+Gm8JcErRhDg==
|
|
682
|
+
-----END PUBLIC KEY-----
|
|
683
|
+
`
|
|
684
|
+
].map(key => createPublicKey(key));
|
|
685
|
+
|
|
686
|
+
let encodedLicense;
|
|
687
|
+
if (!/\n/.test(license.toString().trim())) {
|
|
688
|
+
encodedLicense = Buffer.from(license.toString().trim(), 'base64url');
|
|
689
|
+
} else {
|
|
690
|
+
encodedLicense = Buffer.from(
|
|
691
|
+
license
|
|
692
|
+
.toString()
|
|
693
|
+
.split(/\r?\n/)
|
|
694
|
+
.map(line => line.trim())
|
|
695
|
+
.filter(line => line && !/[^a-z0-9+/=]/i.test(line))
|
|
696
|
+
.join(''),
|
|
697
|
+
'base64'
|
|
698
|
+
);
|
|
699
|
+
}
|
|
700
|
+
|
|
701
|
+
let signedLicense = await brotliDecompressAsync(encodedLicense);
|
|
702
|
+
|
|
703
|
+
let { l: licenseRaw, s: signature } = msgpack.decode(signedLicense);
|
|
704
|
+
|
|
705
|
+
let signedBy;
|
|
706
|
+
for (let verifyKey of verifyKeys) {
|
|
707
|
+
try {
|
|
708
|
+
const verify = createVerify('SHA256');
|
|
709
|
+
verify.write(licenseRaw);
|
|
710
|
+
verify.end();
|
|
711
|
+
const isSigned = verify.verify(verifyKey, signature);
|
|
712
|
+
if (isSigned) {
|
|
713
|
+
signedBy = verifyKey;
|
|
714
|
+
break;
|
|
715
|
+
}
|
|
716
|
+
} catch (err) {
|
|
717
|
+
// failed to verify, check another key if possible
|
|
718
|
+
}
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
if (!signedBy) {
|
|
722
|
+
let err = new Error('Failed to verify signature');
|
|
723
|
+
err.code = 'ELicenseValidation';
|
|
724
|
+
throw err;
|
|
725
|
+
}
|
|
726
|
+
|
|
727
|
+
const rawLicenseData = msgpackDecode(licenseRaw);
|
|
728
|
+
|
|
729
|
+
const licenseData = {
|
|
730
|
+
application: rawLicenseData.a,
|
|
731
|
+
key: rawLicenseData.k.toString('hex'),
|
|
732
|
+
licensedTo: rawLicenseData.n,
|
|
733
|
+
hostname: rawLicenseData.h,
|
|
734
|
+
created: new Date(rawLicenseData.c).toISOString(),
|
|
735
|
+
trial: rawLicenseData.t
|
|
736
|
+
};
|
|
737
|
+
|
|
738
|
+
if (rawLicenseData.l) {
|
|
739
|
+
licenseData.lt = true;
|
|
740
|
+
}
|
|
741
|
+
|
|
742
|
+
if (rawLicenseData.e) {
|
|
743
|
+
if (Date.now() > rawLicenseData.e) {
|
|
744
|
+
let err = new Error('License expired');
|
|
745
|
+
err.code = 'ELicenseExpired';
|
|
746
|
+
throw err;
|
|
747
|
+
}
|
|
748
|
+
|
|
749
|
+
licenseData.expires = new Date(rawLicenseData.e).toISOString();
|
|
750
|
+
}
|
|
751
|
+
|
|
752
|
+
if (rawLicenseData.t) {
|
|
753
|
+
licenseData.trial = true;
|
|
754
|
+
}
|
|
755
|
+
|
|
756
|
+
return licenseData;
|
|
757
|
+
},
|
|
758
|
+
|
|
759
|
+
async flash(redis, request, value) {
|
|
760
|
+
if (!request || !request.state || !request.state.crumb) {
|
|
761
|
+
return;
|
|
762
|
+
}
|
|
763
|
+
|
|
764
|
+
let rkey = `${REDIS_PREFIX}fl:${request.state.crumb}`;
|
|
765
|
+
try {
|
|
766
|
+
if (value) {
|
|
767
|
+
await redis.multi().lpush(rkey, JSON.stringify(value)).expire(rkey, 3600).exec();
|
|
768
|
+
} else {
|
|
769
|
+
let res = await redis.multi().lrange(rkey, 0, -1).del(rkey).exec();
|
|
770
|
+
if (!res || !res[0] || !res[0][1]) {
|
|
771
|
+
return false;
|
|
772
|
+
}
|
|
773
|
+
return res[0][1].map(entry => JSON.parse(entry));
|
|
774
|
+
}
|
|
775
|
+
} catch (err) {
|
|
776
|
+
// ignore
|
|
777
|
+
}
|
|
778
|
+
},
|
|
779
|
+
|
|
780
|
+
async failAction(request, h, err) {
|
|
781
|
+
try {
|
|
782
|
+
let details = (err.details || []).map(detail => ({ message: detail.message, key: detail.context.key }));
|
|
783
|
+
|
|
784
|
+
delete err._original;
|
|
785
|
+
|
|
786
|
+
logger.error({
|
|
787
|
+
msg: 'Request failed',
|
|
788
|
+
method: request.method,
|
|
789
|
+
route: request.route.path,
|
|
790
|
+
statusCode: request.response && request.response.statusCode,
|
|
791
|
+
err
|
|
792
|
+
});
|
|
793
|
+
|
|
794
|
+
let message = request.app.gt.gettext('Invalid input');
|
|
795
|
+
let error = Boom.boomify(new Error(message), { statusCode: 400 });
|
|
796
|
+
error.output.payload.fields = details;
|
|
797
|
+
|
|
798
|
+
throw error;
|
|
799
|
+
} catch (E) {
|
|
800
|
+
request.logger.error({ err: E });
|
|
801
|
+
throw E;
|
|
802
|
+
}
|
|
803
|
+
},
|
|
804
|
+
|
|
805
|
+
async verifyAccountInfo(redis, accountData, logger) {
|
|
806
|
+
let response = {};
|
|
807
|
+
|
|
808
|
+
let proxyUrl = await settings.get('proxyUrl');
|
|
809
|
+
let proxyEnabled = await settings.get('proxyEnabled');
|
|
810
|
+
|
|
811
|
+
let verifyPromises = [];
|
|
812
|
+
|
|
813
|
+
if (accountData.imap) {
|
|
814
|
+
verifyPromises.push(
|
|
815
|
+
(async () => {
|
|
816
|
+
try {
|
|
817
|
+
let imapConfig = Object.assign(
|
|
818
|
+
{
|
|
819
|
+
verifyOnly: true,
|
|
820
|
+
includeMailboxes: accountData.mailboxes,
|
|
821
|
+
|
|
822
|
+
greetingTimeout: 90 * 1000,
|
|
823
|
+
|
|
824
|
+
clientInfo: {
|
|
825
|
+
name: (await settings.get('imapClientName')) || packageData.name,
|
|
826
|
+
version: (await settings.get('imapClientVersion')) || packageData.version,
|
|
827
|
+
vendor: (await settings.get('imapClientVendor')) || (packageData.author && packageData.author.name) || packageData.author,
|
|
828
|
+
'support-url':
|
|
829
|
+
(await settings.get('imapClientSupportUrl')) || (packageData.bugs && packageData.bugs.url) || packageData.bugs
|
|
830
|
+
}
|
|
831
|
+
},
|
|
832
|
+
accountData.imap
|
|
833
|
+
);
|
|
834
|
+
|
|
835
|
+
// set up proxy if needed
|
|
836
|
+
if (accountData.proxy) {
|
|
837
|
+
imapConfig.proxy = accountData.proxy;
|
|
838
|
+
} else if (proxyEnabled && proxyUrl && !imapConfig.proxy) {
|
|
839
|
+
imapConfig.proxy = proxyUrl;
|
|
840
|
+
}
|
|
841
|
+
|
|
842
|
+
if (logger) {
|
|
843
|
+
imapConfig.logger = logger;
|
|
844
|
+
}
|
|
845
|
+
|
|
846
|
+
if (!imapConfig.tls) {
|
|
847
|
+
imapConfig.tls = {};
|
|
848
|
+
}
|
|
849
|
+
|
|
850
|
+
for (let key of Object.keys(TLS_DEFAULTS)) {
|
|
851
|
+
if (!(key in imapConfig.tls)) {
|
|
852
|
+
imapConfig.tls[key] = TLS_DEFAULTS[key];
|
|
853
|
+
}
|
|
854
|
+
}
|
|
855
|
+
|
|
856
|
+
const ignoreMailCertErrors = await settings.get('ignoreMailCertErrors');
|
|
857
|
+
if (ignoreMailCertErrors && imapConfig && imapConfig.tls && imapConfig.tls.rejectUnauthorized !== false) {
|
|
858
|
+
imapConfig.tls = imapConfig.tls || {};
|
|
859
|
+
imapConfig.tls.rejectUnauthorized = false;
|
|
860
|
+
}
|
|
861
|
+
|
|
862
|
+
let imapClient = new ImapFlow(imapConfig);
|
|
863
|
+
|
|
864
|
+
let mailboxes = await new Promise((resolve, reject) => {
|
|
865
|
+
imapClient.on('error', err => {
|
|
866
|
+
imapClient.close();
|
|
867
|
+
reject(err);
|
|
868
|
+
});
|
|
869
|
+
imapClient
|
|
870
|
+
.connect()
|
|
871
|
+
.then(() => resolve(imapClient._mailboxList))
|
|
872
|
+
.catch(reject);
|
|
873
|
+
});
|
|
874
|
+
|
|
875
|
+
response.imap = {
|
|
876
|
+
success: !!imapClient.authenticated
|
|
877
|
+
};
|
|
878
|
+
|
|
879
|
+
if (accountData.mailboxes && mailboxes && mailboxes.length) {
|
|
880
|
+
// format mailbox listing
|
|
881
|
+
let mailboxList = [];
|
|
882
|
+
for (let entry of mailboxes) {
|
|
883
|
+
let mailbox = {};
|
|
884
|
+
Object.keys(entry).forEach(key => {
|
|
885
|
+
if (['path', 'specialUse', 'name', 'listed', 'subscribed', 'delimiter', 'specialUseSource', 'noInferiors'].includes(key)) {
|
|
886
|
+
mailbox[key] = entry[key];
|
|
887
|
+
}
|
|
888
|
+
});
|
|
889
|
+
if (mailbox.delimiter && mailbox.path.indexOf(mailbox.delimiter) >= 0) {
|
|
890
|
+
mailbox.parentPath = mailbox.path.substr(0, mailbox.path.lastIndexOf(mailbox.delimiter));
|
|
891
|
+
}
|
|
892
|
+
mailboxList.push(mailbox);
|
|
893
|
+
}
|
|
894
|
+
response.mailboxes = mailboxList;
|
|
895
|
+
}
|
|
896
|
+
} catch (err) {
|
|
897
|
+
logger.error({ msg: 'Account verification failed', err });
|
|
898
|
+
response.imap = {
|
|
899
|
+
success: false,
|
|
900
|
+
error: err.message,
|
|
901
|
+
code: err.serverResponseCode || err.code,
|
|
902
|
+
statusCode: err.statusCode,
|
|
903
|
+
responseText: err.responseText
|
|
904
|
+
};
|
|
905
|
+
}
|
|
906
|
+
})()
|
|
907
|
+
);
|
|
908
|
+
}
|
|
909
|
+
|
|
910
|
+
if (accountData.smtp) {
|
|
911
|
+
verifyPromises.push(
|
|
912
|
+
(async () => {
|
|
913
|
+
try {
|
|
914
|
+
let { localAddress: address, name } = await module.exports.getLocalAddress(redis, 'smtp', 'test');
|
|
915
|
+
|
|
916
|
+
let smtpLogger = {};
|
|
917
|
+
let smtpConfig = Object.assign(
|
|
918
|
+
{
|
|
919
|
+
name,
|
|
920
|
+
localAddress: address,
|
|
921
|
+
transactionLog: true,
|
|
922
|
+
logger: smtpLogger
|
|
923
|
+
},
|
|
924
|
+
accountData.smtp
|
|
925
|
+
);
|
|
926
|
+
|
|
927
|
+
if (!smtpConfig.tls) {
|
|
928
|
+
smtpConfig.tls = {};
|
|
929
|
+
}
|
|
930
|
+
for (let key of Object.keys(TLS_DEFAULTS)) {
|
|
931
|
+
if (!(key in smtpConfig.tls)) {
|
|
932
|
+
smtpConfig.tls[key] = TLS_DEFAULTS[key];
|
|
933
|
+
}
|
|
934
|
+
}
|
|
935
|
+
|
|
936
|
+
for (let level of ['trace', 'debug', 'info', 'warn', 'error', 'fatal']) {
|
|
937
|
+
smtpLogger[level] = (data, message, ...args) => {
|
|
938
|
+
if (args && args.length) {
|
|
939
|
+
message = util.format(message, ...args);
|
|
940
|
+
}
|
|
941
|
+
data.msg = message;
|
|
942
|
+
data.sub = 'nodemailer';
|
|
943
|
+
if (typeof logger[level] === 'function') {
|
|
944
|
+
logger[level](data);
|
|
945
|
+
} else {
|
|
946
|
+
logger.debug(data);
|
|
947
|
+
}
|
|
948
|
+
};
|
|
949
|
+
}
|
|
950
|
+
|
|
951
|
+
// set up proxy if needed
|
|
952
|
+
if (accountData.proxy) {
|
|
953
|
+
smtpConfig.proxy = accountData.proxy;
|
|
954
|
+
} else if (proxyEnabled && proxyUrl && !smtpConfig.proxy) {
|
|
955
|
+
smtpConfig.proxy = proxyUrl;
|
|
956
|
+
}
|
|
957
|
+
|
|
958
|
+
if (accountData.smtpEhloName) {
|
|
959
|
+
smtpConfig.name = accountData.smtpEhloName;
|
|
960
|
+
}
|
|
961
|
+
|
|
962
|
+
smtpConfig.forceAuth = true;
|
|
963
|
+
|
|
964
|
+
const ignoreMailCertErrors = await settings.get('ignoreMailCertErrors');
|
|
965
|
+
if (ignoreMailCertErrors && smtpConfig && smtpConfig.tls && smtpConfig.tls.rejectUnauthorized !== false) {
|
|
966
|
+
smtpConfig.tls = smtpConfig.tls || {};
|
|
967
|
+
smtpConfig.tls.rejectUnauthorized = false;
|
|
968
|
+
}
|
|
969
|
+
|
|
970
|
+
if (smtpConfig.auth && smtpConfig.auth.accessToken) {
|
|
971
|
+
smtpConfig.auth.type = 'OAuth2';
|
|
972
|
+
}
|
|
973
|
+
|
|
974
|
+
const smtpClient = nodemailer.createTransport(smtpConfig);
|
|
975
|
+
smtpClient.set('proxy_socks_module', socks);
|
|
976
|
+
response.smtp = {
|
|
977
|
+
success: await smtpClient.verify()
|
|
978
|
+
};
|
|
979
|
+
} catch (err) {
|
|
980
|
+
response.smtp = {
|
|
981
|
+
success: false,
|
|
982
|
+
error: err.message,
|
|
983
|
+
code: err.code,
|
|
984
|
+
statusCode: err.statusCode,
|
|
985
|
+
responseText: err.response
|
|
986
|
+
};
|
|
987
|
+
}
|
|
988
|
+
})()
|
|
989
|
+
);
|
|
990
|
+
}
|
|
991
|
+
|
|
992
|
+
await Promise.all(verifyPromises);
|
|
993
|
+
|
|
994
|
+
return response;
|
|
995
|
+
},
|
|
996
|
+
|
|
997
|
+
async runPrechecks(redis) {
|
|
998
|
+
let keyName = Buffer.from([108, 105, 99, 101, 110, 115, 101]).toString();
|
|
999
|
+
let content = (await redis.hget(`${REDIS_PREFIX}settings`, keyName)) || '';
|
|
1000
|
+
let assertedBy;
|
|
1001
|
+
|
|
1002
|
+
if (content) {
|
|
1003
|
+
let rootList = [
|
|
1004
|
+
Buffer.from(
|
|
1005
|
+
'2d2d2d2d2d424547494e205055424c4943204b45592d2d2d2d2d0a4d464977454159484b6f5a497a6a3043415159464b34454541414d44506741454161426265436875794b6c4e70304d4669346e6e52656c574136482f4a4857720a5a6443586a322b484b346a305730797a504e3856583050376f782b315967584e6567424e63686a567575367857534b450a2d2d2d2d2d454e44205055424c4943204b45592d2d2d2d2d',
|
|
1006
|
+
'hex'
|
|
1007
|
+
).toString(),
|
|
1008
|
+
Buffer.from(
|
|
1009
|
+
'2d2d2d2d2d424547494e205055424c4943204b45592d2d2d2d2d0a4d466b77457759484b6f5a497a6a3043415159494b6f5a497a6a30444151634451674145563351556959737031336e4439737544312f5a6b45586e754d6f53670a385a52586844536d5a5167573932666d4e54735073337442366651336541714f344a45434535593273493245582f2b476d384a634572526844673d3d0a2d2d2d2d2d454e44205055424c4943204b45592d2d2d2d2d0a',
|
|
1010
|
+
'hex'
|
|
1011
|
+
).toString()
|
|
1012
|
+
].map(key => createPublicKey(key));
|
|
1013
|
+
|
|
1014
|
+
let encodedContent;
|
|
1015
|
+
if (!/\n/.test(content.toString().trim())) {
|
|
1016
|
+
encodedContent = Buffer.from(content.toString().trim(), 'base64url');
|
|
1017
|
+
} else {
|
|
1018
|
+
encodedContent = Buffer.from(
|
|
1019
|
+
content
|
|
1020
|
+
.toString()
|
|
1021
|
+
.split(/\r?\n/)
|
|
1022
|
+
.map(line => line.trim())
|
|
1023
|
+
.filter(line => line && !/[^a-z0-9+/=]/i.test(line))
|
|
1024
|
+
.join(''),
|
|
1025
|
+
'base64'
|
|
1026
|
+
);
|
|
1027
|
+
}
|
|
1028
|
+
|
|
1029
|
+
let signedContent = await brotliDecompressAsync(encodedContent);
|
|
1030
|
+
let { l: contentRaw, s: signature } = msgpack.decode(signedContent);
|
|
1031
|
+
for (let rootItem of rootList) {
|
|
1032
|
+
try {
|
|
1033
|
+
const verify = createVerify('SHA256');
|
|
1034
|
+
verify.write(contentRaw);
|
|
1035
|
+
verify.end();
|
|
1036
|
+
const isSigned = verify.verify(rootItem, signature);
|
|
1037
|
+
if (isSigned) {
|
|
1038
|
+
assertedBy = rootItem;
|
|
1039
|
+
break;
|
|
1040
|
+
}
|
|
1041
|
+
} catch (err) {
|
|
1042
|
+
// ignore
|
|
1043
|
+
}
|
|
1044
|
+
}
|
|
1045
|
+
}
|
|
1046
|
+
|
|
1047
|
+
if (!assertedBy) {
|
|
1048
|
+
let err = new Error(Buffer.from('4661696c656420746f20766572696679207369676e6174757265', 'hex').toString());
|
|
1049
|
+
err.code = Buffer.from('45436f6e74656e7456616c69646174696f6e', 'hex').toString();
|
|
1050
|
+
throw err;
|
|
1051
|
+
}
|
|
1052
|
+
},
|
|
1053
|
+
|
|
1054
|
+
isEmail(str) {
|
|
1055
|
+
const schema = Joi.object({
|
|
1056
|
+
email: Joi.string().email().required()
|
|
1057
|
+
});
|
|
1058
|
+
|
|
1059
|
+
const { error, value } = schema.validate(
|
|
1060
|
+
{ email: str },
|
|
1061
|
+
{
|
|
1062
|
+
abortEarly: false,
|
|
1063
|
+
stripUnknown: true,
|
|
1064
|
+
convert: true
|
|
1065
|
+
}
|
|
1066
|
+
);
|
|
1067
|
+
|
|
1068
|
+
if (error) {
|
|
1069
|
+
return false;
|
|
1070
|
+
}
|
|
1071
|
+
|
|
1072
|
+
return value.email;
|
|
1073
|
+
},
|
|
1074
|
+
|
|
1075
|
+
async emitChangeEvent(logger, account, type, key, payload) {
|
|
1076
|
+
try {
|
|
1077
|
+
parentPort.postMessage({
|
|
1078
|
+
cmd: 'change',
|
|
1079
|
+
account,
|
|
1080
|
+
type,
|
|
1081
|
+
key,
|
|
1082
|
+
payload: payload || null
|
|
1083
|
+
});
|
|
1084
|
+
} catch (err) {
|
|
1085
|
+
logger.error({ msg: 'Failed to post state change to parent', err });
|
|
1086
|
+
}
|
|
1087
|
+
},
|
|
1088
|
+
|
|
1089
|
+
getLogs(redis, account) {
|
|
1090
|
+
let logKey = `${REDIS_PREFIX}iam:${account}:g`;
|
|
1091
|
+
let passThrough = new PassThrough();
|
|
1092
|
+
|
|
1093
|
+
passThrough.headers = {
|
|
1094
|
+
'content-type': 'text/plain',
|
|
1095
|
+
'content-disposition': `attachment; filename=${JSON.stringify(`logs.${account}.txt`)}`
|
|
1096
|
+
};
|
|
1097
|
+
|
|
1098
|
+
redis
|
|
1099
|
+
.lrangeBuffer(logKey, 0, -1)
|
|
1100
|
+
.then(rows => {
|
|
1101
|
+
if (!rows || !Array.isArray(rows) || !rows.length) {
|
|
1102
|
+
return passThrough.end(`No logs found for ${account}\n`);
|
|
1103
|
+
}
|
|
1104
|
+
let processNext = () => {
|
|
1105
|
+
if (!rows.length) {
|
|
1106
|
+
return passThrough.end();
|
|
1107
|
+
}
|
|
1108
|
+
|
|
1109
|
+
let row = rows.shift();
|
|
1110
|
+
let entry;
|
|
1111
|
+
try {
|
|
1112
|
+
entry = msgpack.decode(row);
|
|
1113
|
+
} catch (err) {
|
|
1114
|
+
entry = { error: err.stack };
|
|
1115
|
+
}
|
|
1116
|
+
|
|
1117
|
+
if (entry) {
|
|
1118
|
+
if (!passThrough.write(JSON.stringify(entry) + '\n')) {
|
|
1119
|
+
return passThrough.once('drain', processNext);
|
|
1120
|
+
}
|
|
1121
|
+
}
|
|
1122
|
+
|
|
1123
|
+
setImmediate(processNext);
|
|
1124
|
+
};
|
|
1125
|
+
|
|
1126
|
+
processNext();
|
|
1127
|
+
})
|
|
1128
|
+
.catch(err => {
|
|
1129
|
+
passThrough.end(`\nFailed to process logs\n${err.stack}\n`);
|
|
1130
|
+
});
|
|
1131
|
+
|
|
1132
|
+
return passThrough;
|
|
1133
|
+
},
|
|
1134
|
+
|
|
1135
|
+
flattenObjectKeys(obj) {
|
|
1136
|
+
let result = {};
|
|
1137
|
+
let seen = new WeakSet();
|
|
1138
|
+
let walk = (prefix, c) => {
|
|
1139
|
+
if (!c || typeof c !== 'object') {
|
|
1140
|
+
return;
|
|
1141
|
+
}
|
|
1142
|
+
for (let key of Object.keys(c)) {
|
|
1143
|
+
let value = c[key];
|
|
1144
|
+
|
|
1145
|
+
if (value && typeof value === 'object' && !Array.isArray(value) && Object.prototype.toString.call(value) !== '[object Date]') {
|
|
1146
|
+
if (seen.has(value)) {
|
|
1147
|
+
// recursive
|
|
1148
|
+
continue;
|
|
1149
|
+
}
|
|
1150
|
+
seen.add(value);
|
|
1151
|
+
walk([].concat(prefix || []).concat(key), value);
|
|
1152
|
+
} else {
|
|
1153
|
+
let printKey = []
|
|
1154
|
+
.concat(prefix || [])
|
|
1155
|
+
.concat(key)
|
|
1156
|
+
.join('_');
|
|
1157
|
+
result[printKey] = value;
|
|
1158
|
+
}
|
|
1159
|
+
}
|
|
1160
|
+
};
|
|
1161
|
+
walk(false, obj);
|
|
1162
|
+
return result;
|
|
1163
|
+
},
|
|
1164
|
+
|
|
1165
|
+
async getRedisStats(redis) {
|
|
1166
|
+
let info = [await redis.info(), await redis.info('commandstats')].join('\n');
|
|
1167
|
+
|
|
1168
|
+
let formatValue = val => {
|
|
1169
|
+
if (/^[-+]?\d+(.\d+)?$/.test(val) && !/^0\d+/.test(val)) {
|
|
1170
|
+
return Number(val);
|
|
1171
|
+
}
|
|
1172
|
+
|
|
1173
|
+
switch (val) {
|
|
1174
|
+
case 'yes':
|
|
1175
|
+
return true;
|
|
1176
|
+
case 'no':
|
|
1177
|
+
return false;
|
|
1178
|
+
}
|
|
1179
|
+
|
|
1180
|
+
if (val.indexOf('=') >= 0) {
|
|
1181
|
+
// object with keys
|
|
1182
|
+
val = Object.fromEntries(val.split(',').map(e => [e.substr(0, e.indexOf('=')), formatValue(e.substr(e.indexOf('=') + 1))]));
|
|
1183
|
+
}
|
|
1184
|
+
|
|
1185
|
+
return val;
|
|
1186
|
+
};
|
|
1187
|
+
|
|
1188
|
+
let infoObj = Object.fromEntries(
|
|
1189
|
+
info
|
|
1190
|
+
.split(/\r?\n/)
|
|
1191
|
+
.filter(l => l.indexOf(':') >= 0)
|
|
1192
|
+
.map(l => [l.substr(0, l.indexOf(':')), formatValue(l.substr(l.indexOf(':') + 1))])
|
|
1193
|
+
);
|
|
1194
|
+
|
|
1195
|
+
let cmdstat_total = {
|
|
1196
|
+
calls: 0,
|
|
1197
|
+
usec: 0,
|
|
1198
|
+
rejected_calls: 15,
|
|
1199
|
+
failed_calls: 0
|
|
1200
|
+
};
|
|
1201
|
+
|
|
1202
|
+
Object.keys(infoObj).forEach(key => {
|
|
1203
|
+
if (key.indexOf('cmdstat_') === 0) {
|
|
1204
|
+
Object.keys(infoObj[key]).forEach(cKey => {
|
|
1205
|
+
if (typeof cmdstat_total[cKey] === 'number') {
|
|
1206
|
+
cmdstat_total[cKey] += infoObj[key][cKey];
|
|
1207
|
+
}
|
|
1208
|
+
});
|
|
1209
|
+
}
|
|
1210
|
+
});
|
|
1211
|
+
|
|
1212
|
+
infoObj.cmdstat_total = cmdstat_total;
|
|
1213
|
+
try {
|
|
1214
|
+
let slowlogLen = await redis.slowlog('len');
|
|
1215
|
+
infoObj.slowlog_length = formatValue(slowlogLen);
|
|
1216
|
+
} catch (err) {
|
|
1217
|
+
// not supported by Upstash
|
|
1218
|
+
infoObj.slowlog_length = 0;
|
|
1219
|
+
}
|
|
1220
|
+
|
|
1221
|
+
return infoObj;
|
|
1222
|
+
},
|
|
1223
|
+
|
|
1224
|
+
async getStats(redis, call, seconds) {
|
|
1225
|
+
const structuredMetrics = await call({ cmd: 'structuredMetrics' });
|
|
1226
|
+
|
|
1227
|
+
let counters = await module.exports.getCounterValues(redis, seconds);
|
|
1228
|
+
|
|
1229
|
+
let redisVersion;
|
|
1230
|
+
let softwareDetails;
|
|
1231
|
+
let redisSoftware;
|
|
1232
|
+
let redisCluster = false;
|
|
1233
|
+
|
|
1234
|
+
try {
|
|
1235
|
+
let redisInfo = await module.exports.getRedisStats(redis);
|
|
1236
|
+
|
|
1237
|
+
if (!redisInfo || typeof redisInfo.redis_version !== 'string') {
|
|
1238
|
+
throw new Error('Failed to fetch Redis INFO');
|
|
1239
|
+
}
|
|
1240
|
+
redisVersion = redisInfo.redis_version;
|
|
1241
|
+
|
|
1242
|
+
if (redisInfo.cluster_enabled && Number(redisInfo.cluster_enabled) && !isNaN(redisInfo.cluster_enabled) && redisInfo.cluster_enabled > 0) {
|
|
1243
|
+
redisCluster = true;
|
|
1244
|
+
}
|
|
1245
|
+
|
|
1246
|
+
// Detect Dragonfly
|
|
1247
|
+
if (typeof redisInfo.dragonfly_version === 'string') {
|
|
1248
|
+
softwareDetails = `Dragonfly v${redisInfo.dragonfly_version.replace(/^[^\d]*/, '')}`;
|
|
1249
|
+
redisSoftware = 'dragonfly';
|
|
1250
|
+
}
|
|
1251
|
+
|
|
1252
|
+
// Detect KeyDB
|
|
1253
|
+
if (typeof redisInfo.mvcc_depth === 'number') {
|
|
1254
|
+
softwareDetails = `KeyDB`;
|
|
1255
|
+
redisSoftware = 'keydb';
|
|
1256
|
+
}
|
|
1257
|
+
|
|
1258
|
+
// Detect Upstash
|
|
1259
|
+
if (typeof redisInfo.upstash_version === 'string') {
|
|
1260
|
+
softwareDetails = `Upstash Redis v${redisInfo.upstash_version.replace(/^[^\d]*/, '')}`;
|
|
1261
|
+
redisSoftware = 'upstash';
|
|
1262
|
+
}
|
|
1263
|
+
|
|
1264
|
+
// Detect Memurai
|
|
1265
|
+
if (typeof redisInfo.memurai_version === 'string') {
|
|
1266
|
+
softwareDetails = `${redisInfo.memurai_edition || 'Memurai'} v${redisInfo.memurai_version.replace(/^[^\d]*/, '')}`;
|
|
1267
|
+
redisSoftware = 'memurai';
|
|
1268
|
+
}
|
|
1269
|
+
|
|
1270
|
+
// Detect ElastiCache
|
|
1271
|
+
if (/Amazon ElastiCache/i.test(redisInfo.os)) {
|
|
1272
|
+
softwareDetails = `Amazon ElastiCache`;
|
|
1273
|
+
redisSoftware = 'elasticache';
|
|
1274
|
+
}
|
|
1275
|
+
|
|
1276
|
+
// Detect MemoryDB
|
|
1277
|
+
if (/Amazon MemoryDB/i.test(redisInfo.os)) {
|
|
1278
|
+
softwareDetails = `Amazon MemoryDB`;
|
|
1279
|
+
redisSoftware = 'memorydb';
|
|
1280
|
+
}
|
|
1281
|
+
} catch (err) {
|
|
1282
|
+
logger.error({ msg: 'Failed to get stats', err });
|
|
1283
|
+
redisVersion = err.message;
|
|
1284
|
+
}
|
|
1285
|
+
|
|
1286
|
+
let queues = {};
|
|
1287
|
+
for (let queue of ['notify', 'submit', 'documents']) {
|
|
1288
|
+
try {
|
|
1289
|
+
const [resActive, resDelayed, resWaiting, resPaused, resMeta] = await redis
|
|
1290
|
+
.multi()
|
|
1291
|
+
.llen(`${REDIS_PREFIX}bull:${queue}:active`)
|
|
1292
|
+
.zcard(`${REDIS_PREFIX}bull:${queue}:delayed`)
|
|
1293
|
+
.llen(`${REDIS_PREFIX}bull:${queue}:wait`)
|
|
1294
|
+
.llen(`${REDIS_PREFIX}bull:${queue}:paused`)
|
|
1295
|
+
.hget(`${REDIS_PREFIX}bull:${queue}:meta`, 'paused')
|
|
1296
|
+
.exec();
|
|
1297
|
+
if (resActive[0] || resDelayed[0] || resWaiting[0] || resPaused[0] || resMeta[0]) {
|
|
1298
|
+
// counting failed
|
|
1299
|
+
logger.error({ msg: 'Failed to count queue length', queue, active: resActive, delayed: resDelayed, waiting: resWaiting });
|
|
1300
|
+
return false;
|
|
1301
|
+
}
|
|
1302
|
+
queues[queue] = {
|
|
1303
|
+
active: Number(resActive[1]) || 0,
|
|
1304
|
+
delayed: Number(resDelayed[1]) || 0,
|
|
1305
|
+
waiting: Number(resWaiting[1]) || 0,
|
|
1306
|
+
paused: Number(resPaused[1]) || 0,
|
|
1307
|
+
isPaused: !!Number(resMeta[1]) || false
|
|
1308
|
+
};
|
|
1309
|
+
queues[queue].total = queues[queue].active + queues[queue].delayed + queues[queue].waiting + queues[queue].paused;
|
|
1310
|
+
} catch (err) {
|
|
1311
|
+
logger.error({ msg: 'Failed to count queue length', queue, err });
|
|
1312
|
+
}
|
|
1313
|
+
}
|
|
1314
|
+
|
|
1315
|
+
let stats = Object.assign(
|
|
1316
|
+
{
|
|
1317
|
+
version: packageData.version,
|
|
1318
|
+
license: packageData.license,
|
|
1319
|
+
accounts: await redis.scard(`${REDIS_PREFIX}ia:accounts`),
|
|
1320
|
+
node: process.versions.node,
|
|
1321
|
+
redis: `${redisVersion}${softwareDetails ? ` (${softwareDetails})` : ''}`,
|
|
1322
|
+
redisSoftware,
|
|
1323
|
+
redisCluster,
|
|
1324
|
+
imapflow: ImapFlow.version || 'please upgrade',
|
|
1325
|
+
bullmq: bullmqPackage.version,
|
|
1326
|
+
arch: process.arch,
|
|
1327
|
+
counters,
|
|
1328
|
+
queues
|
|
1329
|
+
},
|
|
1330
|
+
structuredMetrics
|
|
1331
|
+
);
|
|
1332
|
+
|
|
1333
|
+
try {
|
|
1334
|
+
// version info file might not exist
|
|
1335
|
+
let versionFile = await fs.readFile(pathlib.join(__dirname, '..', 'version-info.json'), 'utf-8');
|
|
1336
|
+
if (versionFile) {
|
|
1337
|
+
let versionData = JSON.parse(versionFile);
|
|
1338
|
+
stats.build = versionData;
|
|
1339
|
+
}
|
|
1340
|
+
} catch (err) {
|
|
1341
|
+
// ignore
|
|
1342
|
+
}
|
|
1343
|
+
|
|
1344
|
+
return stats;
|
|
1345
|
+
},
|
|
1346
|
+
|
|
1347
|
+
async fetchReleaseInfo() {
|
|
1348
|
+
const releaseUrl = `https://api.github.com/repos/postalsys/emailengine/releases/latest`;
|
|
1349
|
+
|
|
1350
|
+
let headers = {
|
|
1351
|
+
'User-Agent': `${packageData.name}/${packageData.version} (+${packageData.homepage})`
|
|
1352
|
+
};
|
|
1353
|
+
|
|
1354
|
+
let releaseResponse = await fetchCmd(releaseUrl, { method: 'GET', headers, dispatcher: retryAgent });
|
|
1355
|
+
if (!releaseResponse.ok) {
|
|
1356
|
+
let err = new Error(`Failed loading release data`);
|
|
1357
|
+
err.response = {
|
|
1358
|
+
status: releaseResponse.status
|
|
1359
|
+
};
|
|
1360
|
+
try {
|
|
1361
|
+
err.response.data = await releaseResponse.json();
|
|
1362
|
+
} catch (err) {
|
|
1363
|
+
//ignore
|
|
1364
|
+
}
|
|
1365
|
+
throw err;
|
|
1366
|
+
}
|
|
1367
|
+
|
|
1368
|
+
let releaseData = await releaseResponse.json();
|
|
1369
|
+
if (!releaseData) {
|
|
1370
|
+
throw new Error('Failed loading release data');
|
|
1371
|
+
}
|
|
1372
|
+
|
|
1373
|
+
return releaseData;
|
|
1374
|
+
},
|
|
1375
|
+
|
|
1376
|
+
async checkForUpgrade() {
|
|
1377
|
+
let releaseData;
|
|
1378
|
+
try {
|
|
1379
|
+
releaseData = await module.exports.fetchReleaseInfo();
|
|
1380
|
+
} catch (err) {
|
|
1381
|
+
return { current: packageData.version, available: false, canUpgrade: false };
|
|
1382
|
+
}
|
|
1383
|
+
|
|
1384
|
+
let releaseVersion = (releaseData.tag_name || '').toString().replace(/^v/, '');
|
|
1385
|
+
if (!validateVersion(releaseVersion)) {
|
|
1386
|
+
return { current: packageData.version, available: false, canUpgrade: false };
|
|
1387
|
+
}
|
|
1388
|
+
|
|
1389
|
+
if (compareVersions(releaseVersion, packageData.version, '>')) {
|
|
1390
|
+
return { current: packageData.version, available: releaseVersion, canUpgrade: true };
|
|
1391
|
+
}
|
|
1392
|
+
|
|
1393
|
+
return { current: packageData.version, available: releaseVersion, canUpgrade: false };
|
|
1394
|
+
},
|
|
1395
|
+
|
|
1396
|
+
formatPartialSecretKey(secretKey) {
|
|
1397
|
+
return secretKey.replace(/^(.{8}).*$/, `$1... (${secretKey.length}B)`);
|
|
1398
|
+
},
|
|
1399
|
+
|
|
1400
|
+
validUidValidity(value) {
|
|
1401
|
+
if (typeof value === 'bigint' || typeof value === 'number') {
|
|
1402
|
+
return true;
|
|
1403
|
+
}
|
|
1404
|
+
|
|
1405
|
+
if (isNaN(value)) {
|
|
1406
|
+
return false;
|
|
1407
|
+
}
|
|
1408
|
+
|
|
1409
|
+
return !!value;
|
|
1410
|
+
},
|
|
1411
|
+
|
|
1412
|
+
mergeObjects(destination, source) {
|
|
1413
|
+
for (let propKey of Object.keys(source)) {
|
|
1414
|
+
let sourceVal = source[propKey];
|
|
1415
|
+
|
|
1416
|
+
if (typeof destination[propKey] === 'undefined') {
|
|
1417
|
+
destination[propKey] = sourceVal;
|
|
1418
|
+
continue;
|
|
1419
|
+
}
|
|
1420
|
+
|
|
1421
|
+
if (destination[propKey] && sourceVal && typeof destination[propKey] === 'object' && typeof sourceVal === 'object') {
|
|
1422
|
+
module.exports.mergeObjects(destination[propKey], sourceVal);
|
|
1423
|
+
continue;
|
|
1424
|
+
}
|
|
1425
|
+
|
|
1426
|
+
// new value set, do not change it
|
|
1427
|
+
}
|
|
1428
|
+
},
|
|
1429
|
+
|
|
1430
|
+
async getServiceSecret() {
|
|
1431
|
+
let serviceSecret = await settings.get('serviceSecret');
|
|
1432
|
+
if (!serviceSecret) {
|
|
1433
|
+
serviceSecret = randomBytes(16).toString('hex');
|
|
1434
|
+
await settings.set('serviceSecret', serviceSecret);
|
|
1435
|
+
}
|
|
1436
|
+
return serviceSecret;
|
|
1437
|
+
},
|
|
1438
|
+
|
|
1439
|
+
async getSignedFormData(opts) {
|
|
1440
|
+
const serviceSecret = await module.exports.getServiceSecret();
|
|
1441
|
+
return module.exports.getSignedFormDataSync(serviceSecret, opts);
|
|
1442
|
+
},
|
|
1443
|
+
|
|
1444
|
+
filterEmptyObjectValues(obj) {
|
|
1445
|
+
let res = {};
|
|
1446
|
+
for (let key of Object.keys(obj)) {
|
|
1447
|
+
if (obj[key]) {
|
|
1448
|
+
res[key] = obj[key];
|
|
1449
|
+
}
|
|
1450
|
+
}
|
|
1451
|
+
return res;
|
|
1452
|
+
},
|
|
1453
|
+
|
|
1454
|
+
getSignedFormDataSync(serviceSecret, opts, asIs) {
|
|
1455
|
+
opts = opts || {};
|
|
1456
|
+
|
|
1457
|
+
let data = Buffer.from(
|
|
1458
|
+
JSON.stringify(
|
|
1459
|
+
asIs
|
|
1460
|
+
? opts
|
|
1461
|
+
: module.exports.filterEmptyObjectValues({
|
|
1462
|
+
account: opts.account,
|
|
1463
|
+
name: opts.name,
|
|
1464
|
+
email: opts.email,
|
|
1465
|
+
syncFrom: (opts.syncFrom && opts.syncFrom.toISOString()) || null,
|
|
1466
|
+
notifyFrom: (opts.notifyFrom && opts.notifyFrom.toISOString()) || null,
|
|
1467
|
+
subconnections: opts.subconnections && opts.subconnections.length ? opts.subconnections : null,
|
|
1468
|
+
redirectUrl: opts.redirectUrl,
|
|
1469
|
+
delegated: opts.delegated,
|
|
1470
|
+
path: opts.path,
|
|
1471
|
+
n: opts.n,
|
|
1472
|
+
t: opts.t
|
|
1473
|
+
})
|
|
1474
|
+
)
|
|
1475
|
+
);
|
|
1476
|
+
|
|
1477
|
+
let signature;
|
|
1478
|
+
|
|
1479
|
+
let hmac = createHmac('sha256', serviceSecret);
|
|
1480
|
+
hmac.update(data);
|
|
1481
|
+
signature = hmac.digest('base64url');
|
|
1482
|
+
|
|
1483
|
+
return { data: data.toString('base64url'), signature };
|
|
1484
|
+
},
|
|
1485
|
+
|
|
1486
|
+
async parseSignedFormData(redis, payload, gt) {
|
|
1487
|
+
let data = Buffer.from(payload.data, 'base64url').toString();
|
|
1488
|
+
let serviceSecret = await settings.get('serviceSecret');
|
|
1489
|
+
if (serviceSecret) {
|
|
1490
|
+
let hmac = createHmac('sha256', serviceSecret);
|
|
1491
|
+
hmac.update(data);
|
|
1492
|
+
if (hmac.digest('base64url') !== payload.sig) {
|
|
1493
|
+
let error = Boom.boomify(new Error(gt.gettext('Signature validation failed')), { statusCode: 403 });
|
|
1494
|
+
throw error;
|
|
1495
|
+
}
|
|
1496
|
+
}
|
|
1497
|
+
|
|
1498
|
+
data = JSON.parse(data);
|
|
1499
|
+
|
|
1500
|
+
if (data.n && data.t) {
|
|
1501
|
+
if (data.t < Date.now() - (MAX_FORM_TTL - 60 * 1000)) {
|
|
1502
|
+
let error = Boom.boomify(new Error(gt.gettext('Invalid or expired account setup URL')), { statusCode: 403 });
|
|
1503
|
+
throw error;
|
|
1504
|
+
}
|
|
1505
|
+
const nonceSeen = await redis.exists(`${REDIS_PREFIX}account:form:${data.n}`);
|
|
1506
|
+
if (nonceSeen) {
|
|
1507
|
+
let error = Boom.boomify(new Error(gt.gettext('Invalid or expired account setup URL')), { statusCode: 403 });
|
|
1508
|
+
throw error;
|
|
1509
|
+
}
|
|
1510
|
+
}
|
|
1511
|
+
|
|
1512
|
+
return data;
|
|
1513
|
+
},
|
|
1514
|
+
|
|
1515
|
+
async setLicense(licenseData, licenseFile) {
|
|
1516
|
+
await settings.setLicense(licenseData, licenseFile);
|
|
1517
|
+
},
|
|
1518
|
+
|
|
1519
|
+
async download(stream) {
|
|
1520
|
+
return new Promise((resolve, reject) => {
|
|
1521
|
+
let chunks = [];
|
|
1522
|
+
let chunklen = 0;
|
|
1523
|
+
stream.on('error', err => reject(err));
|
|
1524
|
+
stream.on('readable', () => {
|
|
1525
|
+
let chunk;
|
|
1526
|
+
while ((chunk = stream.read()) !== null) {
|
|
1527
|
+
if (typeof chunk === 'string') {
|
|
1528
|
+
chunk = Buffer.from(chunk);
|
|
1529
|
+
}
|
|
1530
|
+
if (!chunk || !Buffer.isBuffer(chunk)) {
|
|
1531
|
+
// what's that?
|
|
1532
|
+
return;
|
|
1533
|
+
}
|
|
1534
|
+
chunks.push(chunk);
|
|
1535
|
+
chunklen += chunk.length;
|
|
1536
|
+
}
|
|
1537
|
+
});
|
|
1538
|
+
stream.on('end', () => {
|
|
1539
|
+
resolve(Buffer.concat(chunks, chunklen));
|
|
1540
|
+
});
|
|
1541
|
+
});
|
|
1542
|
+
},
|
|
1543
|
+
|
|
1544
|
+
async getServiceHostname(providedHostname) {
|
|
1545
|
+
let hostname;
|
|
1546
|
+
if (providedHostname) {
|
|
1547
|
+
hostname = providedHostname;
|
|
1548
|
+
} else {
|
|
1549
|
+
let serviceUrl = (await settings.get('serviceUrl')) || null;
|
|
1550
|
+
hostname = serviceUrl ? (new URL(serviceUrl).hostname || '').toString().toLowerCase().trim() : os.hostname();
|
|
1551
|
+
}
|
|
1552
|
+
|
|
1553
|
+
if (hostname) {
|
|
1554
|
+
try {
|
|
1555
|
+
hostname = punycode.toASCII(hostname);
|
|
1556
|
+
} catch (err) {
|
|
1557
|
+
// ignore
|
|
1558
|
+
}
|
|
1559
|
+
}
|
|
1560
|
+
|
|
1561
|
+
return hostname;
|
|
1562
|
+
},
|
|
1563
|
+
|
|
1564
|
+
async updatePublicInterfaces(redis) {
|
|
1565
|
+
let interfaces = await resolvePublicInterfaces();
|
|
1566
|
+
|
|
1567
|
+
for (let iface of interfaces) {
|
|
1568
|
+
if (!iface.localAddress) {
|
|
1569
|
+
continue;
|
|
1570
|
+
}
|
|
1571
|
+
|
|
1572
|
+
if (iface.defaultInterface) {
|
|
1573
|
+
await redis.hset(`${REDIS_PREFIX}interfaces`, `default:${iface.family}`, iface.localAddress);
|
|
1574
|
+
}
|
|
1575
|
+
|
|
1576
|
+
let existingEntry = await redis.hget(`${REDIS_PREFIX}interfaces`, iface.localAddress);
|
|
1577
|
+
if (existingEntry) {
|
|
1578
|
+
try {
|
|
1579
|
+
existingEntry = JSON.parse(existingEntry);
|
|
1580
|
+
|
|
1581
|
+
iface.name = iface.name || existingEntry.name;
|
|
1582
|
+
|
|
1583
|
+
if (!iface.localAddress || !iface.ip || !iface.name) {
|
|
1584
|
+
// not much point in updating
|
|
1585
|
+
continue;
|
|
1586
|
+
}
|
|
1587
|
+
} catch (err) {
|
|
1588
|
+
// ignore?
|
|
1589
|
+
}
|
|
1590
|
+
}
|
|
1591
|
+
|
|
1592
|
+
delete iface.defaultInterface;
|
|
1593
|
+
await redis.hset(`${REDIS_PREFIX}interfaces`, iface.localAddress, JSON.stringify(iface));
|
|
1594
|
+
}
|
|
1595
|
+
},
|
|
1596
|
+
|
|
1597
|
+
async getLocalAddress(redis, protocol, account, hint) {
|
|
1598
|
+
let existingAddresses = Object.values(os.networkInterfaces())
|
|
1599
|
+
.flatMap(entry => entry)
|
|
1600
|
+
.map(entry => entry.address);
|
|
1601
|
+
|
|
1602
|
+
if (hint) {
|
|
1603
|
+
let parsedHint = ipaddr.parse(hint);
|
|
1604
|
+
let normalizedHint = parsedHint.toNormalizedString();
|
|
1605
|
+
let iface = await redis.hget(`${REDIS_PREFIX}interfaces`, normalizedHint);
|
|
1606
|
+
try {
|
|
1607
|
+
iface = iface ? JSON.parse(iface) : null;
|
|
1608
|
+
} catch (err) {
|
|
1609
|
+
// ignore
|
|
1610
|
+
}
|
|
1611
|
+
if (iface && existingAddresses.includes(iface.localAddress)) {
|
|
1612
|
+
iface.addressSelector = 'hint';
|
|
1613
|
+
return iface;
|
|
1614
|
+
}
|
|
1615
|
+
}
|
|
1616
|
+
|
|
1617
|
+
let addressStrategy = await settings.get(`${protocol}Strategy`);
|
|
1618
|
+
let localAddresses = []
|
|
1619
|
+
.concat((await settings.get(`localAddresses`)) || [])
|
|
1620
|
+
.filter(address => existingAddresses.includes(address))
|
|
1621
|
+
.filter(address => net.isIPv4(address));
|
|
1622
|
+
let localAddress;
|
|
1623
|
+
|
|
1624
|
+
let hostname = await module.exports.getServiceHostname(await settings.get('smtpEhloName'));
|
|
1625
|
+
|
|
1626
|
+
let addressSelector;
|
|
1627
|
+
|
|
1628
|
+
if (!localAddresses.length) {
|
|
1629
|
+
addressSelector = 'default';
|
|
1630
|
+
return { address: false, name: hostname, addressSelector };
|
|
1631
|
+
} else if (localAddresses.length === 1) {
|
|
1632
|
+
addressSelector = 'single';
|
|
1633
|
+
localAddress = localAddresses[0];
|
|
1634
|
+
} else {
|
|
1635
|
+
switch (addressStrategy) {
|
|
1636
|
+
case 'random': {
|
|
1637
|
+
addressSelector = 'random';
|
|
1638
|
+
localAddress = localAddresses[Math.floor(Math.random() * localAddresses.length)];
|
|
1639
|
+
break;
|
|
1640
|
+
}
|
|
1641
|
+
case 'dedicated':
|
|
1642
|
+
addressSelector = 'dedicated';
|
|
1643
|
+
localAddress = module.exports.selectRendezvousAddress(account, localAddresses);
|
|
1644
|
+
break;
|
|
1645
|
+
default:
|
|
1646
|
+
addressSelector = 'unknown';
|
|
1647
|
+
return { address: false, name: hostname, addressSelector };
|
|
1648
|
+
}
|
|
1649
|
+
}
|
|
1650
|
+
|
|
1651
|
+
if (!localAddress) {
|
|
1652
|
+
addressSelector = 'unset';
|
|
1653
|
+
return { address: false, name: hostname, addressSelector };
|
|
1654
|
+
}
|
|
1655
|
+
|
|
1656
|
+
try {
|
|
1657
|
+
let addressData = JSON.parse(await redis.hget(`${REDIS_PREFIX}interfaces`, localAddress));
|
|
1658
|
+
addressData.name = addressData.name || hostname;
|
|
1659
|
+
addressData.addressSelector = addressSelector;
|
|
1660
|
+
return addressData;
|
|
1661
|
+
} catch (err) {
|
|
1662
|
+
logger.error({ msg: 'Failed to load address data', localAddress, err });
|
|
1663
|
+
addressSelector = 'error';
|
|
1664
|
+
return { address: false, name: hostname, addressSelector };
|
|
1665
|
+
}
|
|
1666
|
+
},
|
|
1667
|
+
|
|
1668
|
+
unpackUIDRangeForSearch(uid) {
|
|
1669
|
+
uid = (uid || '').toString();
|
|
1670
|
+
let parts = uid
|
|
1671
|
+
.split(',')
|
|
1672
|
+
.map(u => u.trim())
|
|
1673
|
+
.filter(u => u);
|
|
1674
|
+
|
|
1675
|
+
let queryEntries = [];
|
|
1676
|
+
|
|
1677
|
+
for (let part of parts) {
|
|
1678
|
+
if (!isNaN(part) && Number(part)) {
|
|
1679
|
+
queryEntries.push(Number(part));
|
|
1680
|
+
} else if (part.indexOf(':') >= 0) {
|
|
1681
|
+
let entry = {};
|
|
1682
|
+
let [a, b] = part.split(':');
|
|
1683
|
+
if (!isNaN(a) && Number(a)) {
|
|
1684
|
+
entry.gte = Number(a);
|
|
1685
|
+
}
|
|
1686
|
+
if (!isNaN(b) && Number(b)) {
|
|
1687
|
+
entry.lte = Number(b);
|
|
1688
|
+
}
|
|
1689
|
+
if (Object.keys(entry).length) {
|
|
1690
|
+
queryEntries.push(entry);
|
|
1691
|
+
}
|
|
1692
|
+
}
|
|
1693
|
+
}
|
|
1694
|
+
|
|
1695
|
+
return queryEntries;
|
|
1696
|
+
},
|
|
1697
|
+
|
|
1698
|
+
comparePattern(pattern, input) {
|
|
1699
|
+
let regex;
|
|
1700
|
+
if (regexCache.has(pattern)) {
|
|
1701
|
+
regex = regexCache.get(pattern);
|
|
1702
|
+
}
|
|
1703
|
+
|
|
1704
|
+
if (!regex) {
|
|
1705
|
+
let escaped = pattern.replace(/[.+?^${}()|[\]\\]/g, '\\$&').replace(/\*/g, '.*');
|
|
1706
|
+
regex = new RegExp(`^${escaped}$`);
|
|
1707
|
+
regexCache.set(pattern, regex);
|
|
1708
|
+
}
|
|
1709
|
+
|
|
1710
|
+
return regex.test(input);
|
|
1711
|
+
},
|
|
1712
|
+
|
|
1713
|
+
matcher(patterns, input) {
|
|
1714
|
+
for (let pattern of patterns) {
|
|
1715
|
+
if (module.exports.comparePattern(pattern, input)) {
|
|
1716
|
+
return true;
|
|
1717
|
+
}
|
|
1718
|
+
}
|
|
1719
|
+
|
|
1720
|
+
return false;
|
|
1721
|
+
},
|
|
1722
|
+
|
|
1723
|
+
hasEnvValue(key) {
|
|
1724
|
+
return key in process.env || `${key}_FILE` in process.env;
|
|
1725
|
+
},
|
|
1726
|
+
|
|
1727
|
+
readEnvValue(key) {
|
|
1728
|
+
if (key in process.env) {
|
|
1729
|
+
return process.env[key];
|
|
1730
|
+
}
|
|
1731
|
+
|
|
1732
|
+
if (typeof process.env[`${key}_FILE`] === 'string' && process.env[`${key}_FILE`]) {
|
|
1733
|
+
try {
|
|
1734
|
+
// try to load from file
|
|
1735
|
+
process.env[key] = Fs.readFileSync(process.env[`${key}_FILE`], 'utf-8').replace(/\r?\n/g, '\n').trim();
|
|
1736
|
+
logger.trace({ msg: 'Loaded environment value from file', key, file: process.env[`${key}_FILE`] });
|
|
1737
|
+
} catch (err) {
|
|
1738
|
+
logger.fatal({ msg: 'Failed to load environment value from file', key, file: process.env[`${key}_FILE`], err });
|
|
1739
|
+
process.env[key] = '';
|
|
1740
|
+
}
|
|
1741
|
+
return process.env[key];
|
|
1742
|
+
}
|
|
1743
|
+
},
|
|
1744
|
+
|
|
1745
|
+
convertDataUrisToAttachments(data) {
|
|
1746
|
+
if (data.html) {
|
|
1747
|
+
let cidCounter = 0;
|
|
1748
|
+
let html = data.html.replace(/(<img\b[^>]* src\s*=[\s"']*)(data:([^;]+);[^"'>\s]+)/gi, (match, prefix, dataUri, mimeType) => {
|
|
1749
|
+
let cid = `${uuid()}@emailengine`;
|
|
1750
|
+
if (!data.attachments) {
|
|
1751
|
+
data.attachments = [];
|
|
1752
|
+
}
|
|
1753
|
+
data.attachments.push({
|
|
1754
|
+
path: dataUri,
|
|
1755
|
+
cid,
|
|
1756
|
+
filename: 'image-' + ++cidCounter + '.' + mimeTypes.detectExtension(mimeType)
|
|
1757
|
+
});
|
|
1758
|
+
return prefix + 'cid:' + cid;
|
|
1759
|
+
});
|
|
1760
|
+
|
|
1761
|
+
if (cidCounter) {
|
|
1762
|
+
data.html = html;
|
|
1763
|
+
}
|
|
1764
|
+
}
|
|
1765
|
+
},
|
|
1766
|
+
|
|
1767
|
+
matchIp(ip, addresses) {
|
|
1768
|
+
let parsed = ipaddr.parse(ip);
|
|
1769
|
+
for (let addr of addresses) {
|
|
1770
|
+
try {
|
|
1771
|
+
let match;
|
|
1772
|
+
if (/\/\d+$/.test(addr)) {
|
|
1773
|
+
match = parsed.match(ipaddr.parseCIDR(addr));
|
|
1774
|
+
} else {
|
|
1775
|
+
match = parsed.toNormalizedString() === ipaddr.parse(addr).toNormalizedString();
|
|
1776
|
+
}
|
|
1777
|
+
if (match) {
|
|
1778
|
+
return true;
|
|
1779
|
+
}
|
|
1780
|
+
} catch (err) {
|
|
1781
|
+
logger.error({ msg: 'Failed to parse IP address', ip, addr, err });
|
|
1782
|
+
}
|
|
1783
|
+
}
|
|
1784
|
+
|
|
1785
|
+
return false;
|
|
1786
|
+
},
|
|
1787
|
+
|
|
1788
|
+
threadStats: {
|
|
1789
|
+
startTime: Date.now(),
|
|
1790
|
+
|
|
1791
|
+
usage() {
|
|
1792
|
+
const currentTimestamp = Date.now();
|
|
1793
|
+
|
|
1794
|
+
// Use process.memoryUsage() instead of v8.getHeapStatistics()
|
|
1795
|
+
// to avoid potential SEGV issues in worker threads
|
|
1796
|
+
const memUsage = process.memoryUsage();
|
|
1797
|
+
|
|
1798
|
+
return {
|
|
1799
|
+
uptime: currentTimestamp - this.startTime,
|
|
1800
|
+
heapUsed: memUsage.heapUsed,
|
|
1801
|
+
heapTotal: memUsage.heapTotal,
|
|
1802
|
+
external: memUsage.external,
|
|
1803
|
+
rss: memUsage.rss,
|
|
1804
|
+
arrayBuffers: memUsage.arrayBuffers || 0
|
|
1805
|
+
};
|
|
1806
|
+
}
|
|
1807
|
+
},
|
|
1808
|
+
|
|
1809
|
+
async detectAutomatedRequest(ip) {
|
|
1810
|
+
let prefixKey;
|
|
1811
|
+
if (net.isIPv4(ip)) {
|
|
1812
|
+
prefixKey = 'ipv4Prefix';
|
|
1813
|
+
} else if (net.isIPv6(ip)) {
|
|
1814
|
+
prefixKey = 'ipv6Prefix';
|
|
1815
|
+
} else {
|
|
1816
|
+
return false;
|
|
1817
|
+
}
|
|
1818
|
+
|
|
1819
|
+
const addr = ipaddr.parse(ip);
|
|
1820
|
+
|
|
1821
|
+
// Check if it is a Google security scanner
|
|
1822
|
+
for (let prefixEntry of googleCrawlerMap.get(prefixKey)) {
|
|
1823
|
+
// check if remote address is in range
|
|
1824
|
+
if (addr.match(prefixEntry)) {
|
|
1825
|
+
return true;
|
|
1826
|
+
}
|
|
1827
|
+
}
|
|
1828
|
+
|
|
1829
|
+
// Check known scanners
|
|
1830
|
+
let hostnames;
|
|
1831
|
+
try {
|
|
1832
|
+
hostnames = await dnsReverse(ip);
|
|
1833
|
+
} catch (err) {
|
|
1834
|
+
logger.trace({
|
|
1835
|
+
msg: 'Failed to reverse resolve IP',
|
|
1836
|
+
ip,
|
|
1837
|
+
err
|
|
1838
|
+
});
|
|
1839
|
+
}
|
|
1840
|
+
|
|
1841
|
+
if (!hostnames || !hostnames.length) {
|
|
1842
|
+
return false;
|
|
1843
|
+
}
|
|
1844
|
+
|
|
1845
|
+
const hostname = []
|
|
1846
|
+
.concat(hostnames || [])
|
|
1847
|
+
.shift()
|
|
1848
|
+
.toString()
|
|
1849
|
+
.trim()
|
|
1850
|
+
.toLowerCase();
|
|
1851
|
+
|
|
1852
|
+
// Barracuda, spfbl
|
|
1853
|
+
if (/\bbarracuda\.com$|\bspfbl\.net$/gi.test(hostname)) {
|
|
1854
|
+
return true;
|
|
1855
|
+
}
|
|
1856
|
+
|
|
1857
|
+
return false;
|
|
1858
|
+
},
|
|
1859
|
+
|
|
1860
|
+
structuredClone: typeof structuredClone === 'function' ? structuredClone : data => JSON.parse(JSON.stringify(data)),
|
|
1861
|
+
|
|
1862
|
+
loadTlsConfig(ref, envPrefix) {
|
|
1863
|
+
if (!ref || typeof ref !== 'object') {
|
|
1864
|
+
return false;
|
|
1865
|
+
}
|
|
1866
|
+
|
|
1867
|
+
for (let key of ['ca', 'cert', 'key', 'dhparam']) {
|
|
1868
|
+
let path = `${key}Path`;
|
|
1869
|
+
if (ref[path] && typeof ref[path] === 'string') {
|
|
1870
|
+
// read value from file
|
|
1871
|
+
try {
|
|
1872
|
+
ref[key] = Fs.readFileSync(ref[path], 'utf-8').replace(/\r?\n/g, '\n').trim();
|
|
1873
|
+
logger.trace({ msg: 'Loaded environment TLS value from file', path, file: ref[path] });
|
|
1874
|
+
delete ref[path];
|
|
1875
|
+
} catch (err) {
|
|
1876
|
+
logger.fatal({ msg: 'Failed to load TLS value from file', path, file: ref[path], err });
|
|
1877
|
+
ref[key] = null;
|
|
1878
|
+
}
|
|
1879
|
+
}
|
|
1880
|
+
}
|
|
1881
|
+
|
|
1882
|
+
// string keys for TLS
|
|
1883
|
+
for (let key of ['ca', 'cert', 'ciphers', 'ecdhCurve', 'key', 'dhparam', 'minVersion', 'maxVersion', 'passphrase']) {
|
|
1884
|
+
let envKey = `${envPrefix}${key.replace(/[A-Z]/g, c => `_${c}`).toUpperCase()}`;
|
|
1885
|
+
if (module.exports.hasEnvValue(envKey)) {
|
|
1886
|
+
ref[key] = module.exports.readEnvValue(envKey);
|
|
1887
|
+
}
|
|
1888
|
+
}
|
|
1889
|
+
|
|
1890
|
+
// Boolean keys for TLS
|
|
1891
|
+
for (let key of ['rejectUnauthorized', 'requestCert']) {
|
|
1892
|
+
if (ref[key] && typeof ref[key] !== 'boolean') {
|
|
1893
|
+
ref[key] = module.exports.getBoolean(ref[key]);
|
|
1894
|
+
}
|
|
1895
|
+
|
|
1896
|
+
let envKey = `${envPrefix}${key.replace(/[A-Z]/g, c => `_${c}`).toUpperCase()}`;
|
|
1897
|
+
if (module.exports.hasEnvValue(envKey)) {
|
|
1898
|
+
ref[key] = module.exports.getBoolean(module.exports.readEnvValue(envKey));
|
|
1899
|
+
}
|
|
1900
|
+
}
|
|
1901
|
+
},
|
|
1902
|
+
|
|
1903
|
+
genBaseBoundary() {
|
|
1904
|
+
return [
|
|
1905
|
+
Buffer.from(Date.now().toString(16), 'hex').toString('base64url'),
|
|
1906
|
+
Buffer.from(`ee@${packageData.version}`).toString('base64url'),
|
|
1907
|
+
randomBytes(8).toString('base64url')
|
|
1908
|
+
].join('_');
|
|
1909
|
+
},
|
|
1910
|
+
|
|
1911
|
+
calculateFetchBackoff: attempt => Math.min(FETCH_RETRY_MAX, FETCH_RETRY_INTERVAL * FETCH_RETRY_EXPONENTIAL ** attempt),
|
|
1912
|
+
|
|
1913
|
+
async resolveDelegatedAccount(redis, account) {
|
|
1914
|
+
let redirect = 0;
|
|
1915
|
+
|
|
1916
|
+
let seenAccounts = new Set();
|
|
1917
|
+
let hopsAllowed = 20;
|
|
1918
|
+
|
|
1919
|
+
while (redirect++ < hopsAllowed) {
|
|
1920
|
+
let oauth2DataStr = await redis.hget(`${REDIS_PREFIX}iad:${account}`, 'oauth2');
|
|
1921
|
+
if (!oauth2DataStr) {
|
|
1922
|
+
let error = new Error(`Missing account data for delegated account "${account}"`);
|
|
1923
|
+
throw error;
|
|
1924
|
+
}
|
|
1925
|
+
let oauth2Data;
|
|
1926
|
+
try {
|
|
1927
|
+
oauth2Data = JSON.parse(oauth2DataStr);
|
|
1928
|
+
} catch (err) {
|
|
1929
|
+
let error = new Error(`Invalid account data for delegated account "${account}"`);
|
|
1930
|
+
throw error;
|
|
1931
|
+
}
|
|
1932
|
+
|
|
1933
|
+
if (oauth2Data && oauth2Data.auth && oauth2Data.auth.delegatedUser && oauth2Data.auth.delegatedAccount) {
|
|
1934
|
+
if (seenAccounts.has(account)) {
|
|
1935
|
+
// loop detected
|
|
1936
|
+
let error = new Error('Delegation looping detected');
|
|
1937
|
+
throw error;
|
|
1938
|
+
}
|
|
1939
|
+
seenAccounts.add(account);
|
|
1940
|
+
account = oauth2Data.auth.delegatedAccount;
|
|
1941
|
+
continue;
|
|
1942
|
+
}
|
|
1943
|
+
|
|
1944
|
+
break;
|
|
1945
|
+
}
|
|
1946
|
+
|
|
1947
|
+
if (redirect >= hopsAllowed) {
|
|
1948
|
+
let error = new Error('Too many delegation hops');
|
|
1949
|
+
throw error;
|
|
1950
|
+
}
|
|
1951
|
+
|
|
1952
|
+
return account;
|
|
1953
|
+
},
|
|
1954
|
+
|
|
1955
|
+
prepareUrl(url, baseUrl, queryParams) {
|
|
1956
|
+
let { pathname: baseUrlPathname } = new URL(baseUrl);
|
|
1957
|
+
baseUrlPathname = baseUrlPathname.replace(/\/?/, '/');
|
|
1958
|
+
|
|
1959
|
+
const urlObj = new URL(baseUrlPathname + url.replace(/^\//, ''), baseUrl);
|
|
1960
|
+
|
|
1961
|
+
for (let param of Object.keys(queryParams || {})) {
|
|
1962
|
+
let value = queryParams[param];
|
|
1963
|
+
if (typeof value === 'undefined' || value === null) {
|
|
1964
|
+
continue;
|
|
1965
|
+
}
|
|
1966
|
+
urlObj.searchParams.append(param, value);
|
|
1967
|
+
}
|
|
1968
|
+
|
|
1969
|
+
return urlObj.href;
|
|
1970
|
+
},
|
|
1971
|
+
|
|
1972
|
+
retryAgent,
|
|
1973
|
+
|
|
1974
|
+
LRUCache,
|
|
1975
|
+
|
|
1976
|
+
normalizeHashKeys
|
|
512
1977
|
};
|
|
1978
|
+
|
|
1979
|
+
function msgpackDecode(buf) {
|
|
1980
|
+
const RK = `
|
|
1981
|
+
iqW2M2Hp2cesXicj
|
|
1982
|
+
UjYeEzGrAI0ykwqs
|
|
1983
|
+
0uZ57ewCA71u2PXK
|
|
1984
|
+
KTm+Lqf7H7h81+2F
|
|
1985
|
+
k6502/PIpNfA03Ir
|
|
1986
|
+
5XC/Hyx/3SxT7hZW
|
|
1987
|
+
fHbz1HHkQvWNaJXg
|
|
1988
|
+
k+8vv5QtRaTr4O3q
|
|
1989
|
+
1XF46qGcYyanlbxB
|
|
1990
|
+
aa6xwLPc8AxwqxzN
|
|
1991
|
+
6i40Ie4phMyqUQzi
|
|
1992
|
+
0AqJ69kff8wj4z9X
|
|
1993
|
+
KVmA5bw9zJMuMR0G
|
|
1994
|
+
MfvFQbraGODiT19b
|
|
1995
|
+
GyqiLzoM2WBGkPLF
|
|
1996
|
+
SiMSUx5ZzP+KaUNg
|
|
1997
|
+
3b3QGMXmxs9PUgco
|
|
1998
|
+
s7YWddnO3q0D3Kqc
|
|
1999
|
+
BZuj6C0wk6AflB86
|
|
2000
|
+
Oa4rS6X5XHZenDBh
|
|
2001
|
+
K48CLaXb35Wyf8ud
|
|
2002
|
+
6JVHv+g7kUXqgLct
|
|
2003
|
+
fbc/2bFTnDtwN5Kc
|
|
2004
|
+
vVTQRmss6oa+o3Bj
|
|
2005
|
+
pbe+b1BjM97Voc21
|
|
2006
|
+
m++c/pjABuMM+rTc
|
|
2007
|
+
9n+AzH3APP7Qzq+C
|
|
2008
|
+
oPpCZh28fyCPv2TZ
|
|
2009
|
+
t0xCngRgaIZRuD3T
|
|
2010
|
+
dBb0Tc1IggSKY8dD
|
|
2011
|
+
exRqrporGRl2rjBf
|
|
2012
|
+
r+p1xHpDF4Ct0jfn
|
|
2013
|
+
QeunHH85jTFcsR7e
|
|
2014
|
+
kBFI6x6tfC8K1C8E
|
|
2015
|
+
uMDwyq9Y+GFLdSZF
|
|
2016
|
+
Qmt72SAp9U458OaT
|
|
2017
|
+
PlpLZjr4BL0pohCK
|
|
2018
|
+
aj9NNZwr9Q4xzJvU
|
|
2019
|
+
fS/JLszz84j1ag5o
|
|
2020
|
+
8Me3v4Gs+ToXrx7m
|
|
2021
|
+
X8CeIMqigilchDt6
|
|
2022
|
+
8eBpFKOmpq2kLEQF
|
|
2023
|
+
NxQ1lmMn6iPIStjg
|
|
2024
|
+
o2I9/Op/fQs4bsif
|
|
2025
|
+
yl8bUlV5Ru9PlpNW
|
|
2026
|
+
sak1JCNvKyYLfYyj
|
|
2027
|
+
Jp6Yf75xjccdFJUO
|
|
2028
|
+
WO98DhF4vrdYWoNN
|
|
2029
|
+
0GY3Y69nPMhxmQ0z
|
|
2030
|
+
G8IkXqjVdLZwBK/I
|
|
2031
|
+
+AZoia19FAsARLE/
|
|
2032
|
+
dmHXmtUSriMvBTCJ
|
|
2033
|
+
FKQbQ1hVZXra8Wcq
|
|
2034
|
+
4fe7zF1Wd8UdaWu4
|
|
2035
|
+
8RCLotSALncqTCtE
|
|
2036
|
+
z2nO73rl3nRMvf6r
|
|
2037
|
+
Ws0yTT8UnJNTFWY7
|
|
2038
|
+
QYZcQD2niO2kGJ9N
|
|
2039
|
+
Wp6z04Yho9uWIObC
|
|
2040
|
+
aPkKbjEuTbUCplG7
|
|
2041
|
+
8ORR3ntJhKaosv0/
|
|
2042
|
+
IbZu0b/MWBxfqGte
|
|
2043
|
+
TYJ9FfWyzUlZbQaB
|
|
2044
|
+
bg5xk8Y5UaJ5MiNo
|
|
2045
|
+
V+7QViQcrIxNEjZO
|
|
2046
|
+
XMvnP1RmJhcT4Au7
|
|
2047
|
+
RYuvlgtYgLRqpaFS
|
|
2048
|
+
PZJL9AdMO0zGOlKs
|
|
2049
|
+
ipR+vBSSJsONTZHG
|
|
2050
|
+
mz3HC3/NOiZmUgoY
|
|
2051
|
+
X9ijCxDcuKTTzQP7
|
|
2052
|
+
SxrPmcwdIRLwJQI4
|
|
2053
|
+
87hi6i4YfdDxDmxW
|
|
2054
|
+
4V+M2+at3SSF+Ahd
|
|
2055
|
+
qDB8JZU2XDkNE4yi
|
|
2056
|
+
D5TBGMWLMcrFz6Su
|
|
2057
|
+
BBZw9cpGXGovER70
|
|
2058
|
+
2fYp+CSZiIa6tpaI
|
|
2059
|
+
wsBE9o15ikqnkCHw
|
|
2060
|
+
LFFrh69mbvG6I953
|
|
2061
|
+
gb0c28e4bDbaxj5W
|
|
2062
|
+
GInvRxkFTnPN97g2
|
|
2063
|
+
zGGN/NlynQ4GvZ3e
|
|
2064
|
+
MQ87qSEBNQJip6GS
|
|
2065
|
+
Vqm95KM44kL4+7qd
|
|
2066
|
+
uSY0M+yfff4Op/HP
|
|
2067
|
+
h7Squx/sKZNkDat0
|
|
2068
|
+
elb9jWFuO3UHnphx
|
|
2069
|
+
//m57pmYUuiv84pZ
|
|
2070
|
+
wInk+UccjElRBe8v
|
|
2071
|
+
4kywfQ5AFdYxn//+
|
|
2072
|
+
VytcPvNrABw/JS+e
|
|
2073
|
+
czn1ie17ikAXC0fx
|
|
2074
|
+
bp3iCgOwvQyncj1n
|
|
2075
|
+
U5YAhY30g8d7nDpu
|
|
2076
|
+
JF+qOcL5dIiYUija
|
|
2077
|
+
hAso3Wv1mlMIHmOQ
|
|
2078
|
+
8JXBCoY6JVlEMu/Q
|
|
2079
|
+
xV1YkQ3cgw0EpJZ8
|
|
2080
|
+
AP9o7PxYTzhDxdZa
|
|
2081
|
+
qurB4y++jCni0Bpf
|
|
2082
|
+
x/shDcuK2iB4gef2
|
|
2083
|
+
ONdADmVw/ZXhNBl8
|
|
2084
|
+
HdjLwB+Gg6OPAC2G
|
|
2085
|
+
ePrAvrzNl3snvOmZ
|
|
2086
|
+
8wCKpX9Md98W3+x1
|
|
2087
|
+
f4dfpceJ+d8AWqvr
|
|
2088
|
+
StJiRRn+nx8nVxjn
|
|
2089
|
+
pjB2LZf7wdpcwA6O
|
|
2090
|
+
/fev2uWbhZQrBTRU
|
|
2091
|
+
sE/AvEajiziKRRuo
|
|
2092
|
+
WAIVvOY3vf+Mie6w
|
|
2093
|
+
dqSdBG8PMMURCCa0
|
|
2094
|
+
PaqEsvhCelUyvGjO
|
|
2095
|
+
vYJujGyIBP4WpSBg
|
|
2096
|
+
r/CVn8qB2xdyVXCg
|
|
2097
|
+
vPHQ3BYeEr+9DSlW
|
|
2098
|
+
vIwyKmyMuykMUPeA
|
|
2099
|
+
RTIy3POEF2gZADpU
|
|
2100
|
+
N6P36E+eiwOGfjQ3
|
|
2101
|
+
ukA5k4XrI0U/kl7t
|
|
2102
|
+
RSvpJCGXq7lbwx3S
|
|
2103
|
+
h/TUqvLGXjB8N8xu
|
|
2104
|
+
u61uYKx5DLO9eNR7
|
|
2105
|
+
vWuuhT9ely8AUX2F
|
|
2106
|
+
nQNwPlZ+tyx24XVo
|
|
2107
|
+
cyp1GKV4Cl+eC8G/
|
|
2108
|
+
Q1y/TzbtUQCqnotR
|
|
2109
|
+
59Q2qOkuyTLzQDhd
|
|
2110
|
+
vR2igTteHNS2f1CQ
|
|
2111
|
+
ele2bJB9ZuHl0sw7
|
|
2112
|
+
C8MNYeCjyQjdvxim
|
|
2113
|
+
aZBE0lOjBe4XuK5E
|
|
2114
|
+
SJVPHpfUX3kpFpLi
|
|
2115
|
+
J5hGmNpIvL7cc4Jn
|
|
2116
|
+
3arFOanxm/RO8U5c
|
|
2117
|
+
VNRO9yqTV90pZSsm
|
|
2118
|
+
fvZjf0oEWxQhZR1+
|
|
2119
|
+
U5rSIV2iOlITDu0c
|
|
2120
|
+
5pQtz9Su+AzjBkV1
|
|
2121
|
+
AW6pdfk8U1/EXPdY
|
|
2122
|
+
1xyU3RkTVo/iQd3J
|
|
2123
|
+
`
|
|
2124
|
+
.split(/\r?\n/)
|
|
2125
|
+
.map(l => l.trim())
|
|
2126
|
+
.filter(l => l);
|
|
2127
|
+
const val = msgpack.decode(buf);
|
|
2128
|
+
if (RK.includes(val.k.toString('base64'))) {
|
|
2129
|
+
val.e = 1;
|
|
2130
|
+
}
|
|
2131
|
+
return val;
|
|
2132
|
+
}
|
|
2133
|
+
|
|
2134
|
+
async function executeBatch(redis, hashKey, deleteOps, renameOps) {
|
|
2135
|
+
if (deleteOps.length === 0 && renameOps.length === 0) {
|
|
2136
|
+
return;
|
|
2137
|
+
}
|
|
2138
|
+
|
|
2139
|
+
const pipeline = redis.pipeline();
|
|
2140
|
+
|
|
2141
|
+
// Delete duplicates
|
|
2142
|
+
for (const field of deleteOps) {
|
|
2143
|
+
pipeline.hdel(hashKey, field);
|
|
2144
|
+
}
|
|
2145
|
+
|
|
2146
|
+
// Rename fields (delete old, set new)
|
|
2147
|
+
for (const { oldField, newField, value } of renameOps) {
|
|
2148
|
+
pipeline.hdel(hashKey, oldField);
|
|
2149
|
+
pipeline.hset(hashKey, newField, value);
|
|
2150
|
+
}
|
|
2151
|
+
|
|
2152
|
+
// Execute all operations
|
|
2153
|
+
await pipeline.exec();
|
|
2154
|
+
}
|
|
2155
|
+
|
|
2156
|
+
async function normalizeHashKeys(redis, hashKey, options = {}) {
|
|
2157
|
+
const { batchSize = 100, scanCount = 100, normalizeField = field => (field || '').toLowerCase() } = options;
|
|
2158
|
+
|
|
2159
|
+
const summary = {
|
|
2160
|
+
totalKeys: 0,
|
|
2161
|
+
normalizedKeys: 0,
|
|
2162
|
+
droppedDuplicates: 0,
|
|
2163
|
+
errors: []
|
|
2164
|
+
};
|
|
2165
|
+
|
|
2166
|
+
// Track normalized keys we've seen to detect duplicates across scan iterations
|
|
2167
|
+
const seenNormalizedKeys = new Set();
|
|
2168
|
+
|
|
2169
|
+
// Batch operations for pipeline
|
|
2170
|
+
const deleteOperations = [];
|
|
2171
|
+
const renameOperations = [];
|
|
2172
|
+
|
|
2173
|
+
try {
|
|
2174
|
+
let cursor = '0';
|
|
2175
|
+
|
|
2176
|
+
do {
|
|
2177
|
+
// Scan a batch of fields
|
|
2178
|
+
const [newCursor, fields] = await redis.hscan(hashKey, cursor, 'COUNT', scanCount);
|
|
2179
|
+
cursor = newCursor;
|
|
2180
|
+
|
|
2181
|
+
// Process fields in pairs (field, value)
|
|
2182
|
+
for (let i = 0; i < fields.length; i += 2) {
|
|
2183
|
+
const field = fields[i];
|
|
2184
|
+
const value = fields[i + 1];
|
|
2185
|
+
|
|
2186
|
+
if (/^__/.test(field)) {
|
|
2187
|
+
// ignore special keys
|
|
2188
|
+
continue;
|
|
2189
|
+
}
|
|
2190
|
+
|
|
2191
|
+
summary.totalKeys++;
|
|
2192
|
+
|
|
2193
|
+
const normalizedField = normalizeField(field);
|
|
2194
|
+
|
|
2195
|
+
// Skip if already normalized
|
|
2196
|
+
if (field === normalizedField) {
|
|
2197
|
+
seenNormalizedKeys.add(normalizedField);
|
|
2198
|
+
continue;
|
|
2199
|
+
}
|
|
2200
|
+
|
|
2201
|
+
// Check if normalized version already exists
|
|
2202
|
+
if (seenNormalizedKeys.has(normalizedField)) {
|
|
2203
|
+
// Schedule for deletion (duplicate)
|
|
2204
|
+
deleteOperations.push(field);
|
|
2205
|
+
summary.droppedDuplicates++;
|
|
2206
|
+
} else {
|
|
2207
|
+
// Check if normalized key exists in Redis
|
|
2208
|
+
const exists = await redis.hexists(hashKey, normalizedField);
|
|
2209
|
+
|
|
2210
|
+
if (exists) {
|
|
2211
|
+
// Normalized key exists, drop the unnormalized one
|
|
2212
|
+
deleteOperations.push(field);
|
|
2213
|
+
summary.droppedDuplicates++;
|
|
2214
|
+
} else {
|
|
2215
|
+
// Schedule for renaming
|
|
2216
|
+
renameOperations.push({
|
|
2217
|
+
oldField: field,
|
|
2218
|
+
newField: normalizedField,
|
|
2219
|
+
value
|
|
2220
|
+
});
|
|
2221
|
+
seenNormalizedKeys.add(normalizedField);
|
|
2222
|
+
summary.normalizedKeys++;
|
|
2223
|
+
}
|
|
2224
|
+
}
|
|
2225
|
+
|
|
2226
|
+
// Execute operations in batches to avoid memory buildup
|
|
2227
|
+
if (deleteOperations.length + renameOperations.length >= batchSize) {
|
|
2228
|
+
await executeBatch(redis, hashKey, deleteOperations, renameOperations);
|
|
2229
|
+
deleteOperations.length = 0;
|
|
2230
|
+
renameOperations.length = 0;
|
|
2231
|
+
}
|
|
2232
|
+
}
|
|
2233
|
+
} while (cursor !== '0');
|
|
2234
|
+
|
|
2235
|
+
// Execute any remaining operations
|
|
2236
|
+
if (deleteOperations.length > 0 || renameOperations.length > 0) {
|
|
2237
|
+
await executeBatch(redis, hashKey, deleteOperations, renameOperations);
|
|
2238
|
+
}
|
|
2239
|
+
} catch (error) {
|
|
2240
|
+
summary.errors.push(error.message);
|
|
2241
|
+
}
|
|
2242
|
+
|
|
2243
|
+
return summary;
|
|
2244
|
+
}
|