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.
Files changed (2154) hide show
  1. package/.env.development +49 -0
  2. package/.env.example +82 -0
  3. package/.env.production +87 -0
  4. package/.eslintignore +1 -0
  5. package/.github/workflows/deploy.yml +104 -0
  6. package/.github/workflows/release.yaml +107 -0
  7. package/.github/workflows/test.yml +82 -0
  8. package/.ncurc.js +19 -5
  9. package/.prettierignore +44 -0
  10. package/CHANGELOG.md +1129 -0
  11. package/DOCKER_DEPLOYMENT.md +495 -0
  12. package/Dockerfile +53 -6
  13. package/Dockerfile-legacy +18 -0
  14. package/Fluid-Attacks-Results.csv +1 -0
  15. package/Gruntfile.js +46 -5
  16. package/LICENSE_EMAILENGINE.txt +110 -0
  17. package/README.md +72 -344
  18. package/app.json +40 -0
  19. package/bin/emailengine.js +671 -46
  20. package/config/default.toml +9 -11
  21. package/config/test.toml +45 -0
  22. package/copy-static-files.sh +34 -0
  23. package/data/google-crawlers.json +797 -0
  24. package/docker-compose.yml +103 -31
  25. package/encrypt.js +85 -10
  26. package/eslint.config.js +110 -0
  27. package/examples/auth-server.js +121 -69
  28. package/examples/grafana-dashboard.json +2375 -0
  29. package/install.sh +426 -0
  30. package/lib/account.js +2348 -124
  31. package/lib/add-trackers.js +119 -0
  32. package/lib/api-routes/bull-board-routes.js +60 -0
  33. package/lib/api-routes/chat-routes.js +519 -0
  34. package/lib/api-routes/template-routes.js +490 -0
  35. package/lib/append-list.js +9 -2
  36. package/lib/arf-detect.js +202 -0
  37. package/lib/autodetect-imap-settings.js +781 -0
  38. package/lib/bounce-detect.js +511 -50
  39. package/lib/capa.js +97 -0
  40. package/lib/consts.js +210 -1
  41. package/lib/db.js +227 -8
  42. package/lib/document-store.js +54 -0
  43. package/lib/email-client/base-client.js +3715 -0
  44. package/lib/email-client/gmail-client.js +2796 -0
  45. package/lib/email-client/imap/mailbox.js +3760 -0
  46. package/lib/email-client/imap/subconnection.js +269 -0
  47. package/lib/email-client/imap-client.js +2628 -0
  48. package/lib/email-client/outlook-client.js +3805 -0
  49. package/lib/encrypt.js +85 -14
  50. package/lib/es.js +784 -0
  51. package/lib/feature-flags.js +42 -0
  52. package/lib/gateway.js +271 -0
  53. package/lib/generate-text-preview.js +56 -0
  54. package/lib/get-raw-email.js +302 -42
  55. package/lib/get-secret.js +23 -67
  56. package/lib/headers-rewriter.js +33 -0
  57. package/lib/imapproxy/imap-core/index.js +4 -0
  58. package/lib/imapproxy/imap-core/lib/commands/append.js +187 -0
  59. package/lib/imapproxy/imap-core/lib/commands/authenticate-plain.js +145 -0
  60. package/lib/imapproxy/imap-core/lib/commands/capability.js +13 -0
  61. package/lib/imapproxy/imap-core/lib/commands/check.js +10 -0
  62. package/lib/imapproxy/imap-core/lib/commands/close.js +44 -0
  63. package/lib/imapproxy/imap-core/lib/commands/compress.js +102 -0
  64. package/lib/imapproxy/imap-core/lib/commands/copy.js +109 -0
  65. package/lib/imapproxy/imap-core/lib/commands/create.js +93 -0
  66. package/lib/imapproxy/imap-core/lib/commands/delete.js +84 -0
  67. package/lib/imapproxy/imap-core/lib/commands/enable.js +36 -0
  68. package/lib/imapproxy/imap-core/lib/commands/expunge.js +68 -0
  69. package/lib/imapproxy/imap-core/lib/commands/fetch.js +385 -0
  70. package/lib/imapproxy/imap-core/lib/commands/getquota.js +85 -0
  71. package/lib/imapproxy/imap-core/lib/commands/getquotaroot.js +111 -0
  72. package/lib/imapproxy/imap-core/lib/commands/id.js +111 -0
  73. package/lib/imapproxy/imap-core/lib/commands/idle.js +45 -0
  74. package/lib/imapproxy/imap-core/lib/commands/list.js +218 -0
  75. package/lib/imapproxy/imap-core/lib/commands/login.js +135 -0
  76. package/lib/imapproxy/imap-core/lib/commands/logout.js +26 -0
  77. package/lib/imapproxy/imap-core/lib/commands/lsub.js +102 -0
  78. package/lib/imapproxy/imap-core/lib/commands/move.js +106 -0
  79. package/lib/imapproxy/imap-core/lib/commands/namespace.js +14 -0
  80. package/lib/imapproxy/imap-core/lib/commands/noop.js +10 -0
  81. package/lib/imapproxy/imap-core/lib/commands/rename.js +102 -0
  82. package/lib/imapproxy/imap-core/lib/commands/search.js +306 -0
  83. package/lib/imapproxy/imap-core/lib/commands/select.js +248 -0
  84. package/lib/imapproxy/imap-core/lib/commands/setquota.js +24 -0
  85. package/lib/imapproxy/imap-core/lib/commands/starttls.js +100 -0
  86. package/lib/imapproxy/imap-core/lib/commands/status.js +149 -0
  87. package/lib/imapproxy/imap-core/lib/commands/store.js +208 -0
  88. package/lib/imapproxy/imap-core/lib/commands/subscribe.js +69 -0
  89. package/lib/imapproxy/imap-core/lib/commands/uid-expunge.js +71 -0
  90. package/lib/imapproxy/imap-core/lib/commands/uid-store.js +170 -0
  91. package/lib/imapproxy/imap-core/lib/commands/unselect.js +14 -0
  92. package/lib/imapproxy/imap-core/lib/commands/unsubscribe.js +69 -0
  93. package/lib/imapproxy/imap-core/lib/handler/README.md +146 -0
  94. package/lib/imapproxy/imap-core/lib/handler/imap-compile-stream.js +252 -0
  95. package/lib/imapproxy/imap-core/lib/handler/imap-compiler.js +134 -0
  96. package/lib/imapproxy/imap-core/lib/handler/imap-formal-syntax.js +147 -0
  97. package/lib/imapproxy/imap-core/lib/handler/imap-handler.js +11 -0
  98. package/lib/imapproxy/imap-core/lib/handler/imap-parser.js +678 -0
  99. package/lib/imapproxy/imap-core/lib/imap-command.js +381 -0
  100. package/lib/imapproxy/imap-core/lib/imap-composer.js +71 -0
  101. package/lib/imapproxy/imap-core/lib/imap-connection.js +929 -0
  102. package/lib/imapproxy/imap-core/lib/imap-server.js +426 -0
  103. package/lib/imapproxy/imap-core/lib/imap-stream.js +172 -0
  104. package/lib/imapproxy/imap-core/lib/imap-tools.js +789 -0
  105. package/lib/imapproxy/imap-core/lib/indexer/body-structure.js +295 -0
  106. package/lib/imapproxy/imap-core/lib/indexer/create-envelope.js +103 -0
  107. package/lib/imapproxy/imap-core/lib/indexer/indexer.js +904 -0
  108. package/lib/imapproxy/imap-core/lib/indexer/parse-mime-tree.js +340 -0
  109. package/lib/imapproxy/imap-core/lib/length-limiter.js +76 -0
  110. package/lib/imapproxy/imap-core/lib/parse-date.js +225 -0
  111. package/lib/imapproxy/imap-core/lib/search.js +330 -0
  112. package/lib/imapproxy/imap-core/lib/tls-options.js +69 -0
  113. package/lib/imapproxy/imap-core/memory-notifier.js +129 -0
  114. package/lib/imapproxy/imap-core/test/client.js +46 -0
  115. package/lib/imapproxy/imap-core/test/fixtures/append.eml +1196 -0
  116. package/lib/imapproxy/imap-core/test/fixtures/chunks.js +44 -0
  117. package/lib/imapproxy/imap-core/test/fixtures/fix1.eml +6 -0
  118. package/lib/imapproxy/imap-core/test/fixtures/fix2.eml +599 -0
  119. package/lib/imapproxy/imap-core/test/fixtures/fix3.eml +32 -0
  120. package/lib/imapproxy/imap-core/test/fixtures/fix4.eml +6 -0
  121. package/lib/imapproxy/imap-core/test/fixtures/mimetorture.eml +599 -0
  122. package/lib/imapproxy/imap-core/test/fixtures/mimetorture.js +2740 -0
  123. package/lib/imapproxy/imap-core/test/fixtures/mimetorture.json +1411 -0
  124. package/lib/imapproxy/imap-core/test/fixtures/mimetree.js +85 -0
  125. package/lib/imapproxy/imap-core/test/fixtures/nodemailer.eml +582 -0
  126. package/lib/imapproxy/imap-core/test/fixtures/ryan_finnie_mime_torture.eml +599 -0
  127. package/lib/imapproxy/imap-core/test/fixtures/simple.eml +42 -0
  128. package/lib/imapproxy/imap-core/test/fixtures/simple.json +164 -0
  129. package/lib/imapproxy/imap-core/test/imap-compile-stream-test.js +671 -0
  130. package/lib/imapproxy/imap-core/test/imap-compiler-test.js +272 -0
  131. package/lib/imapproxy/imap-core/test/imap-indexer-test.js +236 -0
  132. package/lib/imapproxy/imap-core/test/imap-parser-test.js +922 -0
  133. package/lib/imapproxy/imap-core/test/memory-notifier.js +129 -0
  134. package/lib/imapproxy/imap-core/test/prepare.sh +74 -0
  135. package/lib/imapproxy/imap-core/test/protocol-test.js +1756 -0
  136. package/lib/imapproxy/imap-core/test/search-test.js +1356 -0
  137. package/lib/imapproxy/imap-core/test/test-client.js +152 -0
  138. package/lib/imapproxy/imap-core/test/test-server.js +623 -0
  139. package/lib/imapproxy/imap-core/test/tools-test.js +22 -0
  140. package/lib/imapproxy/imap-server.js +577 -0
  141. package/lib/lists.js +92 -0
  142. package/lib/llm-pre-process.js +141 -0
  143. package/lib/logger.js +43 -4
  144. package/lib/lua/ee-get-idempotency.lua +74 -0
  145. package/lib/lua/ee-list-add.lua +34 -0
  146. package/lib/lua/ee-list-remove.lua +37 -0
  147. package/lib/lua/h-incrby-exists.lua +28 -0
  148. package/lib/lua/h-push.lua +32 -0
  149. package/lib/lua/h-set-bigger.lua +40 -0
  150. package/lib/lua/h-set-exists.lua +29 -0
  151. package/lib/lua/h-set-new.lua +29 -0
  152. package/lib/lua/h-update-bigger.lua +45 -0
  153. package/lib/lua/s-list-accounts.lua +64 -14
  154. package/lib/lua/z-expunge.lua +86 -10
  155. package/lib/lua/z-get-by-uid.lua +28 -5
  156. package/lib/lua/z-get-mailbox-id.lua +24 -2
  157. package/lib/lua/z-get-mailbox-path.lua +16 -0
  158. package/lib/lua/z-get.lua +27 -4
  159. package/lib/lua/z-set.lua +24 -2
  160. package/lib/metrics-collector.js +209 -0
  161. package/lib/oauth/gmail.js +663 -0
  162. package/lib/oauth/mail-ru.js +310 -0
  163. package/lib/oauth/outlook.js +541 -0
  164. package/lib/oauth/pubsub/google.js +247 -0
  165. package/lib/oauth2-apps.js +1420 -0
  166. package/lib/outbox.js +140 -0
  167. package/lib/payload-examples-documents.json +404 -0
  168. package/lib/payload-examples-webhooks.json +266 -0
  169. package/lib/pre-process.js +193 -0
  170. package/lib/rate-limit.js +32 -0
  171. package/lib/reconnection-manager.js +106 -0
  172. package/lib/redis-scan-delete.js +82 -0
  173. package/lib/redis-url.js +78 -0
  174. package/lib/rewrite-text-nodes.js +267 -0
  175. package/lib/routes-ui.js +10247 -0
  176. package/lib/schemas.js +1576 -187
  177. package/lib/settings.js +263 -12
  178. package/lib/sub-script.js +109 -0
  179. package/lib/templates.js +240 -0
  180. package/lib/threads.js +155 -0
  181. package/lib/tokens.js +353 -0
  182. package/lib/tools.js +1773 -41
  183. package/lib/translations.js +33 -0
  184. package/lib/webhooks.js +605 -0
  185. package/list-generate.js +96 -0
  186. package/package.json +129 -54
  187. package/render.yaml +44 -0
  188. package/sbom.json +1 -0
  189. package/scan.js +14 -2
  190. package/scripts/README.md +50 -0
  191. package/scripts/refresh-test-tokens.js +180 -0
  192. package/server.js +2902 -376
  193. package/setup-production.sh +201 -0
  194. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.css +3 -3
  195. package/static/bootstrap-4.6.2-dist/css/bootstrap-grid.css.map +1 -0
  196. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.min.css +3 -3
  197. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-grid.min.css.map +1 -1
  198. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-reboot.css +3 -3
  199. package/static/bootstrap-4.6.2-dist/css/bootstrap-reboot.css.map +1 -0
  200. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap-reboot.min.css +3 -3
  201. package/static/bootstrap-4.6.2-dist/css/bootstrap-reboot.min.css.map +1 -0
  202. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/css/bootstrap.css +60 -26
  203. package/static/bootstrap-4.6.2-dist/css/bootstrap.css.map +1 -0
  204. package/static/bootstrap-4.6.2-dist/css/bootstrap.min.css +7 -0
  205. package/static/bootstrap-4.6.2-dist/css/bootstrap.min.css.map +1 -0
  206. package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.js +7155 -0
  207. package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.js.map +784 -0
  208. package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.min.js +7 -0
  209. package/static/bootstrap-4.6.2-dist/js/bootstrap.bundle.min.js.map +959 -0
  210. package/static/{bootstrap-4.6.0-dist → bootstrap-4.6.2-dist}/js/bootstrap.js +792 -868
  211. package/static/bootstrap-4.6.2-dist/js/bootstrap.js.map +1 -0
  212. package/static/bootstrap-4.6.2-dist/js/bootstrap.min.js +7 -0
  213. package/static/bootstrap-4.6.2-dist/js/bootstrap.min.js.map +1 -0
  214. package/static/css/app.css +146 -0
  215. package/static/css/arena.css +777 -0
  216. package/static/css/default.min.css +9 -0
  217. package/static/css/highlight.min.css +9 -0
  218. package/static/css/sb-admin-2.min.css +10 -0
  219. package/static/emailengine.ico +0 -0
  220. package/static/favicon/android-chrome-192x192.png +0 -0
  221. package/static/favicon/android-chrome-512x512.png +0 -0
  222. package/static/favicon/apple-touch-icon.png +0 -0
  223. package/static/favicon/favicon-16x16.png +0 -0
  224. package/static/favicon/favicon-32x32.png +0 -0
  225. package/static/favicon/favicon.ico +0 -0
  226. package/static/favicon.ico +0 -0
  227. package/static/fonts/nunito/OFL.txt +93 -0
  228. package/static/fonts/nunito/XRXV3I6Li01BKofINeaBTMnFcQ.woff2 +0 -0
  229. package/static/fonts/nunito-font.css +66 -0
  230. package/static/front/EmailEngine_logo_horiz.png +0 -0
  231. package/static/front/EmailEngine_logo_vert.png +0 -0
  232. package/static/front/index.html +57 -0
  233. package/static/front/logo.png +0 -0
  234. package/static/imap-capabilities-1.csv +71 -0
  235. package/static/index.html +30 -713
  236. package/static/js/ace/README.txt +1 -0
  237. package/static/js/ace/ace.js +23 -0
  238. package/static/js/ace/ext-language_tools.js +8 -0
  239. package/static/js/ace/ext-searchbox.js +8 -0
  240. package/static/js/ace/mode-handlebars.js +8 -0
  241. package/static/js/ace/mode-html.js +8 -0
  242. package/static/js/ace/mode-javascript.js +8 -0
  243. package/static/js/ace/mode-json.js +8 -0
  244. package/static/js/ace/mode-markdown.js +8 -0
  245. package/static/js/ace/snippets/javascript.js +8 -0
  246. package/static/js/ace/snippets/markdown.js +8 -0
  247. package/static/js/ace/theme-kuroir.js +8 -0
  248. package/static/js/ace/theme-xcode.js +8 -0
  249. package/static/js/ace/worker-html.js +1 -0
  250. package/static/js/ace/worker-javascript.js +1 -0
  251. package/static/js/ace/worker-json.js +1 -0
  252. package/static/js/app.js +526 -0
  253. package/static/js/bootstrap-autocomplete.min.js +1 -0
  254. package/static/js/clipboard.min.js +517 -0
  255. package/static/js/ee-client.js +1977 -0
  256. package/static/js/evaluation-worker.js +47 -0
  257. package/static/js/highlight.min.js +1173 -0
  258. package/static/js/jquery-3.6.0.min.js +2 -0
  259. package/static/js/sb-admin-2.min.js +7 -0
  260. package/static/licenses.html +6606 -50
  261. package/static/logo/EmailEngine_logo_horiz.png +0 -0
  262. package/static/logo/EmailEngine_logo_vert.png +0 -0
  263. package/static/logo.png +0 -0
  264. package/static/logo_transparent.png +0 -0
  265. package/static/logo_transparent_small.png +0 -0
  266. package/static/logo_wide.png +0 -0
  267. package/static/preview/header-template.png +0 -0
  268. package/static/preview/render.png +0 -0
  269. package/static/preview/translation.png +0 -0
  270. package/static/providers/google_dark.png +0 -0
  271. package/static/providers/google_dark_edited.png +0 -0
  272. package/static/providers/google_light.png +0 -0
  273. package/static/providers/ms_dark.svg +1 -0
  274. package/static/providers/ms_light.svg +1 -0
  275. package/static/robots.txt +4 -0
  276. package/static/undraw_profile.svg +38 -0
  277. package/static/vendor/fontawesome-free/LICENSE.txt +34 -0
  278. package/static/vendor/fontawesome-free/attribution.js +3 -0
  279. package/static/vendor/fontawesome-free/css/all.css +4619 -0
  280. package/static/vendor/fontawesome-free/css/all.min.css +5 -0
  281. package/static/vendor/fontawesome-free/css/brands.css +15 -0
  282. package/static/vendor/fontawesome-free/css/brands.min.css +5 -0
  283. package/static/vendor/fontawesome-free/css/fontawesome.css +4585 -0
  284. package/static/vendor/fontawesome-free/css/fontawesome.min.css +5 -0
  285. package/static/vendor/fontawesome-free/css/regular.css +15 -0
  286. package/static/vendor/fontawesome-free/css/regular.min.css +5 -0
  287. package/static/vendor/fontawesome-free/css/solid.css +16 -0
  288. package/static/vendor/fontawesome-free/css/solid.min.css +5 -0
  289. package/static/vendor/fontawesome-free/css/svg-with-js.css +371 -0
  290. package/static/vendor/fontawesome-free/css/svg-with-js.min.css +5 -0
  291. package/static/vendor/fontawesome-free/css/v4-shims.css +2172 -0
  292. package/static/vendor/fontawesome-free/css/v4-shims.min.css +5 -0
  293. package/static/vendor/fontawesome-free/js/all.js +4467 -0
  294. package/static/vendor/fontawesome-free/js/all.min.js +5 -0
  295. package/static/vendor/fontawesome-free/js/brands.js +586 -0
  296. package/static/vendor/fontawesome-free/js/brands.min.js +5 -0
  297. package/static/vendor/fontawesome-free/js/conflict-detection.js +998 -0
  298. package/static/vendor/fontawesome-free/js/conflict-detection.min.js +5 -0
  299. package/static/vendor/fontawesome-free/js/fontawesome.js +2483 -0
  300. package/static/vendor/fontawesome-free/js/fontawesome.min.js +5 -0
  301. package/static/vendor/fontawesome-free/js/regular.js +280 -0
  302. package/static/vendor/fontawesome-free/js/regular.min.js +5 -0
  303. package/static/vendor/fontawesome-free/js/solid.js +1130 -0
  304. package/static/vendor/fontawesome-free/js/solid.min.js +5 -0
  305. package/static/vendor/fontawesome-free/js/v4-shims.js +68 -0
  306. package/static/vendor/fontawesome-free/js/v4-shims.min.js +5 -0
  307. package/static/vendor/fontawesome-free/less/_animated.less +19 -0
  308. package/static/vendor/fontawesome-free/less/_bordered-pulled.less +16 -0
  309. package/static/vendor/fontawesome-free/less/_core.less +12 -0
  310. package/static/vendor/fontawesome-free/less/_fixed-width.less +6 -0
  311. package/static/vendor/fontawesome-free/less/_icons.less +1462 -0
  312. package/static/vendor/fontawesome-free/less/_larger.less +27 -0
  313. package/static/vendor/fontawesome-free/less/_list.less +18 -0
  314. package/static/vendor/fontawesome-free/less/_mixins.less +56 -0
  315. package/static/vendor/fontawesome-free/less/_rotated-flipped.less +24 -0
  316. package/static/vendor/fontawesome-free/less/_screen-reader.less +5 -0
  317. package/static/vendor/fontawesome-free/less/_shims.less +2066 -0
  318. package/static/vendor/fontawesome-free/less/_stacked.less +22 -0
  319. package/static/vendor/fontawesome-free/less/_variables.less +1474 -0
  320. package/static/vendor/fontawesome-free/less/brands.less +23 -0
  321. package/static/vendor/fontawesome-free/less/fontawesome.less +16 -0
  322. package/static/vendor/fontawesome-free/less/regular.less +23 -0
  323. package/static/vendor/fontawesome-free/less/solid.less +24 -0
  324. package/static/vendor/fontawesome-free/less/v4-shims.less +6 -0
  325. package/static/vendor/fontawesome-free/metadata/categories.yml +2572 -0
  326. package/static/vendor/fontawesome-free/metadata/icons.yml +21783 -0
  327. package/static/vendor/fontawesome-free/metadata/shims.yml +298 -0
  328. package/static/vendor/fontawesome-free/metadata/sponsors.yml +744 -0
  329. package/static/vendor/fontawesome-free/package.json +58 -0
  330. package/static/vendor/fontawesome-free/scss/_animated.scss +20 -0
  331. package/static/vendor/fontawesome-free/scss/_bordered-pulled.scss +20 -0
  332. package/static/vendor/fontawesome-free/scss/_core.scss +21 -0
  333. package/static/vendor/fontawesome-free/scss/_fixed-width.scss +6 -0
  334. package/static/vendor/fontawesome-free/scss/_icons.scss +1462 -0
  335. package/static/vendor/fontawesome-free/scss/_larger.scss +23 -0
  336. package/static/vendor/fontawesome-free/scss/_list.scss +18 -0
  337. package/static/vendor/fontawesome-free/scss/_mixins.scss +56 -0
  338. package/static/vendor/fontawesome-free/scss/_rotated-flipped.scss +24 -0
  339. package/static/vendor/fontawesome-free/scss/_screen-reader.scss +5 -0
  340. package/static/vendor/fontawesome-free/scss/_shims.scss +2066 -0
  341. package/static/vendor/fontawesome-free/scss/_stacked.scss +31 -0
  342. package/static/vendor/fontawesome-free/scss/_variables.scss +1479 -0
  343. package/static/vendor/fontawesome-free/scss/brands.scss +23 -0
  344. package/static/vendor/fontawesome-free/scss/fontawesome.scss +16 -0
  345. package/static/vendor/fontawesome-free/scss/regular.scss +23 -0
  346. package/static/vendor/fontawesome-free/scss/solid.scss +24 -0
  347. package/static/vendor/fontawesome-free/scss/v4-shims.scss +6 -0
  348. package/static/vendor/fontawesome-free/sprites/brands.svg +1381 -0
  349. package/static/vendor/fontawesome-free/sprites/regular.svg +463 -0
  350. package/static/vendor/fontawesome-free/sprites/solid.svg +3013 -0
  351. package/static/vendor/fontawesome-free/svgs/brands/500px.svg +1 -0
  352. package/static/vendor/fontawesome-free/svgs/brands/accessible-icon.svg +1 -0
  353. package/static/vendor/fontawesome-free/svgs/brands/accusoft.svg +1 -0
  354. package/static/vendor/fontawesome-free/svgs/brands/acquisitions-incorporated.svg +1 -0
  355. package/static/vendor/fontawesome-free/svgs/brands/adn.svg +1 -0
  356. package/static/vendor/fontawesome-free/svgs/brands/adversal.svg +1 -0
  357. package/static/vendor/fontawesome-free/svgs/brands/affiliatetheme.svg +1 -0
  358. package/static/vendor/fontawesome-free/svgs/brands/airbnb.svg +1 -0
  359. package/static/vendor/fontawesome-free/svgs/brands/algolia.svg +1 -0
  360. package/static/vendor/fontawesome-free/svgs/brands/alipay.svg +1 -0
  361. package/static/vendor/fontawesome-free/svgs/brands/amazon-pay.svg +1 -0
  362. package/static/vendor/fontawesome-free/svgs/brands/amazon.svg +1 -0
  363. package/static/vendor/fontawesome-free/svgs/brands/amilia.svg +1 -0
  364. package/static/vendor/fontawesome-free/svgs/brands/android.svg +1 -0
  365. package/static/vendor/fontawesome-free/svgs/brands/angellist.svg +1 -0
  366. package/static/vendor/fontawesome-free/svgs/brands/angrycreative.svg +1 -0
  367. package/static/vendor/fontawesome-free/svgs/brands/angular.svg +1 -0
  368. package/static/vendor/fontawesome-free/svgs/brands/app-store-ios.svg +1 -0
  369. package/static/vendor/fontawesome-free/svgs/brands/app-store.svg +1 -0
  370. package/static/vendor/fontawesome-free/svgs/brands/apper.svg +1 -0
  371. package/static/vendor/fontawesome-free/svgs/brands/apple-pay.svg +1 -0
  372. package/static/vendor/fontawesome-free/svgs/brands/apple.svg +1 -0
  373. package/static/vendor/fontawesome-free/svgs/brands/artstation.svg +1 -0
  374. package/static/vendor/fontawesome-free/svgs/brands/asymmetrik.svg +1 -0
  375. package/static/vendor/fontawesome-free/svgs/brands/atlassian.svg +1 -0
  376. package/static/vendor/fontawesome-free/svgs/brands/audible.svg +1 -0
  377. package/static/vendor/fontawesome-free/svgs/brands/autoprefixer.svg +1 -0
  378. package/static/vendor/fontawesome-free/svgs/brands/avianex.svg +1 -0
  379. package/static/vendor/fontawesome-free/svgs/brands/aviato.svg +1 -0
  380. package/static/vendor/fontawesome-free/svgs/brands/aws.svg +1 -0
  381. package/static/vendor/fontawesome-free/svgs/brands/bandcamp.svg +1 -0
  382. package/static/vendor/fontawesome-free/svgs/brands/battle-net.svg +1 -0
  383. package/static/vendor/fontawesome-free/svgs/brands/behance-square.svg +1 -0
  384. package/static/vendor/fontawesome-free/svgs/brands/behance.svg +1 -0
  385. package/static/vendor/fontawesome-free/svgs/brands/bimobject.svg +1 -0
  386. package/static/vendor/fontawesome-free/svgs/brands/bitbucket.svg +1 -0
  387. package/static/vendor/fontawesome-free/svgs/brands/bitcoin.svg +1 -0
  388. package/static/vendor/fontawesome-free/svgs/brands/bity.svg +1 -0
  389. package/static/vendor/fontawesome-free/svgs/brands/black-tie.svg +1 -0
  390. package/static/vendor/fontawesome-free/svgs/brands/blackberry.svg +1 -0
  391. package/static/vendor/fontawesome-free/svgs/brands/blogger-b.svg +1 -0
  392. package/static/vendor/fontawesome-free/svgs/brands/blogger.svg +1 -0
  393. package/static/vendor/fontawesome-free/svgs/brands/bluetooth-b.svg +1 -0
  394. package/static/vendor/fontawesome-free/svgs/brands/bluetooth.svg +1 -0
  395. package/static/vendor/fontawesome-free/svgs/brands/bootstrap.svg +1 -0
  396. package/static/vendor/fontawesome-free/svgs/brands/btc.svg +1 -0
  397. package/static/vendor/fontawesome-free/svgs/brands/buffer.svg +1 -0
  398. package/static/vendor/fontawesome-free/svgs/brands/buromobelexperte.svg +1 -0
  399. package/static/vendor/fontawesome-free/svgs/brands/buy-n-large.svg +1 -0
  400. package/static/vendor/fontawesome-free/svgs/brands/buysellads.svg +1 -0
  401. package/static/vendor/fontawesome-free/svgs/brands/canadian-maple-leaf.svg +1 -0
  402. package/static/vendor/fontawesome-free/svgs/brands/cc-amazon-pay.svg +1 -0
  403. package/static/vendor/fontawesome-free/svgs/brands/cc-amex.svg +1 -0
  404. package/static/vendor/fontawesome-free/svgs/brands/cc-apple-pay.svg +1 -0
  405. package/static/vendor/fontawesome-free/svgs/brands/cc-diners-club.svg +1 -0
  406. package/static/vendor/fontawesome-free/svgs/brands/cc-discover.svg +1 -0
  407. package/static/vendor/fontawesome-free/svgs/brands/cc-jcb.svg +1 -0
  408. package/static/vendor/fontawesome-free/svgs/brands/cc-mastercard.svg +1 -0
  409. package/static/vendor/fontawesome-free/svgs/brands/cc-paypal.svg +1 -0
  410. package/static/vendor/fontawesome-free/svgs/brands/cc-stripe.svg +1 -0
  411. package/static/vendor/fontawesome-free/svgs/brands/cc-visa.svg +1 -0
  412. package/static/vendor/fontawesome-free/svgs/brands/centercode.svg +1 -0
  413. package/static/vendor/fontawesome-free/svgs/brands/centos.svg +1 -0
  414. package/static/vendor/fontawesome-free/svgs/brands/chrome.svg +1 -0
  415. package/static/vendor/fontawesome-free/svgs/brands/chromecast.svg +1 -0
  416. package/static/vendor/fontawesome-free/svgs/brands/cloudflare.svg +1 -0
  417. package/static/vendor/fontawesome-free/svgs/brands/cloudscale.svg +1 -0
  418. package/static/vendor/fontawesome-free/svgs/brands/cloudsmith.svg +1 -0
  419. package/static/vendor/fontawesome-free/svgs/brands/cloudversify.svg +1 -0
  420. package/static/vendor/fontawesome-free/svgs/brands/codepen.svg +1 -0
  421. package/static/vendor/fontawesome-free/svgs/brands/codiepie.svg +1 -0
  422. package/static/vendor/fontawesome-free/svgs/brands/confluence.svg +1 -0
  423. package/static/vendor/fontawesome-free/svgs/brands/connectdevelop.svg +1 -0
  424. package/static/vendor/fontawesome-free/svgs/brands/contao.svg +1 -0
  425. package/static/vendor/fontawesome-free/svgs/brands/cotton-bureau.svg +1 -0
  426. package/static/vendor/fontawesome-free/svgs/brands/cpanel.svg +1 -0
  427. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-by.svg +1 -0
  428. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc-eu.svg +1 -0
  429. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc-jp.svg +1 -0
  430. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nc.svg +1 -0
  431. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-nd.svg +1 -0
  432. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-pd-alt.svg +1 -0
  433. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-pd.svg +1 -0
  434. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-remix.svg +1 -0
  435. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sa.svg +1 -0
  436. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sampling-plus.svg +1 -0
  437. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-sampling.svg +1 -0
  438. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-share.svg +1 -0
  439. package/static/vendor/fontawesome-free/svgs/brands/creative-commons-zero.svg +1 -0
  440. package/static/vendor/fontawesome-free/svgs/brands/creative-commons.svg +1 -0
  441. package/static/vendor/fontawesome-free/svgs/brands/critical-role.svg +1 -0
  442. package/static/vendor/fontawesome-free/svgs/brands/css3-alt.svg +1 -0
  443. package/static/vendor/fontawesome-free/svgs/brands/css3.svg +1 -0
  444. package/static/vendor/fontawesome-free/svgs/brands/cuttlefish.svg +1 -0
  445. package/static/vendor/fontawesome-free/svgs/brands/d-and-d-beyond.svg +1 -0
  446. package/static/vendor/fontawesome-free/svgs/brands/d-and-d.svg +1 -0
  447. package/static/vendor/fontawesome-free/svgs/brands/dailymotion.svg +1 -0
  448. package/static/vendor/fontawesome-free/svgs/brands/dashcube.svg +1 -0
  449. package/static/vendor/fontawesome-free/svgs/brands/deezer.svg +1 -0
  450. package/static/vendor/fontawesome-free/svgs/brands/delicious.svg +1 -0
  451. package/static/vendor/fontawesome-free/svgs/brands/deploydog.svg +1 -0
  452. package/static/vendor/fontawesome-free/svgs/brands/deskpro.svg +1 -0
  453. package/static/vendor/fontawesome-free/svgs/brands/dev.svg +1 -0
  454. package/static/vendor/fontawesome-free/svgs/brands/deviantart.svg +1 -0
  455. package/static/vendor/fontawesome-free/svgs/brands/dhl.svg +1 -0
  456. package/static/vendor/fontawesome-free/svgs/brands/diaspora.svg +1 -0
  457. package/static/vendor/fontawesome-free/svgs/brands/digg.svg +1 -0
  458. package/static/vendor/fontawesome-free/svgs/brands/digital-ocean.svg +1 -0
  459. package/static/vendor/fontawesome-free/svgs/brands/discord.svg +1 -0
  460. package/static/vendor/fontawesome-free/svgs/brands/discourse.svg +1 -0
  461. package/static/vendor/fontawesome-free/svgs/brands/dochub.svg +1 -0
  462. package/static/vendor/fontawesome-free/svgs/brands/docker.svg +1 -0
  463. package/static/vendor/fontawesome-free/svgs/brands/draft2digital.svg +1 -0
  464. package/static/vendor/fontawesome-free/svgs/brands/dribbble-square.svg +1 -0
  465. package/static/vendor/fontawesome-free/svgs/brands/dribbble.svg +1 -0
  466. package/static/vendor/fontawesome-free/svgs/brands/dropbox.svg +1 -0
  467. package/static/vendor/fontawesome-free/svgs/brands/drupal.svg +1 -0
  468. package/static/vendor/fontawesome-free/svgs/brands/dyalog.svg +1 -0
  469. package/static/vendor/fontawesome-free/svgs/brands/earlybirds.svg +1 -0
  470. package/static/vendor/fontawesome-free/svgs/brands/ebay.svg +1 -0
  471. package/static/vendor/fontawesome-free/svgs/brands/edge-legacy.svg +1 -0
  472. package/static/vendor/fontawesome-free/svgs/brands/edge.svg +1 -0
  473. package/static/vendor/fontawesome-free/svgs/brands/elementor.svg +1 -0
  474. package/static/vendor/fontawesome-free/svgs/brands/ello.svg +1 -0
  475. package/static/vendor/fontawesome-free/svgs/brands/ember.svg +1 -0
  476. package/static/vendor/fontawesome-free/svgs/brands/empire.svg +1 -0
  477. package/static/vendor/fontawesome-free/svgs/brands/envira.svg +1 -0
  478. package/static/vendor/fontawesome-free/svgs/brands/erlang.svg +1 -0
  479. package/static/vendor/fontawesome-free/svgs/brands/ethereum.svg +1 -0
  480. package/static/vendor/fontawesome-free/svgs/brands/etsy.svg +1 -0
  481. package/static/vendor/fontawesome-free/svgs/brands/evernote.svg +1 -0
  482. package/static/vendor/fontawesome-free/svgs/brands/expeditedssl.svg +1 -0
  483. package/static/vendor/fontawesome-free/svgs/brands/facebook-f.svg +1 -0
  484. package/static/vendor/fontawesome-free/svgs/brands/facebook-messenger.svg +1 -0
  485. package/static/vendor/fontawesome-free/svgs/brands/facebook-square.svg +1 -0
  486. package/static/vendor/fontawesome-free/svgs/brands/facebook.svg +1 -0
  487. package/static/vendor/fontawesome-free/svgs/brands/fantasy-flight-games.svg +1 -0
  488. package/static/vendor/fontawesome-free/svgs/brands/fedex.svg +1 -0
  489. package/static/vendor/fontawesome-free/svgs/brands/fedora.svg +1 -0
  490. package/static/vendor/fontawesome-free/svgs/brands/figma.svg +1 -0
  491. package/static/vendor/fontawesome-free/svgs/brands/firefox-browser.svg +1 -0
  492. package/static/vendor/fontawesome-free/svgs/brands/firefox.svg +1 -0
  493. package/static/vendor/fontawesome-free/svgs/brands/first-order-alt.svg +1 -0
  494. package/static/vendor/fontawesome-free/svgs/brands/first-order.svg +1 -0
  495. package/static/vendor/fontawesome-free/svgs/brands/firstdraft.svg +1 -0
  496. package/static/vendor/fontawesome-free/svgs/brands/flickr.svg +1 -0
  497. package/static/vendor/fontawesome-free/svgs/brands/flipboard.svg +1 -0
  498. package/static/vendor/fontawesome-free/svgs/brands/fly.svg +1 -0
  499. package/static/vendor/fontawesome-free/svgs/brands/font-awesome-alt.svg +1 -0
  500. package/static/vendor/fontawesome-free/svgs/brands/font-awesome-flag.svg +1 -0
  501. package/static/vendor/fontawesome-free/svgs/brands/font-awesome-logo-full.svg +1 -0
  502. package/static/vendor/fontawesome-free/svgs/brands/font-awesome.svg +1 -0
  503. package/static/vendor/fontawesome-free/svgs/brands/fonticons-fi.svg +1 -0
  504. package/static/vendor/fontawesome-free/svgs/brands/fonticons.svg +1 -0
  505. package/static/vendor/fontawesome-free/svgs/brands/fort-awesome-alt.svg +1 -0
  506. package/static/vendor/fontawesome-free/svgs/brands/fort-awesome.svg +1 -0
  507. package/static/vendor/fontawesome-free/svgs/brands/forumbee.svg +1 -0
  508. package/static/vendor/fontawesome-free/svgs/brands/foursquare.svg +1 -0
  509. package/static/vendor/fontawesome-free/svgs/brands/free-code-camp.svg +1 -0
  510. package/static/vendor/fontawesome-free/svgs/brands/freebsd.svg +1 -0
  511. package/static/vendor/fontawesome-free/svgs/brands/fulcrum.svg +1 -0
  512. package/static/vendor/fontawesome-free/svgs/brands/galactic-republic.svg +1 -0
  513. package/static/vendor/fontawesome-free/svgs/brands/galactic-senate.svg +1 -0
  514. package/static/vendor/fontawesome-free/svgs/brands/get-pocket.svg +1 -0
  515. package/static/vendor/fontawesome-free/svgs/brands/gg-circle.svg +1 -0
  516. package/static/vendor/fontawesome-free/svgs/brands/gg.svg +1 -0
  517. package/static/vendor/fontawesome-free/svgs/brands/git-alt.svg +1 -0
  518. package/static/vendor/fontawesome-free/svgs/brands/git-square.svg +1 -0
  519. package/static/vendor/fontawesome-free/svgs/brands/git.svg +1 -0
  520. package/static/vendor/fontawesome-free/svgs/brands/github-alt.svg +1 -0
  521. package/static/vendor/fontawesome-free/svgs/brands/github-square.svg +1 -0
  522. package/static/vendor/fontawesome-free/svgs/brands/github.svg +1 -0
  523. package/static/vendor/fontawesome-free/svgs/brands/gitkraken.svg +1 -0
  524. package/static/vendor/fontawesome-free/svgs/brands/gitlab.svg +1 -0
  525. package/static/vendor/fontawesome-free/svgs/brands/gitter.svg +1 -0
  526. package/static/vendor/fontawesome-free/svgs/brands/glide-g.svg +1 -0
  527. package/static/vendor/fontawesome-free/svgs/brands/glide.svg +1 -0
  528. package/static/vendor/fontawesome-free/svgs/brands/gofore.svg +1 -0
  529. package/static/vendor/fontawesome-free/svgs/brands/goodreads-g.svg +1 -0
  530. package/static/vendor/fontawesome-free/svgs/brands/goodreads.svg +1 -0
  531. package/static/vendor/fontawesome-free/svgs/brands/google-drive.svg +1 -0
  532. package/static/vendor/fontawesome-free/svgs/brands/google-pay.svg +1 -0
  533. package/static/vendor/fontawesome-free/svgs/brands/google-play.svg +1 -0
  534. package/static/vendor/fontawesome-free/svgs/brands/google-plus-g.svg +1 -0
  535. package/static/vendor/fontawesome-free/svgs/brands/google-plus-square.svg +1 -0
  536. package/static/vendor/fontawesome-free/svgs/brands/google-plus.svg +1 -0
  537. package/static/vendor/fontawesome-free/svgs/brands/google-wallet.svg +1 -0
  538. package/static/vendor/fontawesome-free/svgs/brands/google.svg +1 -0
  539. package/static/vendor/fontawesome-free/svgs/brands/gratipay.svg +1 -0
  540. package/static/vendor/fontawesome-free/svgs/brands/grav.svg +1 -0
  541. package/static/vendor/fontawesome-free/svgs/brands/gripfire.svg +1 -0
  542. package/static/vendor/fontawesome-free/svgs/brands/grunt.svg +1 -0
  543. package/static/vendor/fontawesome-free/svgs/brands/guilded.svg +1 -0
  544. package/static/vendor/fontawesome-free/svgs/brands/gulp.svg +1 -0
  545. package/static/vendor/fontawesome-free/svgs/brands/hacker-news-square.svg +1 -0
  546. package/static/vendor/fontawesome-free/svgs/brands/hacker-news.svg +1 -0
  547. package/static/vendor/fontawesome-free/svgs/brands/hackerrank.svg +1 -0
  548. package/static/vendor/fontawesome-free/svgs/brands/hips.svg +1 -0
  549. package/static/vendor/fontawesome-free/svgs/brands/hire-a-helper.svg +1 -0
  550. package/static/vendor/fontawesome-free/svgs/brands/hive.svg +1 -0
  551. package/static/vendor/fontawesome-free/svgs/brands/hooli.svg +1 -0
  552. package/static/vendor/fontawesome-free/svgs/brands/hornbill.svg +1 -0
  553. package/static/vendor/fontawesome-free/svgs/brands/hotjar.svg +1 -0
  554. package/static/vendor/fontawesome-free/svgs/brands/houzz.svg +1 -0
  555. package/static/vendor/fontawesome-free/svgs/brands/html5.svg +1 -0
  556. package/static/vendor/fontawesome-free/svgs/brands/hubspot.svg +1 -0
  557. package/static/vendor/fontawesome-free/svgs/brands/ideal.svg +1 -0
  558. package/static/vendor/fontawesome-free/svgs/brands/imdb.svg +1 -0
  559. package/static/vendor/fontawesome-free/svgs/brands/innosoft.svg +1 -0
  560. package/static/vendor/fontawesome-free/svgs/brands/instagram-square.svg +1 -0
  561. package/static/vendor/fontawesome-free/svgs/brands/instagram.svg +1 -0
  562. package/static/vendor/fontawesome-free/svgs/brands/instalod.svg +1 -0
  563. package/static/vendor/fontawesome-free/svgs/brands/intercom.svg +1 -0
  564. package/static/vendor/fontawesome-free/svgs/brands/internet-explorer.svg +1 -0
  565. package/static/vendor/fontawesome-free/svgs/brands/invision.svg +1 -0
  566. package/static/vendor/fontawesome-free/svgs/brands/ioxhost.svg +1 -0
  567. package/static/vendor/fontawesome-free/svgs/brands/itch-io.svg +1 -0
  568. package/static/vendor/fontawesome-free/svgs/brands/itunes-note.svg +1 -0
  569. package/static/vendor/fontawesome-free/svgs/brands/itunes.svg +1 -0
  570. package/static/vendor/fontawesome-free/svgs/brands/java.svg +1 -0
  571. package/static/vendor/fontawesome-free/svgs/brands/jedi-order.svg +1 -0
  572. package/static/vendor/fontawesome-free/svgs/brands/jenkins.svg +1 -0
  573. package/static/vendor/fontawesome-free/svgs/brands/jira.svg +1 -0
  574. package/static/vendor/fontawesome-free/svgs/brands/joget.svg +1 -0
  575. package/static/vendor/fontawesome-free/svgs/brands/joomla.svg +1 -0
  576. package/static/vendor/fontawesome-free/svgs/brands/js-square.svg +1 -0
  577. package/static/vendor/fontawesome-free/svgs/brands/js.svg +1 -0
  578. package/static/vendor/fontawesome-free/svgs/brands/jsfiddle.svg +1 -0
  579. package/static/vendor/fontawesome-free/svgs/brands/kaggle.svg +1 -0
  580. package/static/vendor/fontawesome-free/svgs/brands/keybase.svg +1 -0
  581. package/static/vendor/fontawesome-free/svgs/brands/keycdn.svg +1 -0
  582. package/static/vendor/fontawesome-free/svgs/brands/kickstarter-k.svg +1 -0
  583. package/static/vendor/fontawesome-free/svgs/brands/kickstarter.svg +1 -0
  584. package/static/vendor/fontawesome-free/svgs/brands/korvue.svg +1 -0
  585. package/static/vendor/fontawesome-free/svgs/brands/laravel.svg +1 -0
  586. package/static/vendor/fontawesome-free/svgs/brands/lastfm-square.svg +1 -0
  587. package/static/vendor/fontawesome-free/svgs/brands/lastfm.svg +1 -0
  588. package/static/vendor/fontawesome-free/svgs/brands/leanpub.svg +1 -0
  589. package/static/vendor/fontawesome-free/svgs/brands/less.svg +1 -0
  590. package/static/vendor/fontawesome-free/svgs/brands/line.svg +1 -0
  591. package/static/vendor/fontawesome-free/svgs/brands/linkedin-in.svg +1 -0
  592. package/static/vendor/fontawesome-free/svgs/brands/linkedin.svg +1 -0
  593. package/static/vendor/fontawesome-free/svgs/brands/linode.svg +1 -0
  594. package/static/vendor/fontawesome-free/svgs/brands/linux.svg +1 -0
  595. package/static/vendor/fontawesome-free/svgs/brands/lyft.svg +1 -0
  596. package/static/vendor/fontawesome-free/svgs/brands/magento.svg +1 -0
  597. package/static/vendor/fontawesome-free/svgs/brands/mailchimp.svg +1 -0
  598. package/static/vendor/fontawesome-free/svgs/brands/mandalorian.svg +1 -0
  599. package/static/vendor/fontawesome-free/svgs/brands/markdown.svg +1 -0
  600. package/static/vendor/fontawesome-free/svgs/brands/mastodon.svg +1 -0
  601. package/static/vendor/fontawesome-free/svgs/brands/maxcdn.svg +1 -0
  602. package/static/vendor/fontawesome-free/svgs/brands/mdb.svg +1 -0
  603. package/static/vendor/fontawesome-free/svgs/brands/medapps.svg +1 -0
  604. package/static/vendor/fontawesome-free/svgs/brands/medium-m.svg +1 -0
  605. package/static/vendor/fontawesome-free/svgs/brands/medium.svg +1 -0
  606. package/static/vendor/fontawesome-free/svgs/brands/medrt.svg +1 -0
  607. package/static/vendor/fontawesome-free/svgs/brands/meetup.svg +1 -0
  608. package/static/vendor/fontawesome-free/svgs/brands/megaport.svg +1 -0
  609. package/static/vendor/fontawesome-free/svgs/brands/mendeley.svg +1 -0
  610. package/static/vendor/fontawesome-free/svgs/brands/microblog.svg +1 -0
  611. package/static/vendor/fontawesome-free/svgs/brands/microsoft.svg +1 -0
  612. package/static/vendor/fontawesome-free/svgs/brands/mix.svg +1 -0
  613. package/static/vendor/fontawesome-free/svgs/brands/mixcloud.svg +1 -0
  614. package/static/vendor/fontawesome-free/svgs/brands/mixer.svg +1 -0
  615. package/static/vendor/fontawesome-free/svgs/brands/mizuni.svg +1 -0
  616. package/static/vendor/fontawesome-free/svgs/brands/modx.svg +1 -0
  617. package/static/vendor/fontawesome-free/svgs/brands/monero.svg +1 -0
  618. package/static/vendor/fontawesome-free/svgs/brands/napster.svg +1 -0
  619. package/static/vendor/fontawesome-free/svgs/brands/neos.svg +1 -0
  620. package/static/vendor/fontawesome-free/svgs/brands/nimblr.svg +1 -0
  621. package/static/vendor/fontawesome-free/svgs/brands/node-js.svg +1 -0
  622. package/static/vendor/fontawesome-free/svgs/brands/node.svg +1 -0
  623. package/static/vendor/fontawesome-free/svgs/brands/npm.svg +1 -0
  624. package/static/vendor/fontawesome-free/svgs/brands/ns8.svg +1 -0
  625. package/static/vendor/fontawesome-free/svgs/brands/nutritionix.svg +1 -0
  626. package/static/vendor/fontawesome-free/svgs/brands/octopus-deploy.svg +1 -0
  627. package/static/vendor/fontawesome-free/svgs/brands/odnoklassniki-square.svg +1 -0
  628. package/static/vendor/fontawesome-free/svgs/brands/odnoklassniki.svg +1 -0
  629. package/static/vendor/fontawesome-free/svgs/brands/old-republic.svg +1 -0
  630. package/static/vendor/fontawesome-free/svgs/brands/opencart.svg +1 -0
  631. package/static/vendor/fontawesome-free/svgs/brands/openid.svg +1 -0
  632. package/static/vendor/fontawesome-free/svgs/brands/opera.svg +1 -0
  633. package/static/vendor/fontawesome-free/svgs/brands/optin-monster.svg +1 -0
  634. package/static/vendor/fontawesome-free/svgs/brands/orcid.svg +1 -0
  635. package/static/vendor/fontawesome-free/svgs/brands/osi.svg +1 -0
  636. package/static/vendor/fontawesome-free/svgs/brands/page4.svg +1 -0
  637. package/static/vendor/fontawesome-free/svgs/brands/pagelines.svg +1 -0
  638. package/static/vendor/fontawesome-free/svgs/brands/palfed.svg +1 -0
  639. package/static/vendor/fontawesome-free/svgs/brands/patreon.svg +1 -0
  640. package/static/vendor/fontawesome-free/svgs/brands/paypal.svg +1 -0
  641. package/static/vendor/fontawesome-free/svgs/brands/penny-arcade.svg +1 -0
  642. package/static/vendor/fontawesome-free/svgs/brands/perbyte.svg +1 -0
  643. package/static/vendor/fontawesome-free/svgs/brands/periscope.svg +1 -0
  644. package/static/vendor/fontawesome-free/svgs/brands/phabricator.svg +1 -0
  645. package/static/vendor/fontawesome-free/svgs/brands/phoenix-framework.svg +1 -0
  646. package/static/vendor/fontawesome-free/svgs/brands/phoenix-squadron.svg +1 -0
  647. package/static/vendor/fontawesome-free/svgs/brands/php.svg +1 -0
  648. package/static/vendor/fontawesome-free/svgs/brands/pied-piper-alt.svg +1 -0
  649. package/static/vendor/fontawesome-free/svgs/brands/pied-piper-hat.svg +1 -0
  650. package/static/vendor/fontawesome-free/svgs/brands/pied-piper-pp.svg +1 -0
  651. package/static/vendor/fontawesome-free/svgs/brands/pied-piper-square.svg +1 -0
  652. package/static/vendor/fontawesome-free/svgs/brands/pied-piper.svg +1 -0
  653. package/static/vendor/fontawesome-free/svgs/brands/pinterest-p.svg +1 -0
  654. package/static/vendor/fontawesome-free/svgs/brands/pinterest-square.svg +1 -0
  655. package/static/vendor/fontawesome-free/svgs/brands/pinterest.svg +1 -0
  656. package/static/vendor/fontawesome-free/svgs/brands/playstation.svg +1 -0
  657. package/static/vendor/fontawesome-free/svgs/brands/product-hunt.svg +1 -0
  658. package/static/vendor/fontawesome-free/svgs/brands/pushed.svg +1 -0
  659. package/static/vendor/fontawesome-free/svgs/brands/python.svg +1 -0
  660. package/static/vendor/fontawesome-free/svgs/brands/qq.svg +1 -0
  661. package/static/vendor/fontawesome-free/svgs/brands/quinscape.svg +1 -0
  662. package/static/vendor/fontawesome-free/svgs/brands/quora.svg +1 -0
  663. package/static/vendor/fontawesome-free/svgs/brands/r-project.svg +1 -0
  664. package/static/vendor/fontawesome-free/svgs/brands/raspberry-pi.svg +1 -0
  665. package/static/vendor/fontawesome-free/svgs/brands/ravelry.svg +1 -0
  666. package/static/vendor/fontawesome-free/svgs/brands/react.svg +1 -0
  667. package/static/vendor/fontawesome-free/svgs/brands/reacteurope.svg +1 -0
  668. package/static/vendor/fontawesome-free/svgs/brands/readme.svg +1 -0
  669. package/static/vendor/fontawesome-free/svgs/brands/rebel.svg +1 -0
  670. package/static/vendor/fontawesome-free/svgs/brands/red-river.svg +1 -0
  671. package/static/vendor/fontawesome-free/svgs/brands/reddit-alien.svg +1 -0
  672. package/static/vendor/fontawesome-free/svgs/brands/reddit-square.svg +1 -0
  673. package/static/vendor/fontawesome-free/svgs/brands/reddit.svg +1 -0
  674. package/static/vendor/fontawesome-free/svgs/brands/redhat.svg +1 -0
  675. package/static/vendor/fontawesome-free/svgs/brands/renren.svg +1 -0
  676. package/static/vendor/fontawesome-free/svgs/brands/replyd.svg +1 -0
  677. package/static/vendor/fontawesome-free/svgs/brands/researchgate.svg +1 -0
  678. package/static/vendor/fontawesome-free/svgs/brands/resolving.svg +1 -0
  679. package/static/vendor/fontawesome-free/svgs/brands/rev.svg +1 -0
  680. package/static/vendor/fontawesome-free/svgs/brands/rocketchat.svg +1 -0
  681. package/static/vendor/fontawesome-free/svgs/brands/rockrms.svg +1 -0
  682. package/static/vendor/fontawesome-free/svgs/brands/rust.svg +1 -0
  683. package/static/vendor/fontawesome-free/svgs/brands/safari.svg +1 -0
  684. package/static/vendor/fontawesome-free/svgs/brands/salesforce.svg +1 -0
  685. package/static/vendor/fontawesome-free/svgs/brands/sass.svg +1 -0
  686. package/static/vendor/fontawesome-free/svgs/brands/schlix.svg +1 -0
  687. package/static/vendor/fontawesome-free/svgs/brands/scribd.svg +1 -0
  688. package/static/vendor/fontawesome-free/svgs/brands/searchengin.svg +1 -0
  689. package/static/vendor/fontawesome-free/svgs/brands/sellcast.svg +1 -0
  690. package/static/vendor/fontawesome-free/svgs/brands/sellsy.svg +1 -0
  691. package/static/vendor/fontawesome-free/svgs/brands/servicestack.svg +1 -0
  692. package/static/vendor/fontawesome-free/svgs/brands/shirtsinbulk.svg +1 -0
  693. package/static/vendor/fontawesome-free/svgs/brands/shopify.svg +1 -0
  694. package/static/vendor/fontawesome-free/svgs/brands/shopware.svg +1 -0
  695. package/static/vendor/fontawesome-free/svgs/brands/simplybuilt.svg +1 -0
  696. package/static/vendor/fontawesome-free/svgs/brands/sistrix.svg +1 -0
  697. package/static/vendor/fontawesome-free/svgs/brands/sith.svg +1 -0
  698. package/static/vendor/fontawesome-free/svgs/brands/sketch.svg +1 -0
  699. package/static/vendor/fontawesome-free/svgs/brands/skyatlas.svg +1 -0
  700. package/static/vendor/fontawesome-free/svgs/brands/skype.svg +1 -0
  701. package/static/vendor/fontawesome-free/svgs/brands/slack-hash.svg +1 -0
  702. package/static/vendor/fontawesome-free/svgs/brands/slack.svg +1 -0
  703. package/static/vendor/fontawesome-free/svgs/brands/slideshare.svg +1 -0
  704. package/static/vendor/fontawesome-free/svgs/brands/snapchat-ghost.svg +1 -0
  705. package/static/vendor/fontawesome-free/svgs/brands/snapchat-square.svg +1 -0
  706. package/static/vendor/fontawesome-free/svgs/brands/snapchat.svg +1 -0
  707. package/static/vendor/fontawesome-free/svgs/brands/soundcloud.svg +1 -0
  708. package/static/vendor/fontawesome-free/svgs/brands/sourcetree.svg +1 -0
  709. package/static/vendor/fontawesome-free/svgs/brands/speakap.svg +1 -0
  710. package/static/vendor/fontawesome-free/svgs/brands/speaker-deck.svg +1 -0
  711. package/static/vendor/fontawesome-free/svgs/brands/spotify.svg +1 -0
  712. package/static/vendor/fontawesome-free/svgs/brands/squarespace.svg +1 -0
  713. package/static/vendor/fontawesome-free/svgs/brands/stack-exchange.svg +1 -0
  714. package/static/vendor/fontawesome-free/svgs/brands/stack-overflow.svg +1 -0
  715. package/static/vendor/fontawesome-free/svgs/brands/stackpath.svg +1 -0
  716. package/static/vendor/fontawesome-free/svgs/brands/staylinked.svg +1 -0
  717. package/static/vendor/fontawesome-free/svgs/brands/steam-square.svg +1 -0
  718. package/static/vendor/fontawesome-free/svgs/brands/steam-symbol.svg +1 -0
  719. package/static/vendor/fontawesome-free/svgs/brands/steam.svg +1 -0
  720. package/static/vendor/fontawesome-free/svgs/brands/sticker-mule.svg +1 -0
  721. package/static/vendor/fontawesome-free/svgs/brands/strava.svg +1 -0
  722. package/static/vendor/fontawesome-free/svgs/brands/stripe-s.svg +1 -0
  723. package/static/vendor/fontawesome-free/svgs/brands/stripe.svg +1 -0
  724. package/static/vendor/fontawesome-free/svgs/brands/studiovinari.svg +1 -0
  725. package/static/vendor/fontawesome-free/svgs/brands/stumbleupon-circle.svg +1 -0
  726. package/static/vendor/fontawesome-free/svgs/brands/stumbleupon.svg +1 -0
  727. package/static/vendor/fontawesome-free/svgs/brands/superpowers.svg +1 -0
  728. package/static/vendor/fontawesome-free/svgs/brands/supple.svg +1 -0
  729. package/static/vendor/fontawesome-free/svgs/brands/suse.svg +1 -0
  730. package/static/vendor/fontawesome-free/svgs/brands/swift.svg +1 -0
  731. package/static/vendor/fontawesome-free/svgs/brands/symfony.svg +1 -0
  732. package/static/vendor/fontawesome-free/svgs/brands/teamspeak.svg +1 -0
  733. package/static/vendor/fontawesome-free/svgs/brands/telegram-plane.svg +1 -0
  734. package/static/vendor/fontawesome-free/svgs/brands/telegram.svg +1 -0
  735. package/static/vendor/fontawesome-free/svgs/brands/tencent-weibo.svg +1 -0
  736. package/static/vendor/fontawesome-free/svgs/brands/the-red-yeti.svg +1 -0
  737. package/static/vendor/fontawesome-free/svgs/brands/themeco.svg +1 -0
  738. package/static/vendor/fontawesome-free/svgs/brands/themeisle.svg +1 -0
  739. package/static/vendor/fontawesome-free/svgs/brands/think-peaks.svg +1 -0
  740. package/static/vendor/fontawesome-free/svgs/brands/tiktok.svg +1 -0
  741. package/static/vendor/fontawesome-free/svgs/brands/trade-federation.svg +1 -0
  742. package/static/vendor/fontawesome-free/svgs/brands/trello.svg +1 -0
  743. package/static/vendor/fontawesome-free/svgs/brands/tripadvisor.svg +1 -0
  744. package/static/vendor/fontawesome-free/svgs/brands/tumblr-square.svg +1 -0
  745. package/static/vendor/fontawesome-free/svgs/brands/tumblr.svg +1 -0
  746. package/static/vendor/fontawesome-free/svgs/brands/twitch.svg +1 -0
  747. package/static/vendor/fontawesome-free/svgs/brands/twitter-square.svg +1 -0
  748. package/static/vendor/fontawesome-free/svgs/brands/twitter.svg +1 -0
  749. package/static/vendor/fontawesome-free/svgs/brands/typo3.svg +1 -0
  750. package/static/vendor/fontawesome-free/svgs/brands/uber.svg +1 -0
  751. package/static/vendor/fontawesome-free/svgs/brands/ubuntu.svg +1 -0
  752. package/static/vendor/fontawesome-free/svgs/brands/uikit.svg +1 -0
  753. package/static/vendor/fontawesome-free/svgs/brands/umbraco.svg +1 -0
  754. package/static/vendor/fontawesome-free/svgs/brands/uncharted.svg +1 -0
  755. package/static/vendor/fontawesome-free/svgs/brands/uniregistry.svg +1 -0
  756. package/static/vendor/fontawesome-free/svgs/brands/unity.svg +1 -0
  757. package/static/vendor/fontawesome-free/svgs/brands/unsplash.svg +1 -0
  758. package/static/vendor/fontawesome-free/svgs/brands/untappd.svg +1 -0
  759. package/static/vendor/fontawesome-free/svgs/brands/ups.svg +1 -0
  760. package/static/vendor/fontawesome-free/svgs/brands/usb.svg +1 -0
  761. package/static/vendor/fontawesome-free/svgs/brands/usps.svg +1 -0
  762. package/static/vendor/fontawesome-free/svgs/brands/ussunnah.svg +1 -0
  763. package/static/vendor/fontawesome-free/svgs/brands/vaadin.svg +1 -0
  764. package/static/vendor/fontawesome-free/svgs/brands/viacoin.svg +1 -0
  765. package/static/vendor/fontawesome-free/svgs/brands/viadeo-square.svg +1 -0
  766. package/static/vendor/fontawesome-free/svgs/brands/viadeo.svg +1 -0
  767. package/static/vendor/fontawesome-free/svgs/brands/viber.svg +1 -0
  768. package/static/vendor/fontawesome-free/svgs/brands/vimeo-square.svg +1 -0
  769. package/static/vendor/fontawesome-free/svgs/brands/vimeo-v.svg +1 -0
  770. package/static/vendor/fontawesome-free/svgs/brands/vimeo.svg +1 -0
  771. package/static/vendor/fontawesome-free/svgs/brands/vine.svg +1 -0
  772. package/static/vendor/fontawesome-free/svgs/brands/vk.svg +1 -0
  773. package/static/vendor/fontawesome-free/svgs/brands/vnv.svg +1 -0
  774. package/static/vendor/fontawesome-free/svgs/brands/vuejs.svg +1 -0
  775. package/static/vendor/fontawesome-free/svgs/brands/watchman-monitoring.svg +1 -0
  776. package/static/vendor/fontawesome-free/svgs/brands/waze.svg +1 -0
  777. package/static/vendor/fontawesome-free/svgs/brands/weebly.svg +1 -0
  778. package/static/vendor/fontawesome-free/svgs/brands/weibo.svg +1 -0
  779. package/static/vendor/fontawesome-free/svgs/brands/weixin.svg +1 -0
  780. package/static/vendor/fontawesome-free/svgs/brands/whatsapp-square.svg +1 -0
  781. package/static/vendor/fontawesome-free/svgs/brands/whatsapp.svg +1 -0
  782. package/static/vendor/fontawesome-free/svgs/brands/whmcs.svg +1 -0
  783. package/static/vendor/fontawesome-free/svgs/brands/wikipedia-w.svg +1 -0
  784. package/static/vendor/fontawesome-free/svgs/brands/windows.svg +1 -0
  785. package/static/vendor/fontawesome-free/svgs/brands/wix.svg +1 -0
  786. package/static/vendor/fontawesome-free/svgs/brands/wizards-of-the-coast.svg +1 -0
  787. package/static/vendor/fontawesome-free/svgs/brands/wodu.svg +1 -0
  788. package/static/vendor/fontawesome-free/svgs/brands/wolf-pack-battalion.svg +1 -0
  789. package/static/vendor/fontawesome-free/svgs/brands/wordpress-simple.svg +1 -0
  790. package/static/vendor/fontawesome-free/svgs/brands/wordpress.svg +1 -0
  791. package/static/vendor/fontawesome-free/svgs/brands/wpbeginner.svg +1 -0
  792. package/static/vendor/fontawesome-free/svgs/brands/wpexplorer.svg +1 -0
  793. package/static/vendor/fontawesome-free/svgs/brands/wpforms.svg +1 -0
  794. package/static/vendor/fontawesome-free/svgs/brands/wpressr.svg +1 -0
  795. package/static/vendor/fontawesome-free/svgs/brands/xbox.svg +1 -0
  796. package/static/vendor/fontawesome-free/svgs/brands/xing-square.svg +1 -0
  797. package/static/vendor/fontawesome-free/svgs/brands/xing.svg +1 -0
  798. package/static/vendor/fontawesome-free/svgs/brands/y-combinator.svg +1 -0
  799. package/static/vendor/fontawesome-free/svgs/brands/yahoo.svg +1 -0
  800. package/static/vendor/fontawesome-free/svgs/brands/yammer.svg +1 -0
  801. package/static/vendor/fontawesome-free/svgs/brands/yandex-international.svg +1 -0
  802. package/static/vendor/fontawesome-free/svgs/brands/yandex.svg +1 -0
  803. package/static/vendor/fontawesome-free/svgs/brands/yarn.svg +1 -0
  804. package/static/vendor/fontawesome-free/svgs/brands/yelp.svg +1 -0
  805. package/static/vendor/fontawesome-free/svgs/brands/yoast.svg +1 -0
  806. package/static/vendor/fontawesome-free/svgs/brands/youtube-square.svg +1 -0
  807. package/static/vendor/fontawesome-free/svgs/brands/youtube.svg +1 -0
  808. package/static/vendor/fontawesome-free/svgs/brands/zhihu.svg +1 -0
  809. package/static/vendor/fontawesome-free/svgs/regular/address-book.svg +1 -0
  810. package/static/vendor/fontawesome-free/svgs/regular/address-card.svg +1 -0
  811. package/static/vendor/fontawesome-free/svgs/regular/angry.svg +1 -0
  812. package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-down.svg +1 -0
  813. package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-left.svg +1 -0
  814. package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-right.svg +1 -0
  815. package/static/vendor/fontawesome-free/svgs/regular/arrow-alt-circle-up.svg +1 -0
  816. package/static/vendor/fontawesome-free/svgs/regular/bell-slash.svg +1 -0
  817. package/static/vendor/fontawesome-free/svgs/regular/bell.svg +1 -0
  818. package/static/vendor/fontawesome-free/svgs/regular/bookmark.svg +1 -0
  819. package/static/vendor/fontawesome-free/svgs/regular/building.svg +1 -0
  820. package/static/vendor/fontawesome-free/svgs/regular/calendar-alt.svg +1 -0
  821. package/static/vendor/fontawesome-free/svgs/regular/calendar-check.svg +1 -0
  822. package/static/vendor/fontawesome-free/svgs/regular/calendar-minus.svg +1 -0
  823. package/static/vendor/fontawesome-free/svgs/regular/calendar-plus.svg +1 -0
  824. package/static/vendor/fontawesome-free/svgs/regular/calendar-times.svg +1 -0
  825. package/static/vendor/fontawesome-free/svgs/regular/calendar.svg +1 -0
  826. package/static/vendor/fontawesome-free/svgs/regular/caret-square-down.svg +1 -0
  827. package/static/vendor/fontawesome-free/svgs/regular/caret-square-left.svg +1 -0
  828. package/static/vendor/fontawesome-free/svgs/regular/caret-square-right.svg +1 -0
  829. package/static/vendor/fontawesome-free/svgs/regular/caret-square-up.svg +1 -0
  830. package/static/vendor/fontawesome-free/svgs/regular/chart-bar.svg +1 -0
  831. package/static/vendor/fontawesome-free/svgs/regular/check-circle.svg +1 -0
  832. package/static/vendor/fontawesome-free/svgs/regular/check-square.svg +1 -0
  833. package/static/vendor/fontawesome-free/svgs/regular/circle.svg +1 -0
  834. package/static/vendor/fontawesome-free/svgs/regular/clipboard.svg +1 -0
  835. package/static/vendor/fontawesome-free/svgs/regular/clock.svg +1 -0
  836. package/static/vendor/fontawesome-free/svgs/regular/clone.svg +1 -0
  837. package/static/vendor/fontawesome-free/svgs/regular/closed-captioning.svg +1 -0
  838. package/static/vendor/fontawesome-free/svgs/regular/comment-alt.svg +1 -0
  839. package/static/vendor/fontawesome-free/svgs/regular/comment-dots.svg +1 -0
  840. package/static/vendor/fontawesome-free/svgs/regular/comment.svg +1 -0
  841. package/static/vendor/fontawesome-free/svgs/regular/comments.svg +1 -0
  842. package/static/vendor/fontawesome-free/svgs/regular/compass.svg +1 -0
  843. package/static/vendor/fontawesome-free/svgs/regular/copy.svg +1 -0
  844. package/static/vendor/fontawesome-free/svgs/regular/copyright.svg +1 -0
  845. package/static/vendor/fontawesome-free/svgs/regular/credit-card.svg +1 -0
  846. package/static/vendor/fontawesome-free/svgs/regular/dizzy.svg +1 -0
  847. package/static/vendor/fontawesome-free/svgs/regular/dot-circle.svg +1 -0
  848. package/static/vendor/fontawesome-free/svgs/regular/edit.svg +1 -0
  849. package/static/vendor/fontawesome-free/svgs/regular/envelope-open.svg +1 -0
  850. package/static/vendor/fontawesome-free/svgs/regular/envelope.svg +1 -0
  851. package/static/vendor/fontawesome-free/svgs/regular/eye-slash.svg +1 -0
  852. package/static/vendor/fontawesome-free/svgs/regular/eye.svg +1 -0
  853. package/static/vendor/fontawesome-free/svgs/regular/file-alt.svg +1 -0
  854. package/static/vendor/fontawesome-free/svgs/regular/file-archive.svg +1 -0
  855. package/static/vendor/fontawesome-free/svgs/regular/file-audio.svg +1 -0
  856. package/static/vendor/fontawesome-free/svgs/regular/file-code.svg +1 -0
  857. package/static/vendor/fontawesome-free/svgs/regular/file-excel.svg +1 -0
  858. package/static/vendor/fontawesome-free/svgs/regular/file-image.svg +1 -0
  859. package/static/vendor/fontawesome-free/svgs/regular/file-pdf.svg +1 -0
  860. package/static/vendor/fontawesome-free/svgs/regular/file-powerpoint.svg +1 -0
  861. package/static/vendor/fontawesome-free/svgs/regular/file-video.svg +1 -0
  862. package/static/vendor/fontawesome-free/svgs/regular/file-word.svg +1 -0
  863. package/static/vendor/fontawesome-free/svgs/regular/file.svg +1 -0
  864. package/static/vendor/fontawesome-free/svgs/regular/flag.svg +1 -0
  865. package/static/vendor/fontawesome-free/svgs/regular/flushed.svg +1 -0
  866. package/static/vendor/fontawesome-free/svgs/regular/folder-open.svg +1 -0
  867. package/static/vendor/fontawesome-free/svgs/regular/folder.svg +1 -0
  868. package/static/vendor/fontawesome-free/svgs/regular/font-awesome-logo-full.svg +1 -0
  869. package/static/vendor/fontawesome-free/svgs/regular/frown-open.svg +1 -0
  870. package/static/vendor/fontawesome-free/svgs/regular/frown.svg +1 -0
  871. package/static/vendor/fontawesome-free/svgs/regular/futbol.svg +1 -0
  872. package/static/vendor/fontawesome-free/svgs/regular/gem.svg +1 -0
  873. package/static/vendor/fontawesome-free/svgs/regular/grimace.svg +1 -0
  874. package/static/vendor/fontawesome-free/svgs/regular/grin-alt.svg +1 -0
  875. package/static/vendor/fontawesome-free/svgs/regular/grin-beam-sweat.svg +1 -0
  876. package/static/vendor/fontawesome-free/svgs/regular/grin-beam.svg +1 -0
  877. package/static/vendor/fontawesome-free/svgs/regular/grin-hearts.svg +1 -0
  878. package/static/vendor/fontawesome-free/svgs/regular/grin-squint-tears.svg +1 -0
  879. package/static/vendor/fontawesome-free/svgs/regular/grin-squint.svg +1 -0
  880. package/static/vendor/fontawesome-free/svgs/regular/grin-stars.svg +1 -0
  881. package/static/vendor/fontawesome-free/svgs/regular/grin-tears.svg +1 -0
  882. package/static/vendor/fontawesome-free/svgs/regular/grin-tongue-squint.svg +1 -0
  883. package/static/vendor/fontawesome-free/svgs/regular/grin-tongue-wink.svg +1 -0
  884. package/static/vendor/fontawesome-free/svgs/regular/grin-tongue.svg +1 -0
  885. package/static/vendor/fontawesome-free/svgs/regular/grin-wink.svg +1 -0
  886. package/static/vendor/fontawesome-free/svgs/regular/grin.svg +1 -0
  887. package/static/vendor/fontawesome-free/svgs/regular/hand-lizard.svg +1 -0
  888. package/static/vendor/fontawesome-free/svgs/regular/hand-paper.svg +1 -0
  889. package/static/vendor/fontawesome-free/svgs/regular/hand-peace.svg +1 -0
  890. package/static/vendor/fontawesome-free/svgs/regular/hand-point-down.svg +1 -0
  891. package/static/vendor/fontawesome-free/svgs/regular/hand-point-left.svg +1 -0
  892. package/static/vendor/fontawesome-free/svgs/regular/hand-point-right.svg +1 -0
  893. package/static/vendor/fontawesome-free/svgs/regular/hand-point-up.svg +1 -0
  894. package/static/vendor/fontawesome-free/svgs/regular/hand-pointer.svg +1 -0
  895. package/static/vendor/fontawesome-free/svgs/regular/hand-rock.svg +1 -0
  896. package/static/vendor/fontawesome-free/svgs/regular/hand-scissors.svg +1 -0
  897. package/static/vendor/fontawesome-free/svgs/regular/hand-spock.svg +1 -0
  898. package/static/vendor/fontawesome-free/svgs/regular/handshake.svg +1 -0
  899. package/static/vendor/fontawesome-free/svgs/regular/hdd.svg +1 -0
  900. package/static/vendor/fontawesome-free/svgs/regular/heart.svg +1 -0
  901. package/static/vendor/fontawesome-free/svgs/regular/hospital.svg +1 -0
  902. package/static/vendor/fontawesome-free/svgs/regular/hourglass.svg +1 -0
  903. package/static/vendor/fontawesome-free/svgs/regular/id-badge.svg +1 -0
  904. package/static/vendor/fontawesome-free/svgs/regular/id-card.svg +1 -0
  905. package/static/vendor/fontawesome-free/svgs/regular/image.svg +1 -0
  906. package/static/vendor/fontawesome-free/svgs/regular/images.svg +1 -0
  907. package/static/vendor/fontawesome-free/svgs/regular/keyboard.svg +1 -0
  908. package/static/vendor/fontawesome-free/svgs/regular/kiss-beam.svg +1 -0
  909. package/static/vendor/fontawesome-free/svgs/regular/kiss-wink-heart.svg +1 -0
  910. package/static/vendor/fontawesome-free/svgs/regular/kiss.svg +1 -0
  911. package/static/vendor/fontawesome-free/svgs/regular/laugh-beam.svg +1 -0
  912. package/static/vendor/fontawesome-free/svgs/regular/laugh-squint.svg +1 -0
  913. package/static/vendor/fontawesome-free/svgs/regular/laugh-wink.svg +1 -0
  914. package/static/vendor/fontawesome-free/svgs/regular/laugh.svg +1 -0
  915. package/static/vendor/fontawesome-free/svgs/regular/lemon.svg +1 -0
  916. package/static/vendor/fontawesome-free/svgs/regular/life-ring.svg +1 -0
  917. package/static/vendor/fontawesome-free/svgs/regular/lightbulb.svg +1 -0
  918. package/static/vendor/fontawesome-free/svgs/regular/list-alt.svg +1 -0
  919. package/static/vendor/fontawesome-free/svgs/regular/map.svg +1 -0
  920. package/static/vendor/fontawesome-free/svgs/regular/meh-blank.svg +1 -0
  921. package/static/vendor/fontawesome-free/svgs/regular/meh-rolling-eyes.svg +1 -0
  922. package/static/vendor/fontawesome-free/svgs/regular/meh.svg +1 -0
  923. package/static/vendor/fontawesome-free/svgs/regular/minus-square.svg +1 -0
  924. package/static/vendor/fontawesome-free/svgs/regular/money-bill-alt.svg +1 -0
  925. package/static/vendor/fontawesome-free/svgs/regular/moon.svg +1 -0
  926. package/static/vendor/fontawesome-free/svgs/regular/newspaper.svg +1 -0
  927. package/static/vendor/fontawesome-free/svgs/regular/object-group.svg +1 -0
  928. package/static/vendor/fontawesome-free/svgs/regular/object-ungroup.svg +1 -0
  929. package/static/vendor/fontawesome-free/svgs/regular/paper-plane.svg +1 -0
  930. package/static/vendor/fontawesome-free/svgs/regular/pause-circle.svg +1 -0
  931. package/static/vendor/fontawesome-free/svgs/regular/play-circle.svg +1 -0
  932. package/static/vendor/fontawesome-free/svgs/regular/plus-square.svg +1 -0
  933. package/static/vendor/fontawesome-free/svgs/regular/question-circle.svg +1 -0
  934. package/static/vendor/fontawesome-free/svgs/regular/registered.svg +1 -0
  935. package/static/vendor/fontawesome-free/svgs/regular/sad-cry.svg +1 -0
  936. package/static/vendor/fontawesome-free/svgs/regular/sad-tear.svg +1 -0
  937. package/static/vendor/fontawesome-free/svgs/regular/save.svg +1 -0
  938. package/static/vendor/fontawesome-free/svgs/regular/share-square.svg +1 -0
  939. package/static/vendor/fontawesome-free/svgs/regular/smile-beam.svg +1 -0
  940. package/static/vendor/fontawesome-free/svgs/regular/smile-wink.svg +1 -0
  941. package/static/vendor/fontawesome-free/svgs/regular/smile.svg +1 -0
  942. package/static/vendor/fontawesome-free/svgs/regular/snowflake.svg +1 -0
  943. package/static/vendor/fontawesome-free/svgs/regular/square.svg +1 -0
  944. package/static/vendor/fontawesome-free/svgs/regular/star-half.svg +1 -0
  945. package/static/vendor/fontawesome-free/svgs/regular/star.svg +1 -0
  946. package/static/vendor/fontawesome-free/svgs/regular/sticky-note.svg +1 -0
  947. package/static/vendor/fontawesome-free/svgs/regular/stop-circle.svg +1 -0
  948. package/static/vendor/fontawesome-free/svgs/regular/sun.svg +1 -0
  949. package/static/vendor/fontawesome-free/svgs/regular/surprise.svg +1 -0
  950. package/static/vendor/fontawesome-free/svgs/regular/thumbs-down.svg +1 -0
  951. package/static/vendor/fontawesome-free/svgs/regular/thumbs-up.svg +1 -0
  952. package/static/vendor/fontawesome-free/svgs/regular/times-circle.svg +1 -0
  953. package/static/vendor/fontawesome-free/svgs/regular/tired.svg +1 -0
  954. package/static/vendor/fontawesome-free/svgs/regular/trash-alt.svg +1 -0
  955. package/static/vendor/fontawesome-free/svgs/regular/user-circle.svg +1 -0
  956. package/static/vendor/fontawesome-free/svgs/regular/user.svg +1 -0
  957. package/static/vendor/fontawesome-free/svgs/regular/window-close.svg +1 -0
  958. package/static/vendor/fontawesome-free/svgs/regular/window-maximize.svg +1 -0
  959. package/static/vendor/fontawesome-free/svgs/regular/window-minimize.svg +1 -0
  960. package/static/vendor/fontawesome-free/svgs/regular/window-restore.svg +1 -0
  961. package/static/vendor/fontawesome-free/svgs/solid/ad.svg +1 -0
  962. package/static/vendor/fontawesome-free/svgs/solid/address-book.svg +1 -0
  963. package/static/vendor/fontawesome-free/svgs/solid/address-card.svg +1 -0
  964. package/static/vendor/fontawesome-free/svgs/solid/adjust.svg +1 -0
  965. package/static/vendor/fontawesome-free/svgs/solid/air-freshener.svg +1 -0
  966. package/static/vendor/fontawesome-free/svgs/solid/align-center.svg +1 -0
  967. package/static/vendor/fontawesome-free/svgs/solid/align-justify.svg +1 -0
  968. package/static/vendor/fontawesome-free/svgs/solid/align-left.svg +1 -0
  969. package/static/vendor/fontawesome-free/svgs/solid/align-right.svg +1 -0
  970. package/static/vendor/fontawesome-free/svgs/solid/allergies.svg +1 -0
  971. package/static/vendor/fontawesome-free/svgs/solid/ambulance.svg +1 -0
  972. package/static/vendor/fontawesome-free/svgs/solid/american-sign-language-interpreting.svg +1 -0
  973. package/static/vendor/fontawesome-free/svgs/solid/anchor.svg +1 -0
  974. package/static/vendor/fontawesome-free/svgs/solid/angle-double-down.svg +1 -0
  975. package/static/vendor/fontawesome-free/svgs/solid/angle-double-left.svg +1 -0
  976. package/static/vendor/fontawesome-free/svgs/solid/angle-double-right.svg +1 -0
  977. package/static/vendor/fontawesome-free/svgs/solid/angle-double-up.svg +1 -0
  978. package/static/vendor/fontawesome-free/svgs/solid/angle-down.svg +1 -0
  979. package/static/vendor/fontawesome-free/svgs/solid/angle-left.svg +1 -0
  980. package/static/vendor/fontawesome-free/svgs/solid/angle-right.svg +1 -0
  981. package/static/vendor/fontawesome-free/svgs/solid/angle-up.svg +1 -0
  982. package/static/vendor/fontawesome-free/svgs/solid/angry.svg +1 -0
  983. package/static/vendor/fontawesome-free/svgs/solid/ankh.svg +1 -0
  984. package/static/vendor/fontawesome-free/svgs/solid/apple-alt.svg +1 -0
  985. package/static/vendor/fontawesome-free/svgs/solid/archive.svg +1 -0
  986. package/static/vendor/fontawesome-free/svgs/solid/archway.svg +1 -0
  987. package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-down.svg +1 -0
  988. package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-left.svg +1 -0
  989. package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-right.svg +1 -0
  990. package/static/vendor/fontawesome-free/svgs/solid/arrow-alt-circle-up.svg +1 -0
  991. package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-down.svg +1 -0
  992. package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-left.svg +1 -0
  993. package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-right.svg +1 -0
  994. package/static/vendor/fontawesome-free/svgs/solid/arrow-circle-up.svg +1 -0
  995. package/static/vendor/fontawesome-free/svgs/solid/arrow-down.svg +1 -0
  996. package/static/vendor/fontawesome-free/svgs/solid/arrow-left.svg +1 -0
  997. package/static/vendor/fontawesome-free/svgs/solid/arrow-right.svg +1 -0
  998. package/static/vendor/fontawesome-free/svgs/solid/arrow-up.svg +1 -0
  999. package/static/vendor/fontawesome-free/svgs/solid/arrows-alt-h.svg +1 -0
  1000. package/static/vendor/fontawesome-free/svgs/solid/arrows-alt-v.svg +1 -0
  1001. package/static/vendor/fontawesome-free/svgs/solid/arrows-alt.svg +1 -0
  1002. package/static/vendor/fontawesome-free/svgs/solid/assistive-listening-systems.svg +1 -0
  1003. package/static/vendor/fontawesome-free/svgs/solid/asterisk.svg +1 -0
  1004. package/static/vendor/fontawesome-free/svgs/solid/at.svg +1 -0
  1005. package/static/vendor/fontawesome-free/svgs/solid/atlas.svg +1 -0
  1006. package/static/vendor/fontawesome-free/svgs/solid/atom.svg +1 -0
  1007. package/static/vendor/fontawesome-free/svgs/solid/audio-description.svg +1 -0
  1008. package/static/vendor/fontawesome-free/svgs/solid/award.svg +1 -0
  1009. package/static/vendor/fontawesome-free/svgs/solid/baby-carriage.svg +1 -0
  1010. package/static/vendor/fontawesome-free/svgs/solid/baby.svg +1 -0
  1011. package/static/vendor/fontawesome-free/svgs/solid/backspace.svg +1 -0
  1012. package/static/vendor/fontawesome-free/svgs/solid/backward.svg +1 -0
  1013. package/static/vendor/fontawesome-free/svgs/solid/bacon.svg +1 -0
  1014. package/static/vendor/fontawesome-free/svgs/solid/bacteria.svg +1 -0
  1015. package/static/vendor/fontawesome-free/svgs/solid/bacterium.svg +1 -0
  1016. package/static/vendor/fontawesome-free/svgs/solid/bahai.svg +1 -0
  1017. package/static/vendor/fontawesome-free/svgs/solid/balance-scale-left.svg +1 -0
  1018. package/static/vendor/fontawesome-free/svgs/solid/balance-scale-right.svg +1 -0
  1019. package/static/vendor/fontawesome-free/svgs/solid/balance-scale.svg +1 -0
  1020. package/static/vendor/fontawesome-free/svgs/solid/ban.svg +1 -0
  1021. package/static/vendor/fontawesome-free/svgs/solid/band-aid.svg +1 -0
  1022. package/static/vendor/fontawesome-free/svgs/solid/barcode.svg +1 -0
  1023. package/static/vendor/fontawesome-free/svgs/solid/bars.svg +1 -0
  1024. package/static/vendor/fontawesome-free/svgs/solid/baseball-ball.svg +1 -0
  1025. package/static/vendor/fontawesome-free/svgs/solid/basketball-ball.svg +1 -0
  1026. package/static/vendor/fontawesome-free/svgs/solid/bath.svg +1 -0
  1027. package/static/vendor/fontawesome-free/svgs/solid/battery-empty.svg +1 -0
  1028. package/static/vendor/fontawesome-free/svgs/solid/battery-full.svg +1 -0
  1029. package/static/vendor/fontawesome-free/svgs/solid/battery-half.svg +1 -0
  1030. package/static/vendor/fontawesome-free/svgs/solid/battery-quarter.svg +1 -0
  1031. package/static/vendor/fontawesome-free/svgs/solid/battery-three-quarters.svg +1 -0
  1032. package/static/vendor/fontawesome-free/svgs/solid/bed.svg +1 -0
  1033. package/static/vendor/fontawesome-free/svgs/solid/beer.svg +1 -0
  1034. package/static/vendor/fontawesome-free/svgs/solid/bell-slash.svg +1 -0
  1035. package/static/vendor/fontawesome-free/svgs/solid/bell.svg +1 -0
  1036. package/static/vendor/fontawesome-free/svgs/solid/bezier-curve.svg +1 -0
  1037. package/static/vendor/fontawesome-free/svgs/solid/bible.svg +1 -0
  1038. package/static/vendor/fontawesome-free/svgs/solid/bicycle.svg +1 -0
  1039. package/static/vendor/fontawesome-free/svgs/solid/biking.svg +1 -0
  1040. package/static/vendor/fontawesome-free/svgs/solid/binoculars.svg +1 -0
  1041. package/static/vendor/fontawesome-free/svgs/solid/biohazard.svg +1 -0
  1042. package/static/vendor/fontawesome-free/svgs/solid/birthday-cake.svg +1 -0
  1043. package/static/vendor/fontawesome-free/svgs/solid/blender-phone.svg +1 -0
  1044. package/static/vendor/fontawesome-free/svgs/solid/blender.svg +1 -0
  1045. package/static/vendor/fontawesome-free/svgs/solid/blind.svg +1 -0
  1046. package/static/vendor/fontawesome-free/svgs/solid/blog.svg +1 -0
  1047. package/static/vendor/fontawesome-free/svgs/solid/bold.svg +1 -0
  1048. package/static/vendor/fontawesome-free/svgs/solid/bolt.svg +1 -0
  1049. package/static/vendor/fontawesome-free/svgs/solid/bomb.svg +1 -0
  1050. package/static/vendor/fontawesome-free/svgs/solid/bone.svg +1 -0
  1051. package/static/vendor/fontawesome-free/svgs/solid/bong.svg +1 -0
  1052. package/static/vendor/fontawesome-free/svgs/solid/book-dead.svg +1 -0
  1053. package/static/vendor/fontawesome-free/svgs/solid/book-medical.svg +1 -0
  1054. package/static/vendor/fontawesome-free/svgs/solid/book-open.svg +1 -0
  1055. package/static/vendor/fontawesome-free/svgs/solid/book-reader.svg +1 -0
  1056. package/static/vendor/fontawesome-free/svgs/solid/book.svg +1 -0
  1057. package/static/vendor/fontawesome-free/svgs/solid/bookmark.svg +1 -0
  1058. package/static/vendor/fontawesome-free/svgs/solid/border-all.svg +1 -0
  1059. package/static/vendor/fontawesome-free/svgs/solid/border-none.svg +1 -0
  1060. package/static/vendor/fontawesome-free/svgs/solid/border-style.svg +1 -0
  1061. package/static/vendor/fontawesome-free/svgs/solid/bowling-ball.svg +1 -0
  1062. package/static/vendor/fontawesome-free/svgs/solid/box-open.svg +1 -0
  1063. package/static/vendor/fontawesome-free/svgs/solid/box-tissue.svg +1 -0
  1064. package/static/vendor/fontawesome-free/svgs/solid/box.svg +1 -0
  1065. package/static/vendor/fontawesome-free/svgs/solid/boxes.svg +1 -0
  1066. package/static/vendor/fontawesome-free/svgs/solid/braille.svg +1 -0
  1067. package/static/vendor/fontawesome-free/svgs/solid/brain.svg +1 -0
  1068. package/static/vendor/fontawesome-free/svgs/solid/bread-slice.svg +1 -0
  1069. package/static/vendor/fontawesome-free/svgs/solid/briefcase-medical.svg +1 -0
  1070. package/static/vendor/fontawesome-free/svgs/solid/briefcase.svg +1 -0
  1071. package/static/vendor/fontawesome-free/svgs/solid/broadcast-tower.svg +1 -0
  1072. package/static/vendor/fontawesome-free/svgs/solid/broom.svg +1 -0
  1073. package/static/vendor/fontawesome-free/svgs/solid/brush.svg +1 -0
  1074. package/static/vendor/fontawesome-free/svgs/solid/bug.svg +1 -0
  1075. package/static/vendor/fontawesome-free/svgs/solid/building.svg +1 -0
  1076. package/static/vendor/fontawesome-free/svgs/solid/bullhorn.svg +1 -0
  1077. package/static/vendor/fontawesome-free/svgs/solid/bullseye.svg +1 -0
  1078. package/static/vendor/fontawesome-free/svgs/solid/burn.svg +1 -0
  1079. package/static/vendor/fontawesome-free/svgs/solid/bus-alt.svg +1 -0
  1080. package/static/vendor/fontawesome-free/svgs/solid/bus.svg +1 -0
  1081. package/static/vendor/fontawesome-free/svgs/solid/business-time.svg +1 -0
  1082. package/static/vendor/fontawesome-free/svgs/solid/calculator.svg +1 -0
  1083. package/static/vendor/fontawesome-free/svgs/solid/calendar-alt.svg +1 -0
  1084. package/static/vendor/fontawesome-free/svgs/solid/calendar-check.svg +1 -0
  1085. package/static/vendor/fontawesome-free/svgs/solid/calendar-day.svg +1 -0
  1086. package/static/vendor/fontawesome-free/svgs/solid/calendar-minus.svg +1 -0
  1087. package/static/vendor/fontawesome-free/svgs/solid/calendar-plus.svg +1 -0
  1088. package/static/vendor/fontawesome-free/svgs/solid/calendar-times.svg +1 -0
  1089. package/static/vendor/fontawesome-free/svgs/solid/calendar-week.svg +1 -0
  1090. package/static/vendor/fontawesome-free/svgs/solid/calendar.svg +1 -0
  1091. package/static/vendor/fontawesome-free/svgs/solid/camera-retro.svg +1 -0
  1092. package/static/vendor/fontawesome-free/svgs/solid/camera.svg +1 -0
  1093. package/static/vendor/fontawesome-free/svgs/solid/campground.svg +1 -0
  1094. package/static/vendor/fontawesome-free/svgs/solid/candy-cane.svg +1 -0
  1095. package/static/vendor/fontawesome-free/svgs/solid/cannabis.svg +1 -0
  1096. package/static/vendor/fontawesome-free/svgs/solid/capsules.svg +1 -0
  1097. package/static/vendor/fontawesome-free/svgs/solid/car-alt.svg +1 -0
  1098. package/static/vendor/fontawesome-free/svgs/solid/car-battery.svg +1 -0
  1099. package/static/vendor/fontawesome-free/svgs/solid/car-crash.svg +1 -0
  1100. package/static/vendor/fontawesome-free/svgs/solid/car-side.svg +1 -0
  1101. package/static/vendor/fontawesome-free/svgs/solid/car.svg +1 -0
  1102. package/static/vendor/fontawesome-free/svgs/solid/caravan.svg +1 -0
  1103. package/static/vendor/fontawesome-free/svgs/solid/caret-down.svg +1 -0
  1104. package/static/vendor/fontawesome-free/svgs/solid/caret-left.svg +1 -0
  1105. package/static/vendor/fontawesome-free/svgs/solid/caret-right.svg +1 -0
  1106. package/static/vendor/fontawesome-free/svgs/solid/caret-square-down.svg +1 -0
  1107. package/static/vendor/fontawesome-free/svgs/solid/caret-square-left.svg +1 -0
  1108. package/static/vendor/fontawesome-free/svgs/solid/caret-square-right.svg +1 -0
  1109. package/static/vendor/fontawesome-free/svgs/solid/caret-square-up.svg +1 -0
  1110. package/static/vendor/fontawesome-free/svgs/solid/caret-up.svg +1 -0
  1111. package/static/vendor/fontawesome-free/svgs/solid/carrot.svg +1 -0
  1112. package/static/vendor/fontawesome-free/svgs/solid/cart-arrow-down.svg +1 -0
  1113. package/static/vendor/fontawesome-free/svgs/solid/cart-plus.svg +1 -0
  1114. package/static/vendor/fontawesome-free/svgs/solid/cash-register.svg +1 -0
  1115. package/static/vendor/fontawesome-free/svgs/solid/cat.svg +1 -0
  1116. package/static/vendor/fontawesome-free/svgs/solid/certificate.svg +1 -0
  1117. package/static/vendor/fontawesome-free/svgs/solid/chair.svg +1 -0
  1118. package/static/vendor/fontawesome-free/svgs/solid/chalkboard-teacher.svg +1 -0
  1119. package/static/vendor/fontawesome-free/svgs/solid/chalkboard.svg +1 -0
  1120. package/static/vendor/fontawesome-free/svgs/solid/charging-station.svg +1 -0
  1121. package/static/vendor/fontawesome-free/svgs/solid/chart-area.svg +1 -0
  1122. package/static/vendor/fontawesome-free/svgs/solid/chart-bar.svg +1 -0
  1123. package/static/vendor/fontawesome-free/svgs/solid/chart-line.svg +1 -0
  1124. package/static/vendor/fontawesome-free/svgs/solid/chart-pie.svg +1 -0
  1125. package/static/vendor/fontawesome-free/svgs/solid/check-circle.svg +1 -0
  1126. package/static/vendor/fontawesome-free/svgs/solid/check-double.svg +1 -0
  1127. package/static/vendor/fontawesome-free/svgs/solid/check-square.svg +1 -0
  1128. package/static/vendor/fontawesome-free/svgs/solid/check.svg +1 -0
  1129. package/static/vendor/fontawesome-free/svgs/solid/cheese.svg +1 -0
  1130. package/static/vendor/fontawesome-free/svgs/solid/chess-bishop.svg +1 -0
  1131. package/static/vendor/fontawesome-free/svgs/solid/chess-board.svg +1 -0
  1132. package/static/vendor/fontawesome-free/svgs/solid/chess-king.svg +1 -0
  1133. package/static/vendor/fontawesome-free/svgs/solid/chess-knight.svg +1 -0
  1134. package/static/vendor/fontawesome-free/svgs/solid/chess-pawn.svg +1 -0
  1135. package/static/vendor/fontawesome-free/svgs/solid/chess-queen.svg +1 -0
  1136. package/static/vendor/fontawesome-free/svgs/solid/chess-rook.svg +1 -0
  1137. package/static/vendor/fontawesome-free/svgs/solid/chess.svg +1 -0
  1138. package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-down.svg +1 -0
  1139. package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-left.svg +1 -0
  1140. package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-right.svg +1 -0
  1141. package/static/vendor/fontawesome-free/svgs/solid/chevron-circle-up.svg +1 -0
  1142. package/static/vendor/fontawesome-free/svgs/solid/chevron-down.svg +1 -0
  1143. package/static/vendor/fontawesome-free/svgs/solid/chevron-left.svg +1 -0
  1144. package/static/vendor/fontawesome-free/svgs/solid/chevron-right.svg +1 -0
  1145. package/static/vendor/fontawesome-free/svgs/solid/chevron-up.svg +1 -0
  1146. package/static/vendor/fontawesome-free/svgs/solid/child.svg +1 -0
  1147. package/static/vendor/fontawesome-free/svgs/solid/church.svg +1 -0
  1148. package/static/vendor/fontawesome-free/svgs/solid/circle-notch.svg +1 -0
  1149. package/static/vendor/fontawesome-free/svgs/solid/circle.svg +1 -0
  1150. package/static/vendor/fontawesome-free/svgs/solid/city.svg +1 -0
  1151. package/static/vendor/fontawesome-free/svgs/solid/clinic-medical.svg +1 -0
  1152. package/static/vendor/fontawesome-free/svgs/solid/clipboard-check.svg +1 -0
  1153. package/static/vendor/fontawesome-free/svgs/solid/clipboard-list.svg +1 -0
  1154. package/static/vendor/fontawesome-free/svgs/solid/clipboard.svg +1 -0
  1155. package/static/vendor/fontawesome-free/svgs/solid/clock.svg +1 -0
  1156. package/static/vendor/fontawesome-free/svgs/solid/clone.svg +1 -0
  1157. package/static/vendor/fontawesome-free/svgs/solid/closed-captioning.svg +1 -0
  1158. package/static/vendor/fontawesome-free/svgs/solid/cloud-download-alt.svg +1 -0
  1159. package/static/vendor/fontawesome-free/svgs/solid/cloud-meatball.svg +1 -0
  1160. package/static/vendor/fontawesome-free/svgs/solid/cloud-moon-rain.svg +1 -0
  1161. package/static/vendor/fontawesome-free/svgs/solid/cloud-moon.svg +1 -0
  1162. package/static/vendor/fontawesome-free/svgs/solid/cloud-rain.svg +1 -0
  1163. package/static/vendor/fontawesome-free/svgs/solid/cloud-showers-heavy.svg +1 -0
  1164. package/static/vendor/fontawesome-free/svgs/solid/cloud-sun-rain.svg +1 -0
  1165. package/static/vendor/fontawesome-free/svgs/solid/cloud-sun.svg +1 -0
  1166. package/static/vendor/fontawesome-free/svgs/solid/cloud-upload-alt.svg +1 -0
  1167. package/static/vendor/fontawesome-free/svgs/solid/cloud.svg +1 -0
  1168. package/static/vendor/fontawesome-free/svgs/solid/cocktail.svg +1 -0
  1169. package/static/vendor/fontawesome-free/svgs/solid/code-branch.svg +1 -0
  1170. package/static/vendor/fontawesome-free/svgs/solid/code.svg +1 -0
  1171. package/static/vendor/fontawesome-free/svgs/solid/coffee.svg +1 -0
  1172. package/static/vendor/fontawesome-free/svgs/solid/cog.svg +1 -0
  1173. package/static/vendor/fontawesome-free/svgs/solid/cogs.svg +1 -0
  1174. package/static/vendor/fontawesome-free/svgs/solid/coins.svg +1 -0
  1175. package/static/vendor/fontawesome-free/svgs/solid/columns.svg +1 -0
  1176. package/static/vendor/fontawesome-free/svgs/solid/comment-alt.svg +1 -0
  1177. package/static/vendor/fontawesome-free/svgs/solid/comment-dollar.svg +1 -0
  1178. package/static/vendor/fontawesome-free/svgs/solid/comment-dots.svg +1 -0
  1179. package/static/vendor/fontawesome-free/svgs/solid/comment-medical.svg +1 -0
  1180. package/static/vendor/fontawesome-free/svgs/solid/comment-slash.svg +1 -0
  1181. package/static/vendor/fontawesome-free/svgs/solid/comment.svg +1 -0
  1182. package/static/vendor/fontawesome-free/svgs/solid/comments-dollar.svg +1 -0
  1183. package/static/vendor/fontawesome-free/svgs/solid/comments.svg +1 -0
  1184. package/static/vendor/fontawesome-free/svgs/solid/compact-disc.svg +1 -0
  1185. package/static/vendor/fontawesome-free/svgs/solid/compass.svg +1 -0
  1186. package/static/vendor/fontawesome-free/svgs/solid/compress-alt.svg +1 -0
  1187. package/static/vendor/fontawesome-free/svgs/solid/compress-arrows-alt.svg +1 -0
  1188. package/static/vendor/fontawesome-free/svgs/solid/compress.svg +1 -0
  1189. package/static/vendor/fontawesome-free/svgs/solid/concierge-bell.svg +1 -0
  1190. package/static/vendor/fontawesome-free/svgs/solid/cookie-bite.svg +1 -0
  1191. package/static/vendor/fontawesome-free/svgs/solid/cookie.svg +1 -0
  1192. package/static/vendor/fontawesome-free/svgs/solid/copy.svg +1 -0
  1193. package/static/vendor/fontawesome-free/svgs/solid/copyright.svg +1 -0
  1194. package/static/vendor/fontawesome-free/svgs/solid/couch.svg +1 -0
  1195. package/static/vendor/fontawesome-free/svgs/solid/credit-card.svg +1 -0
  1196. package/static/vendor/fontawesome-free/svgs/solid/crop-alt.svg +1 -0
  1197. package/static/vendor/fontawesome-free/svgs/solid/crop.svg +1 -0
  1198. package/static/vendor/fontawesome-free/svgs/solid/cross.svg +1 -0
  1199. package/static/vendor/fontawesome-free/svgs/solid/crosshairs.svg +1 -0
  1200. package/static/vendor/fontawesome-free/svgs/solid/crow.svg +1 -0
  1201. package/static/vendor/fontawesome-free/svgs/solid/crown.svg +1 -0
  1202. package/static/vendor/fontawesome-free/svgs/solid/crutch.svg +1 -0
  1203. package/static/vendor/fontawesome-free/svgs/solid/cube.svg +1 -0
  1204. package/static/vendor/fontawesome-free/svgs/solid/cubes.svg +1 -0
  1205. package/static/vendor/fontawesome-free/svgs/solid/cut.svg +1 -0
  1206. package/static/vendor/fontawesome-free/svgs/solid/database.svg +1 -0
  1207. package/static/vendor/fontawesome-free/svgs/solid/deaf.svg +1 -0
  1208. package/static/vendor/fontawesome-free/svgs/solid/democrat.svg +1 -0
  1209. package/static/vendor/fontawesome-free/svgs/solid/desktop.svg +1 -0
  1210. package/static/vendor/fontawesome-free/svgs/solid/dharmachakra.svg +1 -0
  1211. package/static/vendor/fontawesome-free/svgs/solid/diagnoses.svg +1 -0
  1212. package/static/vendor/fontawesome-free/svgs/solid/dice-d20.svg +1 -0
  1213. package/static/vendor/fontawesome-free/svgs/solid/dice-d6.svg +1 -0
  1214. package/static/vendor/fontawesome-free/svgs/solid/dice-five.svg +1 -0
  1215. package/static/vendor/fontawesome-free/svgs/solid/dice-four.svg +1 -0
  1216. package/static/vendor/fontawesome-free/svgs/solid/dice-one.svg +1 -0
  1217. package/static/vendor/fontawesome-free/svgs/solid/dice-six.svg +1 -0
  1218. package/static/vendor/fontawesome-free/svgs/solid/dice-three.svg +1 -0
  1219. package/static/vendor/fontawesome-free/svgs/solid/dice-two.svg +1 -0
  1220. package/static/vendor/fontawesome-free/svgs/solid/dice.svg +1 -0
  1221. package/static/vendor/fontawesome-free/svgs/solid/digital-tachograph.svg +1 -0
  1222. package/static/vendor/fontawesome-free/svgs/solid/directions.svg +1 -0
  1223. package/static/vendor/fontawesome-free/svgs/solid/disease.svg +1 -0
  1224. package/static/vendor/fontawesome-free/svgs/solid/divide.svg +1 -0
  1225. package/static/vendor/fontawesome-free/svgs/solid/dizzy.svg +1 -0
  1226. package/static/vendor/fontawesome-free/svgs/solid/dna.svg +1 -0
  1227. package/static/vendor/fontawesome-free/svgs/solid/dog.svg +1 -0
  1228. package/static/vendor/fontawesome-free/svgs/solid/dollar-sign.svg +1 -0
  1229. package/static/vendor/fontawesome-free/svgs/solid/dolly-flatbed.svg +1 -0
  1230. package/static/vendor/fontawesome-free/svgs/solid/dolly.svg +1 -0
  1231. package/static/vendor/fontawesome-free/svgs/solid/donate.svg +1 -0
  1232. package/static/vendor/fontawesome-free/svgs/solid/door-closed.svg +1 -0
  1233. package/static/vendor/fontawesome-free/svgs/solid/door-open.svg +1 -0
  1234. package/static/vendor/fontawesome-free/svgs/solid/dot-circle.svg +1 -0
  1235. package/static/vendor/fontawesome-free/svgs/solid/dove.svg +1 -0
  1236. package/static/vendor/fontawesome-free/svgs/solid/download.svg +1 -0
  1237. package/static/vendor/fontawesome-free/svgs/solid/drafting-compass.svg +1 -0
  1238. package/static/vendor/fontawesome-free/svgs/solid/dragon.svg +1 -0
  1239. package/static/vendor/fontawesome-free/svgs/solid/draw-polygon.svg +1 -0
  1240. package/static/vendor/fontawesome-free/svgs/solid/drum-steelpan.svg +1 -0
  1241. package/static/vendor/fontawesome-free/svgs/solid/drum.svg +1 -0
  1242. package/static/vendor/fontawesome-free/svgs/solid/drumstick-bite.svg +1 -0
  1243. package/static/vendor/fontawesome-free/svgs/solid/dumbbell.svg +1 -0
  1244. package/static/vendor/fontawesome-free/svgs/solid/dumpster-fire.svg +1 -0
  1245. package/static/vendor/fontawesome-free/svgs/solid/dumpster.svg +1 -0
  1246. package/static/vendor/fontawesome-free/svgs/solid/dungeon.svg +1 -0
  1247. package/static/vendor/fontawesome-free/svgs/solid/edit.svg +1 -0
  1248. package/static/vendor/fontawesome-free/svgs/solid/egg.svg +1 -0
  1249. package/static/vendor/fontawesome-free/svgs/solid/eject.svg +1 -0
  1250. package/static/vendor/fontawesome-free/svgs/solid/ellipsis-h.svg +1 -0
  1251. package/static/vendor/fontawesome-free/svgs/solid/ellipsis-v.svg +1 -0
  1252. package/static/vendor/fontawesome-free/svgs/solid/envelope-open-text.svg +1 -0
  1253. package/static/vendor/fontawesome-free/svgs/solid/envelope-open.svg +1 -0
  1254. package/static/vendor/fontawesome-free/svgs/solid/envelope-square.svg +1 -0
  1255. package/static/vendor/fontawesome-free/svgs/solid/envelope.svg +1 -0
  1256. package/static/vendor/fontawesome-free/svgs/solid/equals.svg +1 -0
  1257. package/static/vendor/fontawesome-free/svgs/solid/eraser.svg +1 -0
  1258. package/static/vendor/fontawesome-free/svgs/solid/ethernet.svg +1 -0
  1259. package/static/vendor/fontawesome-free/svgs/solid/euro-sign.svg +1 -0
  1260. package/static/vendor/fontawesome-free/svgs/solid/exchange-alt.svg +1 -0
  1261. package/static/vendor/fontawesome-free/svgs/solid/exclamation-circle.svg +1 -0
  1262. package/static/vendor/fontawesome-free/svgs/solid/exclamation-triangle.svg +1 -0
  1263. package/static/vendor/fontawesome-free/svgs/solid/exclamation.svg +1 -0
  1264. package/static/vendor/fontawesome-free/svgs/solid/expand-alt.svg +1 -0
  1265. package/static/vendor/fontawesome-free/svgs/solid/expand-arrows-alt.svg +1 -0
  1266. package/static/vendor/fontawesome-free/svgs/solid/expand.svg +1 -0
  1267. package/static/vendor/fontawesome-free/svgs/solid/external-link-alt.svg +1 -0
  1268. package/static/vendor/fontawesome-free/svgs/solid/external-link-square-alt.svg +1 -0
  1269. package/static/vendor/fontawesome-free/svgs/solid/eye-dropper.svg +1 -0
  1270. package/static/vendor/fontawesome-free/svgs/solid/eye-slash.svg +1 -0
  1271. package/static/vendor/fontawesome-free/svgs/solid/eye.svg +1 -0
  1272. package/static/vendor/fontawesome-free/svgs/solid/fan.svg +1 -0
  1273. package/static/vendor/fontawesome-free/svgs/solid/fast-backward.svg +1 -0
  1274. package/static/vendor/fontawesome-free/svgs/solid/fast-forward.svg +1 -0
  1275. package/static/vendor/fontawesome-free/svgs/solid/faucet.svg +1 -0
  1276. package/static/vendor/fontawesome-free/svgs/solid/fax.svg +1 -0
  1277. package/static/vendor/fontawesome-free/svgs/solid/feather-alt.svg +1 -0
  1278. package/static/vendor/fontawesome-free/svgs/solid/feather.svg +1 -0
  1279. package/static/vendor/fontawesome-free/svgs/solid/female.svg +1 -0
  1280. package/static/vendor/fontawesome-free/svgs/solid/fighter-jet.svg +1 -0
  1281. package/static/vendor/fontawesome-free/svgs/solid/file-alt.svg +1 -0
  1282. package/static/vendor/fontawesome-free/svgs/solid/file-archive.svg +1 -0
  1283. package/static/vendor/fontawesome-free/svgs/solid/file-audio.svg +1 -0
  1284. package/static/vendor/fontawesome-free/svgs/solid/file-code.svg +1 -0
  1285. package/static/vendor/fontawesome-free/svgs/solid/file-contract.svg +1 -0
  1286. package/static/vendor/fontawesome-free/svgs/solid/file-csv.svg +1 -0
  1287. package/static/vendor/fontawesome-free/svgs/solid/file-download.svg +1 -0
  1288. package/static/vendor/fontawesome-free/svgs/solid/file-excel.svg +1 -0
  1289. package/static/vendor/fontawesome-free/svgs/solid/file-export.svg +1 -0
  1290. package/static/vendor/fontawesome-free/svgs/solid/file-image.svg +1 -0
  1291. package/static/vendor/fontawesome-free/svgs/solid/file-import.svg +1 -0
  1292. package/static/vendor/fontawesome-free/svgs/solid/file-invoice-dollar.svg +1 -0
  1293. package/static/vendor/fontawesome-free/svgs/solid/file-invoice.svg +1 -0
  1294. package/static/vendor/fontawesome-free/svgs/solid/file-medical-alt.svg +1 -0
  1295. package/static/vendor/fontawesome-free/svgs/solid/file-medical.svg +1 -0
  1296. package/static/vendor/fontawesome-free/svgs/solid/file-pdf.svg +1 -0
  1297. package/static/vendor/fontawesome-free/svgs/solid/file-powerpoint.svg +1 -0
  1298. package/static/vendor/fontawesome-free/svgs/solid/file-prescription.svg +1 -0
  1299. package/static/vendor/fontawesome-free/svgs/solid/file-signature.svg +1 -0
  1300. package/static/vendor/fontawesome-free/svgs/solid/file-upload.svg +1 -0
  1301. package/static/vendor/fontawesome-free/svgs/solid/file-video.svg +1 -0
  1302. package/static/vendor/fontawesome-free/svgs/solid/file-word.svg +1 -0
  1303. package/static/vendor/fontawesome-free/svgs/solid/file.svg +1 -0
  1304. package/static/vendor/fontawesome-free/svgs/solid/fill-drip.svg +1 -0
  1305. package/static/vendor/fontawesome-free/svgs/solid/fill.svg +1 -0
  1306. package/static/vendor/fontawesome-free/svgs/solid/film.svg +1 -0
  1307. package/static/vendor/fontawesome-free/svgs/solid/filter.svg +1 -0
  1308. package/static/vendor/fontawesome-free/svgs/solid/fingerprint.svg +1 -0
  1309. package/static/vendor/fontawesome-free/svgs/solid/fire-alt.svg +1 -0
  1310. package/static/vendor/fontawesome-free/svgs/solid/fire-extinguisher.svg +1 -0
  1311. package/static/vendor/fontawesome-free/svgs/solid/fire.svg +1 -0
  1312. package/static/vendor/fontawesome-free/svgs/solid/first-aid.svg +1 -0
  1313. package/static/vendor/fontawesome-free/svgs/solid/fish.svg +1 -0
  1314. package/static/vendor/fontawesome-free/svgs/solid/fist-raised.svg +1 -0
  1315. package/static/vendor/fontawesome-free/svgs/solid/flag-checkered.svg +1 -0
  1316. package/static/vendor/fontawesome-free/svgs/solid/flag-usa.svg +1 -0
  1317. package/static/vendor/fontawesome-free/svgs/solid/flag.svg +1 -0
  1318. package/static/vendor/fontawesome-free/svgs/solid/flask.svg +1 -0
  1319. package/static/vendor/fontawesome-free/svgs/solid/flushed.svg +1 -0
  1320. package/static/vendor/fontawesome-free/svgs/solid/folder-minus.svg +1 -0
  1321. package/static/vendor/fontawesome-free/svgs/solid/folder-open.svg +1 -0
  1322. package/static/vendor/fontawesome-free/svgs/solid/folder-plus.svg +1 -0
  1323. package/static/vendor/fontawesome-free/svgs/solid/folder.svg +1 -0
  1324. package/static/vendor/fontawesome-free/svgs/solid/font-awesome-logo-full.svg +1 -0
  1325. package/static/vendor/fontawesome-free/svgs/solid/font.svg +1 -0
  1326. package/static/vendor/fontawesome-free/svgs/solid/football-ball.svg +1 -0
  1327. package/static/vendor/fontawesome-free/svgs/solid/forward.svg +1 -0
  1328. package/static/vendor/fontawesome-free/svgs/solid/frog.svg +1 -0
  1329. package/static/vendor/fontawesome-free/svgs/solid/frown-open.svg +1 -0
  1330. package/static/vendor/fontawesome-free/svgs/solid/frown.svg +1 -0
  1331. package/static/vendor/fontawesome-free/svgs/solid/funnel-dollar.svg +1 -0
  1332. package/static/vendor/fontawesome-free/svgs/solid/futbol.svg +1 -0
  1333. package/static/vendor/fontawesome-free/svgs/solid/gamepad.svg +1 -0
  1334. package/static/vendor/fontawesome-free/svgs/solid/gas-pump.svg +1 -0
  1335. package/static/vendor/fontawesome-free/svgs/solid/gavel.svg +1 -0
  1336. package/static/vendor/fontawesome-free/svgs/solid/gem.svg +1 -0
  1337. package/static/vendor/fontawesome-free/svgs/solid/genderless.svg +1 -0
  1338. package/static/vendor/fontawesome-free/svgs/solid/ghost.svg +1 -0
  1339. package/static/vendor/fontawesome-free/svgs/solid/gift.svg +1 -0
  1340. package/static/vendor/fontawesome-free/svgs/solid/gifts.svg +1 -0
  1341. package/static/vendor/fontawesome-free/svgs/solid/glass-cheers.svg +1 -0
  1342. package/static/vendor/fontawesome-free/svgs/solid/glass-martini-alt.svg +1 -0
  1343. package/static/vendor/fontawesome-free/svgs/solid/glass-martini.svg +1 -0
  1344. package/static/vendor/fontawesome-free/svgs/solid/glass-whiskey.svg +1 -0
  1345. package/static/vendor/fontawesome-free/svgs/solid/glasses.svg +1 -0
  1346. package/static/vendor/fontawesome-free/svgs/solid/globe-africa.svg +1 -0
  1347. package/static/vendor/fontawesome-free/svgs/solid/globe-americas.svg +1 -0
  1348. package/static/vendor/fontawesome-free/svgs/solid/globe-asia.svg +1 -0
  1349. package/static/vendor/fontawesome-free/svgs/solid/globe-europe.svg +1 -0
  1350. package/static/vendor/fontawesome-free/svgs/solid/globe.svg +1 -0
  1351. package/static/vendor/fontawesome-free/svgs/solid/golf-ball.svg +1 -0
  1352. package/static/vendor/fontawesome-free/svgs/solid/gopuram.svg +1 -0
  1353. package/static/vendor/fontawesome-free/svgs/solid/graduation-cap.svg +1 -0
  1354. package/static/vendor/fontawesome-free/svgs/solid/greater-than-equal.svg +1 -0
  1355. package/static/vendor/fontawesome-free/svgs/solid/greater-than.svg +1 -0
  1356. package/static/vendor/fontawesome-free/svgs/solid/grimace.svg +1 -0
  1357. package/static/vendor/fontawesome-free/svgs/solid/grin-alt.svg +1 -0
  1358. package/static/vendor/fontawesome-free/svgs/solid/grin-beam-sweat.svg +1 -0
  1359. package/static/vendor/fontawesome-free/svgs/solid/grin-beam.svg +1 -0
  1360. package/static/vendor/fontawesome-free/svgs/solid/grin-hearts.svg +1 -0
  1361. package/static/vendor/fontawesome-free/svgs/solid/grin-squint-tears.svg +1 -0
  1362. package/static/vendor/fontawesome-free/svgs/solid/grin-squint.svg +1 -0
  1363. package/static/vendor/fontawesome-free/svgs/solid/grin-stars.svg +1 -0
  1364. package/static/vendor/fontawesome-free/svgs/solid/grin-tears.svg +1 -0
  1365. package/static/vendor/fontawesome-free/svgs/solid/grin-tongue-squint.svg +1 -0
  1366. package/static/vendor/fontawesome-free/svgs/solid/grin-tongue-wink.svg +1 -0
  1367. package/static/vendor/fontawesome-free/svgs/solid/grin-tongue.svg +1 -0
  1368. package/static/vendor/fontawesome-free/svgs/solid/grin-wink.svg +1 -0
  1369. package/static/vendor/fontawesome-free/svgs/solid/grin.svg +1 -0
  1370. package/static/vendor/fontawesome-free/svgs/solid/grip-horizontal.svg +1 -0
  1371. package/static/vendor/fontawesome-free/svgs/solid/grip-lines-vertical.svg +1 -0
  1372. package/static/vendor/fontawesome-free/svgs/solid/grip-lines.svg +1 -0
  1373. package/static/vendor/fontawesome-free/svgs/solid/grip-vertical.svg +1 -0
  1374. package/static/vendor/fontawesome-free/svgs/solid/guitar.svg +1 -0
  1375. package/static/vendor/fontawesome-free/svgs/solid/h-square.svg +1 -0
  1376. package/static/vendor/fontawesome-free/svgs/solid/hamburger.svg +1 -0
  1377. package/static/vendor/fontawesome-free/svgs/solid/hammer.svg +1 -0
  1378. package/static/vendor/fontawesome-free/svgs/solid/hamsa.svg +1 -0
  1379. package/static/vendor/fontawesome-free/svgs/solid/hand-holding-heart.svg +1 -0
  1380. package/static/vendor/fontawesome-free/svgs/solid/hand-holding-medical.svg +1 -0
  1381. package/static/vendor/fontawesome-free/svgs/solid/hand-holding-usd.svg +1 -0
  1382. package/static/vendor/fontawesome-free/svgs/solid/hand-holding-water.svg +1 -0
  1383. package/static/vendor/fontawesome-free/svgs/solid/hand-holding.svg +1 -0
  1384. package/static/vendor/fontawesome-free/svgs/solid/hand-lizard.svg +1 -0
  1385. package/static/vendor/fontawesome-free/svgs/solid/hand-middle-finger.svg +1 -0
  1386. package/static/vendor/fontawesome-free/svgs/solid/hand-paper.svg +1 -0
  1387. package/static/vendor/fontawesome-free/svgs/solid/hand-peace.svg +1 -0
  1388. package/static/vendor/fontawesome-free/svgs/solid/hand-point-down.svg +1 -0
  1389. package/static/vendor/fontawesome-free/svgs/solid/hand-point-left.svg +1 -0
  1390. package/static/vendor/fontawesome-free/svgs/solid/hand-point-right.svg +1 -0
  1391. package/static/vendor/fontawesome-free/svgs/solid/hand-point-up.svg +1 -0
  1392. package/static/vendor/fontawesome-free/svgs/solid/hand-pointer.svg +1 -0
  1393. package/static/vendor/fontawesome-free/svgs/solid/hand-rock.svg +1 -0
  1394. package/static/vendor/fontawesome-free/svgs/solid/hand-scissors.svg +1 -0
  1395. package/static/vendor/fontawesome-free/svgs/solid/hand-sparkles.svg +1 -0
  1396. package/static/vendor/fontawesome-free/svgs/solid/hand-spock.svg +1 -0
  1397. package/static/vendor/fontawesome-free/svgs/solid/hands-helping.svg +1 -0
  1398. package/static/vendor/fontawesome-free/svgs/solid/hands-wash.svg +1 -0
  1399. package/static/vendor/fontawesome-free/svgs/solid/hands.svg +1 -0
  1400. package/static/vendor/fontawesome-free/svgs/solid/handshake-alt-slash.svg +1 -0
  1401. package/static/vendor/fontawesome-free/svgs/solid/handshake-slash.svg +1 -0
  1402. package/static/vendor/fontawesome-free/svgs/solid/handshake.svg +1 -0
  1403. package/static/vendor/fontawesome-free/svgs/solid/hanukiah.svg +1 -0
  1404. package/static/vendor/fontawesome-free/svgs/solid/hard-hat.svg +1 -0
  1405. package/static/vendor/fontawesome-free/svgs/solid/hashtag.svg +1 -0
  1406. package/static/vendor/fontawesome-free/svgs/solid/hat-cowboy-side.svg +1 -0
  1407. package/static/vendor/fontawesome-free/svgs/solid/hat-cowboy.svg +1 -0
  1408. package/static/vendor/fontawesome-free/svgs/solid/hat-wizard.svg +1 -0
  1409. package/static/vendor/fontawesome-free/svgs/solid/hdd.svg +1 -0
  1410. package/static/vendor/fontawesome-free/svgs/solid/head-side-cough-slash.svg +1 -0
  1411. package/static/vendor/fontawesome-free/svgs/solid/head-side-cough.svg +1 -0
  1412. package/static/vendor/fontawesome-free/svgs/solid/head-side-mask.svg +1 -0
  1413. package/static/vendor/fontawesome-free/svgs/solid/head-side-virus.svg +1 -0
  1414. package/static/vendor/fontawesome-free/svgs/solid/heading.svg +1 -0
  1415. package/static/vendor/fontawesome-free/svgs/solid/headphones-alt.svg +1 -0
  1416. package/static/vendor/fontawesome-free/svgs/solid/headphones.svg +1 -0
  1417. package/static/vendor/fontawesome-free/svgs/solid/headset.svg +1 -0
  1418. package/static/vendor/fontawesome-free/svgs/solid/heart-broken.svg +1 -0
  1419. package/static/vendor/fontawesome-free/svgs/solid/heart.svg +1 -0
  1420. package/static/vendor/fontawesome-free/svgs/solid/heartbeat.svg +1 -0
  1421. package/static/vendor/fontawesome-free/svgs/solid/helicopter.svg +1 -0
  1422. package/static/vendor/fontawesome-free/svgs/solid/highlighter.svg +1 -0
  1423. package/static/vendor/fontawesome-free/svgs/solid/hiking.svg +1 -0
  1424. package/static/vendor/fontawesome-free/svgs/solid/hippo.svg +1 -0
  1425. package/static/vendor/fontawesome-free/svgs/solid/history.svg +1 -0
  1426. package/static/vendor/fontawesome-free/svgs/solid/hockey-puck.svg +1 -0
  1427. package/static/vendor/fontawesome-free/svgs/solid/holly-berry.svg +1 -0
  1428. package/static/vendor/fontawesome-free/svgs/solid/home.svg +1 -0
  1429. package/static/vendor/fontawesome-free/svgs/solid/horse-head.svg +1 -0
  1430. package/static/vendor/fontawesome-free/svgs/solid/horse.svg +1 -0
  1431. package/static/vendor/fontawesome-free/svgs/solid/hospital-alt.svg +1 -0
  1432. package/static/vendor/fontawesome-free/svgs/solid/hospital-symbol.svg +1 -0
  1433. package/static/vendor/fontawesome-free/svgs/solid/hospital-user.svg +1 -0
  1434. package/static/vendor/fontawesome-free/svgs/solid/hospital.svg +1 -0
  1435. package/static/vendor/fontawesome-free/svgs/solid/hot-tub.svg +1 -0
  1436. package/static/vendor/fontawesome-free/svgs/solid/hotdog.svg +1 -0
  1437. package/static/vendor/fontawesome-free/svgs/solid/hotel.svg +1 -0
  1438. package/static/vendor/fontawesome-free/svgs/solid/hourglass-end.svg +1 -0
  1439. package/static/vendor/fontawesome-free/svgs/solid/hourglass-half.svg +1 -0
  1440. package/static/vendor/fontawesome-free/svgs/solid/hourglass-start.svg +1 -0
  1441. package/static/vendor/fontawesome-free/svgs/solid/hourglass.svg +1 -0
  1442. package/static/vendor/fontawesome-free/svgs/solid/house-damage.svg +1 -0
  1443. package/static/vendor/fontawesome-free/svgs/solid/house-user.svg +1 -0
  1444. package/static/vendor/fontawesome-free/svgs/solid/hryvnia.svg +1 -0
  1445. package/static/vendor/fontawesome-free/svgs/solid/i-cursor.svg +1 -0
  1446. package/static/vendor/fontawesome-free/svgs/solid/ice-cream.svg +1 -0
  1447. package/static/vendor/fontawesome-free/svgs/solid/icicles.svg +1 -0
  1448. package/static/vendor/fontawesome-free/svgs/solid/icons.svg +1 -0
  1449. package/static/vendor/fontawesome-free/svgs/solid/id-badge.svg +1 -0
  1450. package/static/vendor/fontawesome-free/svgs/solid/id-card-alt.svg +1 -0
  1451. package/static/vendor/fontawesome-free/svgs/solid/id-card.svg +1 -0
  1452. package/static/vendor/fontawesome-free/svgs/solid/igloo.svg +1 -0
  1453. package/static/vendor/fontawesome-free/svgs/solid/image.svg +1 -0
  1454. package/static/vendor/fontawesome-free/svgs/solid/images.svg +1 -0
  1455. package/static/vendor/fontawesome-free/svgs/solid/inbox.svg +1 -0
  1456. package/static/vendor/fontawesome-free/svgs/solid/indent.svg +1 -0
  1457. package/static/vendor/fontawesome-free/svgs/solid/industry.svg +1 -0
  1458. package/static/vendor/fontawesome-free/svgs/solid/infinity.svg +1 -0
  1459. package/static/vendor/fontawesome-free/svgs/solid/info-circle.svg +1 -0
  1460. package/static/vendor/fontawesome-free/svgs/solid/info.svg +1 -0
  1461. package/static/vendor/fontawesome-free/svgs/solid/italic.svg +1 -0
  1462. package/static/vendor/fontawesome-free/svgs/solid/jedi.svg +1 -0
  1463. package/static/vendor/fontawesome-free/svgs/solid/joint.svg +1 -0
  1464. package/static/vendor/fontawesome-free/svgs/solid/journal-whills.svg +1 -0
  1465. package/static/vendor/fontawesome-free/svgs/solid/kaaba.svg +1 -0
  1466. package/static/vendor/fontawesome-free/svgs/solid/key.svg +1 -0
  1467. package/static/vendor/fontawesome-free/svgs/solid/keyboard.svg +1 -0
  1468. package/static/vendor/fontawesome-free/svgs/solid/khanda.svg +1 -0
  1469. package/static/vendor/fontawesome-free/svgs/solid/kiss-beam.svg +1 -0
  1470. package/static/vendor/fontawesome-free/svgs/solid/kiss-wink-heart.svg +1 -0
  1471. package/static/vendor/fontawesome-free/svgs/solid/kiss.svg +1 -0
  1472. package/static/vendor/fontawesome-free/svgs/solid/kiwi-bird.svg +1 -0
  1473. package/static/vendor/fontawesome-free/svgs/solid/landmark.svg +1 -0
  1474. package/static/vendor/fontawesome-free/svgs/solid/language.svg +1 -0
  1475. package/static/vendor/fontawesome-free/svgs/solid/laptop-code.svg +1 -0
  1476. package/static/vendor/fontawesome-free/svgs/solid/laptop-house.svg +1 -0
  1477. package/static/vendor/fontawesome-free/svgs/solid/laptop-medical.svg +1 -0
  1478. package/static/vendor/fontawesome-free/svgs/solid/laptop.svg +1 -0
  1479. package/static/vendor/fontawesome-free/svgs/solid/laugh-beam.svg +1 -0
  1480. package/static/vendor/fontawesome-free/svgs/solid/laugh-squint.svg +1 -0
  1481. package/static/vendor/fontawesome-free/svgs/solid/laugh-wink.svg +1 -0
  1482. package/static/vendor/fontawesome-free/svgs/solid/laugh.svg +1 -0
  1483. package/static/vendor/fontawesome-free/svgs/solid/layer-group.svg +1 -0
  1484. package/static/vendor/fontawesome-free/svgs/solid/leaf.svg +1 -0
  1485. package/static/vendor/fontawesome-free/svgs/solid/lemon.svg +1 -0
  1486. package/static/vendor/fontawesome-free/svgs/solid/less-than-equal.svg +1 -0
  1487. package/static/vendor/fontawesome-free/svgs/solid/less-than.svg +1 -0
  1488. package/static/vendor/fontawesome-free/svgs/solid/level-down-alt.svg +1 -0
  1489. package/static/vendor/fontawesome-free/svgs/solid/level-up-alt.svg +1 -0
  1490. package/static/vendor/fontawesome-free/svgs/solid/life-ring.svg +1 -0
  1491. package/static/vendor/fontawesome-free/svgs/solid/lightbulb.svg +1 -0
  1492. package/static/vendor/fontawesome-free/svgs/solid/link.svg +1 -0
  1493. package/static/vendor/fontawesome-free/svgs/solid/lira-sign.svg +1 -0
  1494. package/static/vendor/fontawesome-free/svgs/solid/list-alt.svg +1 -0
  1495. package/static/vendor/fontawesome-free/svgs/solid/list-ol.svg +1 -0
  1496. package/static/vendor/fontawesome-free/svgs/solid/list-ul.svg +1 -0
  1497. package/static/vendor/fontawesome-free/svgs/solid/list.svg +1 -0
  1498. package/static/vendor/fontawesome-free/svgs/solid/location-arrow.svg +1 -0
  1499. package/static/vendor/fontawesome-free/svgs/solid/lock-open.svg +1 -0
  1500. package/static/vendor/fontawesome-free/svgs/solid/lock.svg +1 -0
  1501. package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-down.svg +1 -0
  1502. package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-left.svg +1 -0
  1503. package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-right.svg +1 -0
  1504. package/static/vendor/fontawesome-free/svgs/solid/long-arrow-alt-up.svg +1 -0
  1505. package/static/vendor/fontawesome-free/svgs/solid/low-vision.svg +1 -0
  1506. package/static/vendor/fontawesome-free/svgs/solid/luggage-cart.svg +1 -0
  1507. package/static/vendor/fontawesome-free/svgs/solid/lungs-virus.svg +1 -0
  1508. package/static/vendor/fontawesome-free/svgs/solid/lungs.svg +1 -0
  1509. package/static/vendor/fontawesome-free/svgs/solid/magic.svg +1 -0
  1510. package/static/vendor/fontawesome-free/svgs/solid/magnet.svg +1 -0
  1511. package/static/vendor/fontawesome-free/svgs/solid/mail-bulk.svg +1 -0
  1512. package/static/vendor/fontawesome-free/svgs/solid/male.svg +1 -0
  1513. package/static/vendor/fontawesome-free/svgs/solid/map-marked-alt.svg +1 -0
  1514. package/static/vendor/fontawesome-free/svgs/solid/map-marked.svg +1 -0
  1515. package/static/vendor/fontawesome-free/svgs/solid/map-marker-alt.svg +1 -0
  1516. package/static/vendor/fontawesome-free/svgs/solid/map-marker.svg +1 -0
  1517. package/static/vendor/fontawesome-free/svgs/solid/map-pin.svg +1 -0
  1518. package/static/vendor/fontawesome-free/svgs/solid/map-signs.svg +1 -0
  1519. package/static/vendor/fontawesome-free/svgs/solid/map.svg +1 -0
  1520. package/static/vendor/fontawesome-free/svgs/solid/marker.svg +1 -0
  1521. package/static/vendor/fontawesome-free/svgs/solid/mars-double.svg +1 -0
  1522. package/static/vendor/fontawesome-free/svgs/solid/mars-stroke-h.svg +1 -0
  1523. package/static/vendor/fontawesome-free/svgs/solid/mars-stroke-v.svg +1 -0
  1524. package/static/vendor/fontawesome-free/svgs/solid/mars-stroke.svg +1 -0
  1525. package/static/vendor/fontawesome-free/svgs/solid/mars.svg +1 -0
  1526. package/static/vendor/fontawesome-free/svgs/solid/mask.svg +1 -0
  1527. package/static/vendor/fontawesome-free/svgs/solid/medal.svg +1 -0
  1528. package/static/vendor/fontawesome-free/svgs/solid/medkit.svg +1 -0
  1529. package/static/vendor/fontawesome-free/svgs/solid/meh-blank.svg +1 -0
  1530. package/static/vendor/fontawesome-free/svgs/solid/meh-rolling-eyes.svg +1 -0
  1531. package/static/vendor/fontawesome-free/svgs/solid/meh.svg +1 -0
  1532. package/static/vendor/fontawesome-free/svgs/solid/memory.svg +1 -0
  1533. package/static/vendor/fontawesome-free/svgs/solid/menorah.svg +1 -0
  1534. package/static/vendor/fontawesome-free/svgs/solid/mercury.svg +1 -0
  1535. package/static/vendor/fontawesome-free/svgs/solid/meteor.svg +1 -0
  1536. package/static/vendor/fontawesome-free/svgs/solid/microchip.svg +1 -0
  1537. package/static/vendor/fontawesome-free/svgs/solid/microphone-alt-slash.svg +1 -0
  1538. package/static/vendor/fontawesome-free/svgs/solid/microphone-alt.svg +1 -0
  1539. package/static/vendor/fontawesome-free/svgs/solid/microphone-slash.svg +1 -0
  1540. package/static/vendor/fontawesome-free/svgs/solid/microphone.svg +1 -0
  1541. package/static/vendor/fontawesome-free/svgs/solid/microscope.svg +1 -0
  1542. package/static/vendor/fontawesome-free/svgs/solid/minus-circle.svg +1 -0
  1543. package/static/vendor/fontawesome-free/svgs/solid/minus-square.svg +1 -0
  1544. package/static/vendor/fontawesome-free/svgs/solid/minus.svg +1 -0
  1545. package/static/vendor/fontawesome-free/svgs/solid/mitten.svg +1 -0
  1546. package/static/vendor/fontawesome-free/svgs/solid/mobile-alt.svg +1 -0
  1547. package/static/vendor/fontawesome-free/svgs/solid/mobile.svg +1 -0
  1548. package/static/vendor/fontawesome-free/svgs/solid/money-bill-alt.svg +1 -0
  1549. package/static/vendor/fontawesome-free/svgs/solid/money-bill-wave-alt.svg +1 -0
  1550. package/static/vendor/fontawesome-free/svgs/solid/money-bill-wave.svg +1 -0
  1551. package/static/vendor/fontawesome-free/svgs/solid/money-bill.svg +1 -0
  1552. package/static/vendor/fontawesome-free/svgs/solid/money-check-alt.svg +1 -0
  1553. package/static/vendor/fontawesome-free/svgs/solid/money-check.svg +1 -0
  1554. package/static/vendor/fontawesome-free/svgs/solid/monument.svg +1 -0
  1555. package/static/vendor/fontawesome-free/svgs/solid/moon.svg +1 -0
  1556. package/static/vendor/fontawesome-free/svgs/solid/mortar-pestle.svg +1 -0
  1557. package/static/vendor/fontawesome-free/svgs/solid/mosque.svg +1 -0
  1558. package/static/vendor/fontawesome-free/svgs/solid/motorcycle.svg +1 -0
  1559. package/static/vendor/fontawesome-free/svgs/solid/mountain.svg +1 -0
  1560. package/static/vendor/fontawesome-free/svgs/solid/mouse-pointer.svg +1 -0
  1561. package/static/vendor/fontawesome-free/svgs/solid/mouse.svg +1 -0
  1562. package/static/vendor/fontawesome-free/svgs/solid/mug-hot.svg +1 -0
  1563. package/static/vendor/fontawesome-free/svgs/solid/music.svg +1 -0
  1564. package/static/vendor/fontawesome-free/svgs/solid/network-wired.svg +1 -0
  1565. package/static/vendor/fontawesome-free/svgs/solid/neuter.svg +1 -0
  1566. package/static/vendor/fontawesome-free/svgs/solid/newspaper.svg +1 -0
  1567. package/static/vendor/fontawesome-free/svgs/solid/not-equal.svg +1 -0
  1568. package/static/vendor/fontawesome-free/svgs/solid/notes-medical.svg +1 -0
  1569. package/static/vendor/fontawesome-free/svgs/solid/object-group.svg +1 -0
  1570. package/static/vendor/fontawesome-free/svgs/solid/object-ungroup.svg +1 -0
  1571. package/static/vendor/fontawesome-free/svgs/solid/oil-can.svg +1 -0
  1572. package/static/vendor/fontawesome-free/svgs/solid/om.svg +1 -0
  1573. package/static/vendor/fontawesome-free/svgs/solid/otter.svg +1 -0
  1574. package/static/vendor/fontawesome-free/svgs/solid/outdent.svg +1 -0
  1575. package/static/vendor/fontawesome-free/svgs/solid/pager.svg +1 -0
  1576. package/static/vendor/fontawesome-free/svgs/solid/paint-brush.svg +1 -0
  1577. package/static/vendor/fontawesome-free/svgs/solid/paint-roller.svg +1 -0
  1578. package/static/vendor/fontawesome-free/svgs/solid/palette.svg +1 -0
  1579. package/static/vendor/fontawesome-free/svgs/solid/pallet.svg +1 -0
  1580. package/static/vendor/fontawesome-free/svgs/solid/paper-plane.svg +1 -0
  1581. package/static/vendor/fontawesome-free/svgs/solid/paperclip.svg +1 -0
  1582. package/static/vendor/fontawesome-free/svgs/solid/parachute-box.svg +1 -0
  1583. package/static/vendor/fontawesome-free/svgs/solid/paragraph.svg +1 -0
  1584. package/static/vendor/fontawesome-free/svgs/solid/parking.svg +1 -0
  1585. package/static/vendor/fontawesome-free/svgs/solid/passport.svg +1 -0
  1586. package/static/vendor/fontawesome-free/svgs/solid/pastafarianism.svg +1 -0
  1587. package/static/vendor/fontawesome-free/svgs/solid/paste.svg +1 -0
  1588. package/static/vendor/fontawesome-free/svgs/solid/pause-circle.svg +1 -0
  1589. package/static/vendor/fontawesome-free/svgs/solid/pause.svg +1 -0
  1590. package/static/vendor/fontawesome-free/svgs/solid/paw.svg +1 -0
  1591. package/static/vendor/fontawesome-free/svgs/solid/peace.svg +1 -0
  1592. package/static/vendor/fontawesome-free/svgs/solid/pen-alt.svg +1 -0
  1593. package/static/vendor/fontawesome-free/svgs/solid/pen-fancy.svg +1 -0
  1594. package/static/vendor/fontawesome-free/svgs/solid/pen-nib.svg +1 -0
  1595. package/static/vendor/fontawesome-free/svgs/solid/pen-square.svg +1 -0
  1596. package/static/vendor/fontawesome-free/svgs/solid/pen.svg +1 -0
  1597. package/static/vendor/fontawesome-free/svgs/solid/pencil-alt.svg +1 -0
  1598. package/static/vendor/fontawesome-free/svgs/solid/pencil-ruler.svg +1 -0
  1599. package/static/vendor/fontawesome-free/svgs/solid/people-arrows.svg +1 -0
  1600. package/static/vendor/fontawesome-free/svgs/solid/people-carry.svg +1 -0
  1601. package/static/vendor/fontawesome-free/svgs/solid/pepper-hot.svg +1 -0
  1602. package/static/vendor/fontawesome-free/svgs/solid/percent.svg +1 -0
  1603. package/static/vendor/fontawesome-free/svgs/solid/percentage.svg +1 -0
  1604. package/static/vendor/fontawesome-free/svgs/solid/person-booth.svg +1 -0
  1605. package/static/vendor/fontawesome-free/svgs/solid/phone-alt.svg +1 -0
  1606. package/static/vendor/fontawesome-free/svgs/solid/phone-slash.svg +1 -0
  1607. package/static/vendor/fontawesome-free/svgs/solid/phone-square-alt.svg +1 -0
  1608. package/static/vendor/fontawesome-free/svgs/solid/phone-square.svg +1 -0
  1609. package/static/vendor/fontawesome-free/svgs/solid/phone-volume.svg +1 -0
  1610. package/static/vendor/fontawesome-free/svgs/solid/phone.svg +1 -0
  1611. package/static/vendor/fontawesome-free/svgs/solid/photo-video.svg +1 -0
  1612. package/static/vendor/fontawesome-free/svgs/solid/piggy-bank.svg +1 -0
  1613. package/static/vendor/fontawesome-free/svgs/solid/pills.svg +1 -0
  1614. package/static/vendor/fontawesome-free/svgs/solid/pizza-slice.svg +1 -0
  1615. package/static/vendor/fontawesome-free/svgs/solid/place-of-worship.svg +1 -0
  1616. package/static/vendor/fontawesome-free/svgs/solid/plane-arrival.svg +1 -0
  1617. package/static/vendor/fontawesome-free/svgs/solid/plane-departure.svg +1 -0
  1618. package/static/vendor/fontawesome-free/svgs/solid/plane-slash.svg +1 -0
  1619. package/static/vendor/fontawesome-free/svgs/solid/plane.svg +1 -0
  1620. package/static/vendor/fontawesome-free/svgs/solid/play-circle.svg +1 -0
  1621. package/static/vendor/fontawesome-free/svgs/solid/play.svg +1 -0
  1622. package/static/vendor/fontawesome-free/svgs/solid/plug.svg +1 -0
  1623. package/static/vendor/fontawesome-free/svgs/solid/plus-circle.svg +1 -0
  1624. package/static/vendor/fontawesome-free/svgs/solid/plus-square.svg +1 -0
  1625. package/static/vendor/fontawesome-free/svgs/solid/plus.svg +1 -0
  1626. package/static/vendor/fontawesome-free/svgs/solid/podcast.svg +1 -0
  1627. package/static/vendor/fontawesome-free/svgs/solid/poll-h.svg +1 -0
  1628. package/static/vendor/fontawesome-free/svgs/solid/poll.svg +1 -0
  1629. package/static/vendor/fontawesome-free/svgs/solid/poo-storm.svg +1 -0
  1630. package/static/vendor/fontawesome-free/svgs/solid/poo.svg +1 -0
  1631. package/static/vendor/fontawesome-free/svgs/solid/poop.svg +1 -0
  1632. package/static/vendor/fontawesome-free/svgs/solid/portrait.svg +1 -0
  1633. package/static/vendor/fontawesome-free/svgs/solid/pound-sign.svg +1 -0
  1634. package/static/vendor/fontawesome-free/svgs/solid/power-off.svg +1 -0
  1635. package/static/vendor/fontawesome-free/svgs/solid/pray.svg +1 -0
  1636. package/static/vendor/fontawesome-free/svgs/solid/praying-hands.svg +1 -0
  1637. package/static/vendor/fontawesome-free/svgs/solid/prescription-bottle-alt.svg +1 -0
  1638. package/static/vendor/fontawesome-free/svgs/solid/prescription-bottle.svg +1 -0
  1639. package/static/vendor/fontawesome-free/svgs/solid/prescription.svg +1 -0
  1640. package/static/vendor/fontawesome-free/svgs/solid/print.svg +1 -0
  1641. package/static/vendor/fontawesome-free/svgs/solid/procedures.svg +1 -0
  1642. package/static/vendor/fontawesome-free/svgs/solid/project-diagram.svg +1 -0
  1643. package/static/vendor/fontawesome-free/svgs/solid/pump-medical.svg +1 -0
  1644. package/static/vendor/fontawesome-free/svgs/solid/pump-soap.svg +1 -0
  1645. package/static/vendor/fontawesome-free/svgs/solid/puzzle-piece.svg +1 -0
  1646. package/static/vendor/fontawesome-free/svgs/solid/qrcode.svg +1 -0
  1647. package/static/vendor/fontawesome-free/svgs/solid/question-circle.svg +1 -0
  1648. package/static/vendor/fontawesome-free/svgs/solid/question.svg +1 -0
  1649. package/static/vendor/fontawesome-free/svgs/solid/quidditch.svg +1 -0
  1650. package/static/vendor/fontawesome-free/svgs/solid/quote-left.svg +1 -0
  1651. package/static/vendor/fontawesome-free/svgs/solid/quote-right.svg +1 -0
  1652. package/static/vendor/fontawesome-free/svgs/solid/quran.svg +1 -0
  1653. package/static/vendor/fontawesome-free/svgs/solid/radiation-alt.svg +1 -0
  1654. package/static/vendor/fontawesome-free/svgs/solid/radiation.svg +1 -0
  1655. package/static/vendor/fontawesome-free/svgs/solid/rainbow.svg +1 -0
  1656. package/static/vendor/fontawesome-free/svgs/solid/random.svg +1 -0
  1657. package/static/vendor/fontawesome-free/svgs/solid/receipt.svg +1 -0
  1658. package/static/vendor/fontawesome-free/svgs/solid/record-vinyl.svg +1 -0
  1659. package/static/vendor/fontawesome-free/svgs/solid/recycle.svg +1 -0
  1660. package/static/vendor/fontawesome-free/svgs/solid/redo-alt.svg +1 -0
  1661. package/static/vendor/fontawesome-free/svgs/solid/redo.svg +1 -0
  1662. package/static/vendor/fontawesome-free/svgs/solid/registered.svg +1 -0
  1663. package/static/vendor/fontawesome-free/svgs/solid/remove-format.svg +1 -0
  1664. package/static/vendor/fontawesome-free/svgs/solid/reply-all.svg +1 -0
  1665. package/static/vendor/fontawesome-free/svgs/solid/reply.svg +1 -0
  1666. package/static/vendor/fontawesome-free/svgs/solid/republican.svg +1 -0
  1667. package/static/vendor/fontawesome-free/svgs/solid/restroom.svg +1 -0
  1668. package/static/vendor/fontawesome-free/svgs/solid/retweet.svg +1 -0
  1669. package/static/vendor/fontawesome-free/svgs/solid/ribbon.svg +1 -0
  1670. package/static/vendor/fontawesome-free/svgs/solid/ring.svg +1 -0
  1671. package/static/vendor/fontawesome-free/svgs/solid/road.svg +1 -0
  1672. package/static/vendor/fontawesome-free/svgs/solid/robot.svg +1 -0
  1673. package/static/vendor/fontawesome-free/svgs/solid/rocket.svg +1 -0
  1674. package/static/vendor/fontawesome-free/svgs/solid/route.svg +1 -0
  1675. package/static/vendor/fontawesome-free/svgs/solid/rss-square.svg +1 -0
  1676. package/static/vendor/fontawesome-free/svgs/solid/rss.svg +1 -0
  1677. package/static/vendor/fontawesome-free/svgs/solid/ruble-sign.svg +1 -0
  1678. package/static/vendor/fontawesome-free/svgs/solid/ruler-combined.svg +1 -0
  1679. package/static/vendor/fontawesome-free/svgs/solid/ruler-horizontal.svg +1 -0
  1680. package/static/vendor/fontawesome-free/svgs/solid/ruler-vertical.svg +1 -0
  1681. package/static/vendor/fontawesome-free/svgs/solid/ruler.svg +1 -0
  1682. package/static/vendor/fontawesome-free/svgs/solid/running.svg +1 -0
  1683. package/static/vendor/fontawesome-free/svgs/solid/rupee-sign.svg +1 -0
  1684. package/static/vendor/fontawesome-free/svgs/solid/sad-cry.svg +1 -0
  1685. package/static/vendor/fontawesome-free/svgs/solid/sad-tear.svg +1 -0
  1686. package/static/vendor/fontawesome-free/svgs/solid/satellite-dish.svg +1 -0
  1687. package/static/vendor/fontawesome-free/svgs/solid/satellite.svg +1 -0
  1688. package/static/vendor/fontawesome-free/svgs/solid/save.svg +1 -0
  1689. package/static/vendor/fontawesome-free/svgs/solid/school.svg +1 -0
  1690. package/static/vendor/fontawesome-free/svgs/solid/screwdriver.svg +1 -0
  1691. package/static/vendor/fontawesome-free/svgs/solid/scroll.svg +1 -0
  1692. package/static/vendor/fontawesome-free/svgs/solid/sd-card.svg +1 -0
  1693. package/static/vendor/fontawesome-free/svgs/solid/search-dollar.svg +1 -0
  1694. package/static/vendor/fontawesome-free/svgs/solid/search-location.svg +1 -0
  1695. package/static/vendor/fontawesome-free/svgs/solid/search-minus.svg +1 -0
  1696. package/static/vendor/fontawesome-free/svgs/solid/search-plus.svg +1 -0
  1697. package/static/vendor/fontawesome-free/svgs/solid/search.svg +1 -0
  1698. package/static/vendor/fontawesome-free/svgs/solid/seedling.svg +1 -0
  1699. package/static/vendor/fontawesome-free/svgs/solid/server.svg +1 -0
  1700. package/static/vendor/fontawesome-free/svgs/solid/shapes.svg +1 -0
  1701. package/static/vendor/fontawesome-free/svgs/solid/share-alt-square.svg +1 -0
  1702. package/static/vendor/fontawesome-free/svgs/solid/share-alt.svg +1 -0
  1703. package/static/vendor/fontawesome-free/svgs/solid/share-square.svg +1 -0
  1704. package/static/vendor/fontawesome-free/svgs/solid/share.svg +1 -0
  1705. package/static/vendor/fontawesome-free/svgs/solid/shekel-sign.svg +1 -0
  1706. package/static/vendor/fontawesome-free/svgs/solid/shield-alt.svg +1 -0
  1707. package/static/vendor/fontawesome-free/svgs/solid/shield-virus.svg +1 -0
  1708. package/static/vendor/fontawesome-free/svgs/solid/ship.svg +1 -0
  1709. package/static/vendor/fontawesome-free/svgs/solid/shipping-fast.svg +1 -0
  1710. package/static/vendor/fontawesome-free/svgs/solid/shoe-prints.svg +1 -0
  1711. package/static/vendor/fontawesome-free/svgs/solid/shopping-bag.svg +1 -0
  1712. package/static/vendor/fontawesome-free/svgs/solid/shopping-basket.svg +1 -0
  1713. package/static/vendor/fontawesome-free/svgs/solid/shopping-cart.svg +1 -0
  1714. package/static/vendor/fontawesome-free/svgs/solid/shower.svg +1 -0
  1715. package/static/vendor/fontawesome-free/svgs/solid/shuttle-van.svg +1 -0
  1716. package/static/vendor/fontawesome-free/svgs/solid/sign-in-alt.svg +1 -0
  1717. package/static/vendor/fontawesome-free/svgs/solid/sign-language.svg +1 -0
  1718. package/static/vendor/fontawesome-free/svgs/solid/sign-out-alt.svg +1 -0
  1719. package/static/vendor/fontawesome-free/svgs/solid/sign.svg +1 -0
  1720. package/static/vendor/fontawesome-free/svgs/solid/signal.svg +1 -0
  1721. package/static/vendor/fontawesome-free/svgs/solid/signature.svg +1 -0
  1722. package/static/vendor/fontawesome-free/svgs/solid/sim-card.svg +1 -0
  1723. package/static/vendor/fontawesome-free/svgs/solid/sink.svg +1 -0
  1724. package/static/vendor/fontawesome-free/svgs/solid/sitemap.svg +1 -0
  1725. package/static/vendor/fontawesome-free/svgs/solid/skating.svg +1 -0
  1726. package/static/vendor/fontawesome-free/svgs/solid/skiing-nordic.svg +1 -0
  1727. package/static/vendor/fontawesome-free/svgs/solid/skiing.svg +1 -0
  1728. package/static/vendor/fontawesome-free/svgs/solid/skull-crossbones.svg +1 -0
  1729. package/static/vendor/fontawesome-free/svgs/solid/skull.svg +1 -0
  1730. package/static/vendor/fontawesome-free/svgs/solid/slash.svg +1 -0
  1731. package/static/vendor/fontawesome-free/svgs/solid/sleigh.svg +1 -0
  1732. package/static/vendor/fontawesome-free/svgs/solid/sliders-h.svg +1 -0
  1733. package/static/vendor/fontawesome-free/svgs/solid/smile-beam.svg +1 -0
  1734. package/static/vendor/fontawesome-free/svgs/solid/smile-wink.svg +1 -0
  1735. package/static/vendor/fontawesome-free/svgs/solid/smile.svg +1 -0
  1736. package/static/vendor/fontawesome-free/svgs/solid/smog.svg +1 -0
  1737. package/static/vendor/fontawesome-free/svgs/solid/smoking-ban.svg +1 -0
  1738. package/static/vendor/fontawesome-free/svgs/solid/smoking.svg +1 -0
  1739. package/static/vendor/fontawesome-free/svgs/solid/sms.svg +1 -0
  1740. package/static/vendor/fontawesome-free/svgs/solid/snowboarding.svg +1 -0
  1741. package/static/vendor/fontawesome-free/svgs/solid/snowflake.svg +1 -0
  1742. package/static/vendor/fontawesome-free/svgs/solid/snowman.svg +1 -0
  1743. package/static/vendor/fontawesome-free/svgs/solid/snowplow.svg +1 -0
  1744. package/static/vendor/fontawesome-free/svgs/solid/soap.svg +1 -0
  1745. package/static/vendor/fontawesome-free/svgs/solid/socks.svg +1 -0
  1746. package/static/vendor/fontawesome-free/svgs/solid/solar-panel.svg +1 -0
  1747. package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-down-alt.svg +1 -0
  1748. package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-down.svg +1 -0
  1749. package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-up-alt.svg +1 -0
  1750. package/static/vendor/fontawesome-free/svgs/solid/sort-alpha-up.svg +1 -0
  1751. package/static/vendor/fontawesome-free/svgs/solid/sort-amount-down-alt.svg +1 -0
  1752. package/static/vendor/fontawesome-free/svgs/solid/sort-amount-down.svg +1 -0
  1753. package/static/vendor/fontawesome-free/svgs/solid/sort-amount-up-alt.svg +1 -0
  1754. package/static/vendor/fontawesome-free/svgs/solid/sort-amount-up.svg +1 -0
  1755. package/static/vendor/fontawesome-free/svgs/solid/sort-down.svg +1 -0
  1756. package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-down-alt.svg +1 -0
  1757. package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-down.svg +1 -0
  1758. package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-up-alt.svg +1 -0
  1759. package/static/vendor/fontawesome-free/svgs/solid/sort-numeric-up.svg +1 -0
  1760. package/static/vendor/fontawesome-free/svgs/solid/sort-up.svg +1 -0
  1761. package/static/vendor/fontawesome-free/svgs/solid/sort.svg +1 -0
  1762. package/static/vendor/fontawesome-free/svgs/solid/spa.svg +1 -0
  1763. package/static/vendor/fontawesome-free/svgs/solid/space-shuttle.svg +1 -0
  1764. package/static/vendor/fontawesome-free/svgs/solid/spell-check.svg +1 -0
  1765. package/static/vendor/fontawesome-free/svgs/solid/spider.svg +1 -0
  1766. package/static/vendor/fontawesome-free/svgs/solid/spinner.svg +1 -0
  1767. package/static/vendor/fontawesome-free/svgs/solid/splotch.svg +1 -0
  1768. package/static/vendor/fontawesome-free/svgs/solid/spray-can.svg +1 -0
  1769. package/static/vendor/fontawesome-free/svgs/solid/square-full.svg +1 -0
  1770. package/static/vendor/fontawesome-free/svgs/solid/square-root-alt.svg +1 -0
  1771. package/static/vendor/fontawesome-free/svgs/solid/square.svg +1 -0
  1772. package/static/vendor/fontawesome-free/svgs/solid/stamp.svg +1 -0
  1773. package/static/vendor/fontawesome-free/svgs/solid/star-and-crescent.svg +1 -0
  1774. package/static/vendor/fontawesome-free/svgs/solid/star-half-alt.svg +1 -0
  1775. package/static/vendor/fontawesome-free/svgs/solid/star-half.svg +1 -0
  1776. package/static/vendor/fontawesome-free/svgs/solid/star-of-david.svg +1 -0
  1777. package/static/vendor/fontawesome-free/svgs/solid/star-of-life.svg +1 -0
  1778. package/static/vendor/fontawesome-free/svgs/solid/star.svg +1 -0
  1779. package/static/vendor/fontawesome-free/svgs/solid/step-backward.svg +1 -0
  1780. package/static/vendor/fontawesome-free/svgs/solid/step-forward.svg +1 -0
  1781. package/static/vendor/fontawesome-free/svgs/solid/stethoscope.svg +1 -0
  1782. package/static/vendor/fontawesome-free/svgs/solid/sticky-note.svg +1 -0
  1783. package/static/vendor/fontawesome-free/svgs/solid/stop-circle.svg +1 -0
  1784. package/static/vendor/fontawesome-free/svgs/solid/stop.svg +1 -0
  1785. package/static/vendor/fontawesome-free/svgs/solid/stopwatch-20.svg +1 -0
  1786. package/static/vendor/fontawesome-free/svgs/solid/stopwatch.svg +1 -0
  1787. package/static/vendor/fontawesome-free/svgs/solid/store-alt-slash.svg +1 -0
  1788. package/static/vendor/fontawesome-free/svgs/solid/store-alt.svg +1 -0
  1789. package/static/vendor/fontawesome-free/svgs/solid/store-slash.svg +1 -0
  1790. package/static/vendor/fontawesome-free/svgs/solid/store.svg +1 -0
  1791. package/static/vendor/fontawesome-free/svgs/solid/stream.svg +1 -0
  1792. package/static/vendor/fontawesome-free/svgs/solid/street-view.svg +1 -0
  1793. package/static/vendor/fontawesome-free/svgs/solid/strikethrough.svg +1 -0
  1794. package/static/vendor/fontawesome-free/svgs/solid/stroopwafel.svg +1 -0
  1795. package/static/vendor/fontawesome-free/svgs/solid/subscript.svg +1 -0
  1796. package/static/vendor/fontawesome-free/svgs/solid/subway.svg +1 -0
  1797. package/static/vendor/fontawesome-free/svgs/solid/suitcase-rolling.svg +1 -0
  1798. package/static/vendor/fontawesome-free/svgs/solid/suitcase.svg +1 -0
  1799. package/static/vendor/fontawesome-free/svgs/solid/sun.svg +1 -0
  1800. package/static/vendor/fontawesome-free/svgs/solid/superscript.svg +1 -0
  1801. package/static/vendor/fontawesome-free/svgs/solid/surprise.svg +1 -0
  1802. package/static/vendor/fontawesome-free/svgs/solid/swatchbook.svg +1 -0
  1803. package/static/vendor/fontawesome-free/svgs/solid/swimmer.svg +1 -0
  1804. package/static/vendor/fontawesome-free/svgs/solid/swimming-pool.svg +1 -0
  1805. package/static/vendor/fontawesome-free/svgs/solid/synagogue.svg +1 -0
  1806. package/static/vendor/fontawesome-free/svgs/solid/sync-alt.svg +1 -0
  1807. package/static/vendor/fontawesome-free/svgs/solid/sync.svg +1 -0
  1808. package/static/vendor/fontawesome-free/svgs/solid/syringe.svg +1 -0
  1809. package/static/vendor/fontawesome-free/svgs/solid/table-tennis.svg +1 -0
  1810. package/static/vendor/fontawesome-free/svgs/solid/table.svg +1 -0
  1811. package/static/vendor/fontawesome-free/svgs/solid/tablet-alt.svg +1 -0
  1812. package/static/vendor/fontawesome-free/svgs/solid/tablet.svg +1 -0
  1813. package/static/vendor/fontawesome-free/svgs/solid/tablets.svg +1 -0
  1814. package/static/vendor/fontawesome-free/svgs/solid/tachometer-alt.svg +1 -0
  1815. package/static/vendor/fontawesome-free/svgs/solid/tag.svg +1 -0
  1816. package/static/vendor/fontawesome-free/svgs/solid/tags.svg +1 -0
  1817. package/static/vendor/fontawesome-free/svgs/solid/tape.svg +1 -0
  1818. package/static/vendor/fontawesome-free/svgs/solid/tasks.svg +1 -0
  1819. package/static/vendor/fontawesome-free/svgs/solid/taxi.svg +1 -0
  1820. package/static/vendor/fontawesome-free/svgs/solid/teeth-open.svg +1 -0
  1821. package/static/vendor/fontawesome-free/svgs/solid/teeth.svg +1 -0
  1822. package/static/vendor/fontawesome-free/svgs/solid/temperature-high.svg +1 -0
  1823. package/static/vendor/fontawesome-free/svgs/solid/temperature-low.svg +1 -0
  1824. package/static/vendor/fontawesome-free/svgs/solid/tenge.svg +1 -0
  1825. package/static/vendor/fontawesome-free/svgs/solid/terminal.svg +1 -0
  1826. package/static/vendor/fontawesome-free/svgs/solid/text-height.svg +1 -0
  1827. package/static/vendor/fontawesome-free/svgs/solid/text-width.svg +1 -0
  1828. package/static/vendor/fontawesome-free/svgs/solid/th-large.svg +1 -0
  1829. package/static/vendor/fontawesome-free/svgs/solid/th-list.svg +1 -0
  1830. package/static/vendor/fontawesome-free/svgs/solid/th.svg +1 -0
  1831. package/static/vendor/fontawesome-free/svgs/solid/theater-masks.svg +1 -0
  1832. package/static/vendor/fontawesome-free/svgs/solid/thermometer-empty.svg +1 -0
  1833. package/static/vendor/fontawesome-free/svgs/solid/thermometer-full.svg +1 -0
  1834. package/static/vendor/fontawesome-free/svgs/solid/thermometer-half.svg +1 -0
  1835. package/static/vendor/fontawesome-free/svgs/solid/thermometer-quarter.svg +1 -0
  1836. package/static/vendor/fontawesome-free/svgs/solid/thermometer-three-quarters.svg +1 -0
  1837. package/static/vendor/fontawesome-free/svgs/solid/thermometer.svg +1 -0
  1838. package/static/vendor/fontawesome-free/svgs/solid/thumbs-down.svg +1 -0
  1839. package/static/vendor/fontawesome-free/svgs/solid/thumbs-up.svg +1 -0
  1840. package/static/vendor/fontawesome-free/svgs/solid/thumbtack.svg +1 -0
  1841. package/static/vendor/fontawesome-free/svgs/solid/ticket-alt.svg +1 -0
  1842. package/static/vendor/fontawesome-free/svgs/solid/times-circle.svg +1 -0
  1843. package/static/vendor/fontawesome-free/svgs/solid/times.svg +1 -0
  1844. package/static/vendor/fontawesome-free/svgs/solid/tint-slash.svg +1 -0
  1845. package/static/vendor/fontawesome-free/svgs/solid/tint.svg +1 -0
  1846. package/static/vendor/fontawesome-free/svgs/solid/tired.svg +1 -0
  1847. package/static/vendor/fontawesome-free/svgs/solid/toggle-off.svg +1 -0
  1848. package/static/vendor/fontawesome-free/svgs/solid/toggle-on.svg +1 -0
  1849. package/static/vendor/fontawesome-free/svgs/solid/toilet-paper-slash.svg +1 -0
  1850. package/static/vendor/fontawesome-free/svgs/solid/toilet-paper.svg +1 -0
  1851. package/static/vendor/fontawesome-free/svgs/solid/toilet.svg +1 -0
  1852. package/static/vendor/fontawesome-free/svgs/solid/toolbox.svg +1 -0
  1853. package/static/vendor/fontawesome-free/svgs/solid/tools.svg +1 -0
  1854. package/static/vendor/fontawesome-free/svgs/solid/tooth.svg +1 -0
  1855. package/static/vendor/fontawesome-free/svgs/solid/torah.svg +1 -0
  1856. package/static/vendor/fontawesome-free/svgs/solid/torii-gate.svg +1 -0
  1857. package/static/vendor/fontawesome-free/svgs/solid/tractor.svg +1 -0
  1858. package/static/vendor/fontawesome-free/svgs/solid/trademark.svg +1 -0
  1859. package/static/vendor/fontawesome-free/svgs/solid/traffic-light.svg +1 -0
  1860. package/static/vendor/fontawesome-free/svgs/solid/trailer.svg +1 -0
  1861. package/static/vendor/fontawesome-free/svgs/solid/train.svg +1 -0
  1862. package/static/vendor/fontawesome-free/svgs/solid/tram.svg +1 -0
  1863. package/static/vendor/fontawesome-free/svgs/solid/transgender-alt.svg +1 -0
  1864. package/static/vendor/fontawesome-free/svgs/solid/transgender.svg +1 -0
  1865. package/static/vendor/fontawesome-free/svgs/solid/trash-alt.svg +1 -0
  1866. package/static/vendor/fontawesome-free/svgs/solid/trash-restore-alt.svg +1 -0
  1867. package/static/vendor/fontawesome-free/svgs/solid/trash-restore.svg +1 -0
  1868. package/static/vendor/fontawesome-free/svgs/solid/trash.svg +1 -0
  1869. package/static/vendor/fontawesome-free/svgs/solid/tree.svg +1 -0
  1870. package/static/vendor/fontawesome-free/svgs/solid/trophy.svg +1 -0
  1871. package/static/vendor/fontawesome-free/svgs/solid/truck-loading.svg +1 -0
  1872. package/static/vendor/fontawesome-free/svgs/solid/truck-monster.svg +1 -0
  1873. package/static/vendor/fontawesome-free/svgs/solid/truck-moving.svg +1 -0
  1874. package/static/vendor/fontawesome-free/svgs/solid/truck-pickup.svg +1 -0
  1875. package/static/vendor/fontawesome-free/svgs/solid/truck.svg +1 -0
  1876. package/static/vendor/fontawesome-free/svgs/solid/tshirt.svg +1 -0
  1877. package/static/vendor/fontawesome-free/svgs/solid/tty.svg +1 -0
  1878. package/static/vendor/fontawesome-free/svgs/solid/tv.svg +1 -0
  1879. package/static/vendor/fontawesome-free/svgs/solid/umbrella-beach.svg +1 -0
  1880. package/static/vendor/fontawesome-free/svgs/solid/umbrella.svg +1 -0
  1881. package/static/vendor/fontawesome-free/svgs/solid/underline.svg +1 -0
  1882. package/static/vendor/fontawesome-free/svgs/solid/undo-alt.svg +1 -0
  1883. package/static/vendor/fontawesome-free/svgs/solid/undo.svg +1 -0
  1884. package/static/vendor/fontawesome-free/svgs/solid/universal-access.svg +1 -0
  1885. package/static/vendor/fontawesome-free/svgs/solid/university.svg +1 -0
  1886. package/static/vendor/fontawesome-free/svgs/solid/unlink.svg +1 -0
  1887. package/static/vendor/fontawesome-free/svgs/solid/unlock-alt.svg +1 -0
  1888. package/static/vendor/fontawesome-free/svgs/solid/unlock.svg +1 -0
  1889. package/static/vendor/fontawesome-free/svgs/solid/upload.svg +1 -0
  1890. package/static/vendor/fontawesome-free/svgs/solid/user-alt-slash.svg +1 -0
  1891. package/static/vendor/fontawesome-free/svgs/solid/user-alt.svg +1 -0
  1892. package/static/vendor/fontawesome-free/svgs/solid/user-astronaut.svg +1 -0
  1893. package/static/vendor/fontawesome-free/svgs/solid/user-check.svg +1 -0
  1894. package/static/vendor/fontawesome-free/svgs/solid/user-circle.svg +1 -0
  1895. package/static/vendor/fontawesome-free/svgs/solid/user-clock.svg +1 -0
  1896. package/static/vendor/fontawesome-free/svgs/solid/user-cog.svg +1 -0
  1897. package/static/vendor/fontawesome-free/svgs/solid/user-edit.svg +1 -0
  1898. package/static/vendor/fontawesome-free/svgs/solid/user-friends.svg +1 -0
  1899. package/static/vendor/fontawesome-free/svgs/solid/user-graduate.svg +1 -0
  1900. package/static/vendor/fontawesome-free/svgs/solid/user-injured.svg +1 -0
  1901. package/static/vendor/fontawesome-free/svgs/solid/user-lock.svg +1 -0
  1902. package/static/vendor/fontawesome-free/svgs/solid/user-md.svg +1 -0
  1903. package/static/vendor/fontawesome-free/svgs/solid/user-minus.svg +1 -0
  1904. package/static/vendor/fontawesome-free/svgs/solid/user-ninja.svg +1 -0
  1905. package/static/vendor/fontawesome-free/svgs/solid/user-nurse.svg +1 -0
  1906. package/static/vendor/fontawesome-free/svgs/solid/user-plus.svg +1 -0
  1907. package/static/vendor/fontawesome-free/svgs/solid/user-secret.svg +1 -0
  1908. package/static/vendor/fontawesome-free/svgs/solid/user-shield.svg +1 -0
  1909. package/static/vendor/fontawesome-free/svgs/solid/user-slash.svg +1 -0
  1910. package/static/vendor/fontawesome-free/svgs/solid/user-tag.svg +1 -0
  1911. package/static/vendor/fontawesome-free/svgs/solid/user-tie.svg +1 -0
  1912. package/static/vendor/fontawesome-free/svgs/solid/user-times.svg +1 -0
  1913. package/static/vendor/fontawesome-free/svgs/solid/user.svg +1 -0
  1914. package/static/vendor/fontawesome-free/svgs/solid/users-cog.svg +1 -0
  1915. package/static/vendor/fontawesome-free/svgs/solid/users-slash.svg +1 -0
  1916. package/static/vendor/fontawesome-free/svgs/solid/users.svg +1 -0
  1917. package/static/vendor/fontawesome-free/svgs/solid/utensil-spoon.svg +1 -0
  1918. package/static/vendor/fontawesome-free/svgs/solid/utensils.svg +1 -0
  1919. package/static/vendor/fontawesome-free/svgs/solid/vector-square.svg +1 -0
  1920. package/static/vendor/fontawesome-free/svgs/solid/venus-double.svg +1 -0
  1921. package/static/vendor/fontawesome-free/svgs/solid/venus-mars.svg +1 -0
  1922. package/static/vendor/fontawesome-free/svgs/solid/venus.svg +1 -0
  1923. package/static/vendor/fontawesome-free/svgs/solid/vest-patches.svg +1 -0
  1924. package/static/vendor/fontawesome-free/svgs/solid/vest.svg +1 -0
  1925. package/static/vendor/fontawesome-free/svgs/solid/vial.svg +1 -0
  1926. package/static/vendor/fontawesome-free/svgs/solid/vials.svg +1 -0
  1927. package/static/vendor/fontawesome-free/svgs/solid/video-slash.svg +1 -0
  1928. package/static/vendor/fontawesome-free/svgs/solid/video.svg +1 -0
  1929. package/static/vendor/fontawesome-free/svgs/solid/vihara.svg +1 -0
  1930. package/static/vendor/fontawesome-free/svgs/solid/virus-slash.svg +1 -0
  1931. package/static/vendor/fontawesome-free/svgs/solid/virus.svg +1 -0
  1932. package/static/vendor/fontawesome-free/svgs/solid/viruses.svg +1 -0
  1933. package/static/vendor/fontawesome-free/svgs/solid/voicemail.svg +1 -0
  1934. package/static/vendor/fontawesome-free/svgs/solid/volleyball-ball.svg +1 -0
  1935. package/static/vendor/fontawesome-free/svgs/solid/volume-down.svg +1 -0
  1936. package/static/vendor/fontawesome-free/svgs/solid/volume-mute.svg +1 -0
  1937. package/static/vendor/fontawesome-free/svgs/solid/volume-off.svg +1 -0
  1938. package/static/vendor/fontawesome-free/svgs/solid/volume-up.svg +1 -0
  1939. package/static/vendor/fontawesome-free/svgs/solid/vote-yea.svg +1 -0
  1940. package/static/vendor/fontawesome-free/svgs/solid/vr-cardboard.svg +1 -0
  1941. package/static/vendor/fontawesome-free/svgs/solid/walking.svg +1 -0
  1942. package/static/vendor/fontawesome-free/svgs/solid/wallet.svg +1 -0
  1943. package/static/vendor/fontawesome-free/svgs/solid/warehouse.svg +1 -0
  1944. package/static/vendor/fontawesome-free/svgs/solid/water.svg +1 -0
  1945. package/static/vendor/fontawesome-free/svgs/solid/wave-square.svg +1 -0
  1946. package/static/vendor/fontawesome-free/svgs/solid/weight-hanging.svg +1 -0
  1947. package/static/vendor/fontawesome-free/svgs/solid/weight.svg +1 -0
  1948. package/static/vendor/fontawesome-free/svgs/solid/wheelchair.svg +1 -0
  1949. package/static/vendor/fontawesome-free/svgs/solid/wifi.svg +1 -0
  1950. package/static/vendor/fontawesome-free/svgs/solid/wind.svg +1 -0
  1951. package/static/vendor/fontawesome-free/svgs/solid/window-close.svg +1 -0
  1952. package/static/vendor/fontawesome-free/svgs/solid/window-maximize.svg +1 -0
  1953. package/static/vendor/fontawesome-free/svgs/solid/window-minimize.svg +1 -0
  1954. package/static/vendor/fontawesome-free/svgs/solid/window-restore.svg +1 -0
  1955. package/static/vendor/fontawesome-free/svgs/solid/wine-bottle.svg +1 -0
  1956. package/static/vendor/fontawesome-free/svgs/solid/wine-glass-alt.svg +1 -0
  1957. package/static/vendor/fontawesome-free/svgs/solid/wine-glass.svg +1 -0
  1958. package/static/vendor/fontawesome-free/svgs/solid/won-sign.svg +1 -0
  1959. package/static/vendor/fontawesome-free/svgs/solid/wrench.svg +1 -0
  1960. package/static/vendor/fontawesome-free/svgs/solid/x-ray.svg +1 -0
  1961. package/static/vendor/fontawesome-free/svgs/solid/yen-sign.svg +1 -0
  1962. package/static/vendor/fontawesome-free/svgs/solid/yin-yang.svg +1 -0
  1963. package/static/vendor/fontawesome-free/webfonts/fa-brands-400.eot +0 -0
  1964. package/static/vendor/fontawesome-free/webfonts/fa-brands-400.svg +3717 -0
  1965. package/static/vendor/fontawesome-free/webfonts/fa-brands-400.ttf +0 -0
  1966. package/static/vendor/fontawesome-free/webfonts/fa-brands-400.woff +0 -0
  1967. package/static/vendor/fontawesome-free/webfonts/fa-brands-400.woff2 +0 -0
  1968. package/static/vendor/fontawesome-free/webfonts/fa-regular-400.eot +0 -0
  1969. package/static/vendor/fontawesome-free/webfonts/fa-regular-400.svg +801 -0
  1970. package/static/vendor/fontawesome-free/webfonts/fa-regular-400.ttf +0 -0
  1971. package/static/vendor/fontawesome-free/webfonts/fa-regular-400.woff +0 -0
  1972. package/static/vendor/fontawesome-free/webfonts/fa-regular-400.woff2 +0 -0
  1973. package/static/vendor/fontawesome-free/webfonts/fa-solid-900.eot +0 -0
  1974. package/static/vendor/fontawesome-free/webfonts/fa-solid-900.svg +5034 -0
  1975. package/static/vendor/fontawesome-free/webfonts/fa-solid-900.ttf +0 -0
  1976. package/static/vendor/fontawesome-free/webfonts/fa-solid-900.woff +0 -0
  1977. package/static/vendor/fontawesome-free/webfonts/fa-solid-900.woff2 +0 -0
  1978. package/static/vendor/handlebars/handlebars.min-v4.7.7.js +29 -0
  1979. package/static/vendor/jquery/jquery.js +10881 -0
  1980. package/static/vendor/jquery/jquery.min.js +2 -0
  1981. package/static/vendor/jquery/jquery.min.map +1 -0
  1982. package/static/vendor/jquery/jquery.slim.js +8782 -0
  1983. package/static/vendor/jquery/jquery.slim.min.js +2 -0
  1984. package/static/vendor/jquery/jquery.slim.min.map +1 -0
  1985. package/static/vendor/jquery-easing/jquery.easing.min.js +1 -0
  1986. package/systemd/emailengine.service +11 -3
  1987. package/systemd/nginx-proxy.conf +1 -1
  1988. package/test/api-test.js +899 -0
  1989. package/test/autoreply-test.js +327 -0
  1990. package/test/bounce-test.js +151 -0
  1991. package/test/complaint-test.js +256 -0
  1992. package/test/fixtures/autoreply/LICENSE +27 -0
  1993. package/test/fixtures/autoreply/rfc3834-01.eml +23 -0
  1994. package/test/fixtures/autoreply/rfc3834-02.eml +24 -0
  1995. package/test/fixtures/autoreply/rfc3834-03.eml +26 -0
  1996. package/test/fixtures/autoreply/rfc3834-04.eml +48 -0
  1997. package/test/fixtures/autoreply/rfc3834-05.eml +19 -0
  1998. package/test/fixtures/autoreply/rfc3834-06.eml +59 -0
  1999. package/test/fixtures/bounces/163.eml +2521 -0
  2000. package/test/fixtures/bounces/fastmail.eml +242 -0
  2001. package/test/fixtures/bounces/gmail.eml +252 -0
  2002. package/test/fixtures/bounces/hotmail.eml +655 -0
  2003. package/test/fixtures/bounces/mailru.eml +121 -0
  2004. package/test/fixtures/bounces/outlook.eml +1107 -0
  2005. package/test/fixtures/bounces/postfix.eml +101 -0
  2006. package/test/fixtures/bounces/rambler.eml +116 -0
  2007. package/test/fixtures/bounces/workmail.eml +142 -0
  2008. package/test/fixtures/bounces/yahoo.eml +139 -0
  2009. package/test/fixtures/bounces/zoho.eml +83 -0
  2010. package/test/fixtures/bounces/zonemta.eml +100 -0
  2011. package/test/fixtures/complaints/LICENSE +27 -0
  2012. package/test/fixtures/complaints/amazonses.eml +72 -0
  2013. package/test/fixtures/complaints/dmarc.eml +59 -0
  2014. package/test/fixtures/complaints/hotmail.eml +49 -0
  2015. package/test/fixtures/complaints/optout.eml +40 -0
  2016. package/test/fixtures/complaints/standard-arf.eml +68 -0
  2017. package/test/fixtures/complaints/yahoo.eml +68 -0
  2018. package/test/oauth2-apps-test.js +301 -0
  2019. package/test/sendonly-test.js +160 -0
  2020. package/test/test-config.js +34 -0
  2021. package/test/webhooks-server.js +39 -0
  2022. package/translations/README.md +16 -0
  2023. package/translations/de.mo +0 -0
  2024. package/translations/de.po +335 -0
  2025. package/translations/en.mo +0 -0
  2026. package/translations/en.po +310 -0
  2027. package/translations/et.mo +0 -0
  2028. package/translations/et.po +331 -0
  2029. package/translations/fr.mo +0 -0
  2030. package/translations/fr.po +333 -0
  2031. package/translations/ja.mo +0 -0
  2032. package/translations/ja.po +322 -0
  2033. package/translations/locales.json +43 -0
  2034. package/translations/messages.pot +323 -0
  2035. package/translations/nl.mo +0 -0
  2036. package/translations/nl.po +325 -0
  2037. package/translations/pl.mo +0 -0
  2038. package/translations/pl.po +328 -0
  2039. package/update-info.sh +10 -0
  2040. package/views/account/login.hbs +54 -0
  2041. package/views/account/password.hbs +88 -0
  2042. package/views/account/security.hbs +269 -0
  2043. package/views/account/totp.hbs +30 -0
  2044. package/views/accounts/account.hbs +1254 -0
  2045. package/views/accounts/browse.hbs +102 -0
  2046. package/views/accounts/edit.hbs +332 -0
  2047. package/views/accounts/index.hbs +143 -0
  2048. package/views/accounts/register/imap-server.hbs +507 -0
  2049. package/views/accounts/register/imap.hbs +56 -0
  2050. package/views/accounts/register/index.hbs +52 -0
  2051. package/views/arena/index.hbs +4 -0
  2052. package/views/config/ai.hbs +820 -0
  2053. package/views/config/document-store/chat.hbs +362 -0
  2054. package/views/config/document-store/index.hbs +231 -0
  2055. package/views/config/document-store/mappings/index.hbs +116 -0
  2056. package/views/config/document-store/mappings/new.hbs +95 -0
  2057. package/views/config/document-store/pre-processing/index.hbs +459 -0
  2058. package/views/config/imap-proxy.hbs +479 -0
  2059. package/views/config/license.hbs +256 -0
  2060. package/views/config/logging.hbs +61 -0
  2061. package/views/config/network.hbs +334 -0
  2062. package/views/config/oauth/app.hbs +309 -0
  2063. package/views/config/oauth/edit.hbs +92 -0
  2064. package/views/config/oauth/index.hbs +150 -0
  2065. package/views/config/oauth/new.hbs +90 -0
  2066. package/views/config/oauth.hbs +354 -0
  2067. package/views/config/service-preview.hbs +14 -0
  2068. package/views/config/service.hbs +718 -0
  2069. package/views/config/smtp.hbs +525 -0
  2070. package/views/config/webhooks.hbs +404 -0
  2071. package/views/dashboard.hbs +315 -0
  2072. package/views/error.hbs +6 -1
  2073. package/views/gateways/edit.hbs +52 -0
  2074. package/views/gateways/gateway.hbs +120 -0
  2075. package/views/gateways/index.hbs +152 -0
  2076. package/views/gateways/new.hbs +61 -0
  2077. package/views/index.hbs +21 -0
  2078. package/views/internals/index.hbs +170 -0
  2079. package/views/internals/thread.hbs +143 -0
  2080. package/views/layout/app.hbs +516 -0
  2081. package/views/layout/login.hbs +78 -0
  2082. package/views/layout/main.hbs +67 -0
  2083. package/views/layout/public.hbs +90 -0
  2084. package/views/legal.hbs +83 -0
  2085. package/views/license.hbs +5 -0
  2086. package/views/partials/accounts_header.hbs +6 -0
  2087. package/views/partials/add_account_modal.hbs +60 -0
  2088. package/views/partials/address_list.hbs +37 -0
  2089. package/views/partials/alerts.hbs +33 -0
  2090. package/views/partials/document_store_header.hbs +52 -0
  2091. package/views/partials/editor_scope_info.hbs +10 -0
  2092. package/views/partials/gateway_form.hbs +65 -0
  2093. package/views/partials/gateway_js.hbs +90 -0
  2094. package/views/partials/gateways_header.hbs +6 -0
  2095. package/views/partials/oauth_config_header.hbs +10 -0
  2096. package/views/partials/oauth_form.hbs +1204 -0
  2097. package/views/partials/scope_info.hbs +134 -0
  2098. package/views/partials/security_header.hbs +11 -0
  2099. package/views/partials/side_menu.hbs +114 -0
  2100. package/views/partials/template_form.hbs +121 -0
  2101. package/views/partials/templates_header.hbs +6 -0
  2102. package/views/partials/test_send.hbs +327 -0
  2103. package/views/partials/tokens_header.hbs +6 -0
  2104. package/views/partials/webhook_form.hbs +151 -0
  2105. package/views/partials/webhooks_editor_functions.hbs +372 -0
  2106. package/views/partials/webhooks_header.hbs +6 -0
  2107. package/views/redirect.hbs +1 -0
  2108. package/views/swagger/index.hbs +76 -0
  2109. package/views/templates/edit.hbs +87 -0
  2110. package/views/templates/index.hbs +208 -0
  2111. package/views/templates/new.hbs +85 -0
  2112. package/views/templates/template.hbs +423 -0
  2113. package/views/tokens/index.hbs +207 -0
  2114. package/views/tokens/new.hbs +230 -0
  2115. package/views/unsubscribe.hbs +93 -0
  2116. package/views/upgrade.hbs +56 -0
  2117. package/views/webhooks/edit.hbs +31 -0
  2118. package/views/webhooks/index.hbs +144 -0
  2119. package/views/webhooks/new.hbs +27 -0
  2120. package/views/webhooks/webhook.hbs +265 -0
  2121. package/winconf.js +93 -0
  2122. package/workers/api.js +8246 -1256
  2123. package/workers/documents.js +1120 -0
  2124. package/workers/imap-proxy.js +91 -0
  2125. package/workers/imap.js +552 -161
  2126. package/workers/smtp.js +355 -82
  2127. package/workers/submit.js +319 -54
  2128. package/workers/webhooks.js +542 -80
  2129. package/.eslintrc +0 -14
  2130. package/.github/FUNDING.yml +0 -4
  2131. package/.github/ISSUE_TEMPLATE/bug_report.md +0 -38
  2132. package/.github/ISSUE_TEMPLATE/feature_request.md +0 -20
  2133. package/LICENSE.txt +0 -661
  2134. package/examples/api.md +0 -137
  2135. package/lib/connection.js +0 -1769
  2136. package/lib/lua/z-push.lua +0 -14
  2137. package/lib/mailbox.js +0 -1546
  2138. package/license-report-config.json +0 -3
  2139. package/licenses.txt +0 -37
  2140. package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.css.map +0 -1
  2141. package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.css.map +0 -1
  2142. package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.min.css.map +0 -1
  2143. package/static/bootstrap-4.6.0-dist/css/bootstrap.css.map +0 -1
  2144. package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css +0 -7
  2145. package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css.map +0 -1
  2146. package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js +0 -7045
  2147. package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js.map +0 -1
  2148. package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js +0 -7
  2149. package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js.map +0 -1
  2150. package/static/bootstrap-4.6.0-dist/js/bootstrap.js.map +0 -1
  2151. package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js +0 -7
  2152. package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js.map +0 -1
  2153. package/static/js/emailengine.js +0 -581
  2154. package/workers/arena.js +0 -89
package/lib/account.js CHANGED
@@ -3,56 +3,372 @@
3
3
  const logger = require('./logger');
4
4
  const Boom = require('@hapi/boom');
5
5
  const msgpack = require('msgpack5')();
6
- const { normalizePath, formatAccountListingResponse } = require('./tools');
6
+ const {
7
+ normalizePath,
8
+ formatAccountListingResponse,
9
+ unpackUIDRangeForSearch,
10
+ mergeObjects,
11
+ download,
12
+ resolveCredentials,
13
+ isEmail,
14
+ resolveDelegatedAccount
15
+ } = require('./tools');
7
16
  const crypto = require('crypto');
8
17
  const { MessageChannel } = require('worker_threads');
9
18
  const { MessagePortReadable } = require('./message-port-stream');
10
19
  const { deepStrictEqual, strictEqual } = require('assert');
11
20
  const { encrypt, decrypt } = require('./encrypt');
12
- const { OAuth2Client } = require('google-auth-library');
21
+ const { oauth2Apps, LEGACY_KEYS } = require('./oauth2-apps');
22
+ const settings = require('./settings');
23
+ const redisScanDelete = require('./redis-scan-delete');
24
+ const { customAlphabet } = require('nanoid');
25
+ const Lock = require('ioredfour');
26
+ const nanoid = customAlphabet('0123456789abcdefghijklmnopqrstuvwxyz', 16);
27
+ const { REDIS_PREFIX, ACCOUNT_DELETED_NOTIFY, MAILBOX_HASH } = require('./consts');
28
+ const { mimeHtml } = require('@postalsys/email-text-tools');
29
+ const { GMAIL_API_SCOPES } = require('./oauth/gmail');
30
+ const { OUTLOOK_API_SCOPES } = require('./oauth/outlook');
13
31
 
14
32
  class Account {
15
33
  constructor(options) {
16
34
  this.redis = options.redis;
17
35
  this.account = options.account || false;
18
36
 
37
+ this.documentsQueue = options.documentsQueue || false;
38
+
19
39
  this.secret = options.secret;
20
40
 
41
+ this.timeout = options.timeout ? Number(options.timeout) : 0;
42
+
43
+ this.esClient = options.esClient;
44
+
21
45
  this.call = options.call; // async method to request data from parent
22
46
 
23
47
  this.logger = options.logger || logger;
24
48
  }
25
49
 
26
- async listAccounts(state, page, limit) {
50
+ getLock() {
51
+ if (!this.lock) {
52
+ this.lock = new Lock({
53
+ redis: this.redis,
54
+ namespace: 'ee'
55
+ });
56
+ }
57
+ return this.lock;
58
+ }
59
+
60
+ /**
61
+ * Checks OAuth2 scopes to determine account capabilities
62
+ *
63
+ * @param {string} provider - OAuth2 provider name (e.g., 'gmail', 'outlook')
64
+ * @param {Array<string>} scopes - Array of OAuth2 scope strings
65
+ * @returns {{hasSendScope: boolean, hasReadScope: boolean}} Object indicating send and read capabilities
66
+ *
67
+ * @example
68
+ * // Gmail send-only account
69
+ * checkAccountScopes('gmail', ['https://www.googleapis.com/auth/gmail.send'])
70
+ * // Returns: { hasSendScope: true, hasReadScope: false }
71
+ *
72
+ * @example
73
+ * // Gmail full access account
74
+ * checkAccountScopes('gmail', ['https://www.googleapis.com/auth/gmail.modify'])
75
+ * // Returns: { hasSendScope: false, hasReadScope: true }
76
+ *
77
+ * @example
78
+ * // Outlook send-only account (global cloud)
79
+ * checkAccountScopes('outlook', ['https://graph.microsoft.com/Mail.Send', 'offline_access'])
80
+ * // Returns: { hasSendScope: true, hasReadScope: false }
81
+ *
82
+ * @example
83
+ * // Outlook full access account (GCC-High cloud)
84
+ * checkAccountScopes('outlook', ['https://graph.microsoft.us/Mail.ReadWrite', 'https://graph.microsoft.us/Mail.Send'])
85
+ * // Returns: { hasSendScope: true, hasReadScope: true }
86
+ *
87
+ * @example
88
+ * // Outlook send-only account (DoD cloud)
89
+ * checkAccountScopes('outlook', ['https://dod-graph.microsoft.us/Mail.Send', 'offline_access'])
90
+ * // Returns: { hasSendScope: true, hasReadScope: false }
91
+ */
92
+ checkAccountScopes(provider, scopes) {
93
+ if (!scopes || !Array.isArray(scopes)) {
94
+ return { hasSendScope: false, hasReadScope: false };
95
+ }
96
+
97
+ if (provider === 'gmail') {
98
+ const hasSendScope = scopes.some(s => s.includes(GMAIL_API_SCOPES.send));
99
+ const hasReadScope = scopes.some(
100
+ s =>
101
+ s.includes(GMAIL_API_SCOPES.modify) ||
102
+ s.includes(GMAIL_API_SCOPES.readonly) ||
103
+ s.includes(GMAIL_API_SCOPES.labels) ||
104
+ s.includes('mail.google.com')
105
+ );
106
+ return { hasSendScope, hasReadScope };
107
+ }
108
+
109
+ if (provider === 'outlook') {
110
+ // Known Microsoft Graph API endpoints across different cloud environments
111
+ const msGraphDomains = [
112
+ 'graph.microsoft.com', // Global cloud
113
+ 'graph.microsoft.us', // GCC-High cloud
114
+ 'dod-graph.microsoft.us', // DoD cloud
115
+ 'microsoftgraph.chinacloudapi.cn' // China cloud
116
+ ];
117
+
118
+ // Normalize scopes by extracting the scope name from the full URL
119
+ // Supports multiple MS Graph endpoints (global, GCC-High, DoD, China)
120
+ // Examples:
121
+ // https://graph.microsoft.com/Mail.Send -> Mail.Send
122
+ // https://graph.microsoft.us/Mail.ReadWrite -> Mail.ReadWrite
123
+ // https://dod-graph.microsoft.us/Mail.Send -> Mail.Send
124
+ // https://microsoftgraph.chinacloudapi.cn/Mail.Send -> Mail.Send
125
+ // offline_access -> offline_access (passed through)
126
+ const normalizedScopes = scopes.map(s => {
127
+ // Handle plain scope names (e.g., offline_access, openid)
128
+ // These are not URLs and should be passed through as-is
129
+ if (!s.includes('://')) {
130
+ return s;
131
+ }
132
+
133
+ // Try to parse as URL to validate it's a Microsoft Graph endpoint
134
+ try {
135
+ const url = new URL(s);
136
+
137
+ // Validate protocol - must be https
138
+ if (url.protocol !== 'https:') {
139
+ this.logger.warn({
140
+ msg: 'Invalid protocol in MS Graph scope URL, expected https',
141
+ scope: s,
142
+ protocol: url.protocol
143
+ });
144
+ return s; // Return as-is for non-https URLs
145
+ }
146
+
147
+ // Check if this is a recognized MS Graph domain
148
+ if (msGraphDomains.includes(url.hostname)) {
149
+ // Extract scope name from path only (ignoring query params and fragments)
150
+ // Examples:
151
+ // /Mail.Send -> Mail.Send
152
+ // /Mail.Send?foo=bar -> Mail.Send
153
+ // /Mail.Send#section -> Mail.Send
154
+ // /Mail.Send/ -> Mail.Send (removes trailing slash)
155
+ const scopeName = url.pathname.substring(1).replace(/\/$/, '');
156
+ if (scopeName) {
157
+ return scopeName;
158
+ }
159
+ this.logger.warn({
160
+ msg: 'MS Graph scope URL has no scope name in path',
161
+ scope: s
162
+ });
163
+ }
164
+ } catch (err) {
165
+ // Invalid URL format
166
+ this.logger.warn({
167
+ msg: 'Failed to parse MS Graph scope URL',
168
+ scope: s,
169
+ err: err.message
170
+ });
171
+ }
172
+ // Return as-is if not a recognized Graph URL or parsing failed
173
+ return s;
174
+ });
175
+
176
+ const hasSendScope = normalizedScopes.some(s => s === OUTLOOK_API_SCOPES.send);
177
+ const hasReadScope = normalizedScopes.some(s => s === OUTLOOK_API_SCOPES.read || s === OUTLOOK_API_SCOPES.readWrite);
178
+ return { hasSendScope, hasReadScope };
179
+ }
180
+
181
+ return { hasSendScope: false, hasReadScope: false };
182
+ }
183
+
184
+ /**
185
+ * Determines if an account is configured for send-only mode
186
+ *
187
+ * This method handles ALL types of send-only accounts:
188
+ * 1. OAuth2 API accounts (Gmail/Outlook) with only send scopes
189
+ * 2. Traditional SMTP-only accounts (IMAP disabled or missing)
190
+ * 3. Delegated accounts (checks the parent account)
191
+ *
192
+ * @param {Object} accountData - Account configuration object
193
+ * @param {Object} [app] - OAuth2 app configuration object (optional, for OAuth2 accounts)
194
+ * @returns {boolean} True if account is send-only (can send but not read)
195
+ *
196
+ * @example
197
+ * // Send-only Gmail API account (has gmail.send scope only)
198
+ * isSendOnlyAccount(accountData, { provider: 'gmail' })
199
+ * // Returns: true
200
+ *
201
+ * @example
202
+ * // Send-only Outlook Graph API account (has Mail.Send scope only)
203
+ * isSendOnlyAccount(accountData, { provider: 'outlook' })
204
+ * // Returns: true
205
+ *
206
+ * @example
207
+ * // SMTP-only account (IMAP disabled)
208
+ * isSendOnlyAccount({ smtp: {...}, imap: { disabled: true } })
209
+ * // Returns: true
210
+ *
211
+ * @example
212
+ * // SMTP-only account (no IMAP configured)
213
+ * isSendOnlyAccount({ smtp: {...} })
214
+ * // Returns: true
215
+ *
216
+ * @example
217
+ * // Full access IMAP account
218
+ * isSendOnlyAccount({ smtp: {...}, imap: { host: 'imap.example.com' } })
219
+ * // Returns: false
220
+ */
221
+ /**
222
+ * Checks if an OAuth2 account has send-only scope configuration
223
+ * @private
224
+ * @param {Object} accountData - Account configuration object
225
+ * @param {string} provider - OAuth2 provider ('gmail' or 'outlook')
226
+ * @returns {boolean} True if account has send scope but not read scope
227
+ */
228
+ _hasOAuth2SendOnlyScopes(accountData, provider) {
229
+ const scopes = accountData.oauth2?.accessToken?.scope || accountData.oauth2?.scope || [];
230
+ const { hasSendScope, hasReadScope } = this.checkAccountScopes(provider, scopes);
231
+ return hasSendScope && !hasReadScope;
232
+ }
233
+
234
+ isSendOnlyAccount(accountData, app) {
235
+ // Check OAuth2 API accounts (Gmail, Outlook Graph API)
236
+ if (app && ['gmail', 'outlook'].includes(app.provider)) {
237
+ return this._hasOAuth2SendOnlyScopes(accountData, app.provider);
238
+ }
239
+
240
+ // Check traditional SMTP-only accounts
241
+ // An account is send-only if:
242
+ // - IMAP is explicitly disabled, OR
243
+ // - IMAP is not configured (missing or empty)
244
+ if (accountData.imap) {
245
+ // IMAP exists - check if it's disabled
246
+ return !!accountData.imap.disabled;
247
+ }
248
+
249
+ // No IMAP configuration at all
250
+ // Check if this is an OAuth2 account without an app parameter
251
+ // (delegated accounts or OAuth2 accounts where app lookup failed)
252
+ if (accountData.oauth2 && !app) {
253
+ // For OAuth2 without app info, we can't determine scope-based send-only
254
+ // Default to false (assume full access) unless IMAP is explicitly disabled
255
+ return false;
256
+ }
257
+
258
+ // No IMAP, no OAuth2 - this is a traditional SMTP-only account
259
+ // Return true only if SMTP is configured (otherwise it's an invalid config)
260
+ return !!accountData.smtp;
261
+ }
262
+
263
+ async listAccounts(state, query, page, limit) {
27
264
  limit = Number(limit) || 20;
28
265
  page = Math.max(Number(page) || 0, 0);
29
266
  let skip = page * limit;
30
267
 
31
- let result = await this.redis.sListAccounts('ia:accounts', state || '*', skip, limit);
268
+ const runIndex = await this.call({
269
+ cmd: 'runIndex'
270
+ });
271
+
272
+ let result = await this.redis.sListAccounts(`${REDIS_PREFIX}ia:accounts`, state || '*', skip, limit, `${REDIS_PREFIX}`, query);
273
+
274
+ let accounts = result[2].map(formatAccountListingResponse).map(this.unserializeAccountData.bind(this));
275
+ let oauthApps = new Map();
276
+
277
+ for (let accountData of accounts) {
278
+ if (accountData.oauth2 && accountData.oauth2.provider) {
279
+ let app;
280
+ if (oauthApps.has(accountData.oauth2.provider)) {
281
+ app = oauthApps.get(accountData.oauth2.provider);
282
+ } else {
283
+ app = await oauth2Apps.get(accountData.oauth2.provider);
284
+ }
285
+ oauthApps.set(accountData.oauth2.provider, app || null);
286
+ if (app) {
287
+ accountData.type = app.provider;
288
+ if (app.baseScopes === 'api') {
289
+ accountData.isApi = true;
290
+ }
291
+
292
+ // Check if this is a send-only OAuth2 API account (Gmail or Outlook, regardless of baseScopes setting)
293
+ if (this.isSendOnlyAccount(accountData, app)) {
294
+ accountData.isApi = true;
295
+ accountData.sendOnly = true;
296
+ }
297
+ } else {
298
+ accountData.type = 'oauth2';
299
+ }
300
+ } else if (accountData.oauth2 && accountData.oauth2.auth && accountData.oauth2.auth.delegatedAccount) {
301
+ accountData.type = 'delegated';
302
+ let delegatedAccount = await resolveDelegatedAccount(this.redis, accountData.account);
303
+ if (delegatedAccount) {
304
+ accountData.delegatedAccount = delegatedAccount;
305
+ let delegatedAccountRow = await this.redis.hgetall(`${REDIS_PREFIX}iad:${delegatedAccount}`);
306
+ let delegatedAccountData = this.unserializeAccountData(delegatedAccountRow);
307
+ if (delegatedAccountData.oauth2 && delegatedAccountData.oauth2.provider) {
308
+ let app;
309
+ if (oauthApps.has(delegatedAccountData.oauth2.provider)) {
310
+ app = oauthApps.get(delegatedAccountData.oauth2.provider);
311
+ } else {
312
+ app = await oauth2Apps.get(delegatedAccountData.oauth2.provider);
313
+ }
314
+ oauthApps.set(delegatedAccountData.oauth2.provider, app || null);
315
+ if (app && app.baseScopes === 'api') {
316
+ accountData.isApi = true;
317
+ }
318
+ }
319
+ }
320
+ } else if (accountData.imap && !accountData.imap.disabled) {
321
+ accountData.type = 'imap';
322
+ } else {
323
+ // Default to 'sending' type for accounts without IMAP
324
+ // Use unified detection to determine if truly send-only
325
+ accountData.type = 'sending';
326
+ accountData.sendOnly = this.isSendOnlyAccount(accountData);
327
+ }
328
+ }
32
329
 
33
330
  let list = {
34
331
  total: result[0],
35
332
  pages: Math.ceil(result[0] / limit),
36
333
  page,
37
- accounts: result[2]
38
- .map(formatAccountListingResponse)
39
- .map(this.unserializeAccountData.bind(this))
40
- .map(entry => ({
41
- account: entry.account,
42
- name: entry.name,
43
- state: entry.state,
44
- syncTime: entry.sync,
45
- lastError: entry.state === 'connected' ? null : entry.lastErrorState
46
- }))
334
+ query: query || false,
335
+ state: state || '*',
336
+ accounts: accounts.map(accountData => ({
337
+ account: accountData.account,
338
+ name: accountData.name,
339
+ email: accountData.email,
340
+ type: accountData.type,
341
+ app:
342
+ accountData.oauth2 && accountData.oauth2.provider && accountData.oauth2.provider !== accountData.type
343
+ ? accountData.oauth2.provider
344
+ : undefined,
345
+ state:
346
+ !runIndex || runIndex <= accountData.runIndex || ['init', 'unset'].includes(accountData.state) || accountData.isApi
347
+ ? accountData.state
348
+ : 'init',
349
+
350
+ webhooks: accountData.webhooks || undefined,
351
+ proxy: accountData.proxy || undefined,
352
+ smtpEhloName: accountData.smtpEhloName || undefined,
353
+
354
+ counters: accountData.counters,
355
+
356
+ syncTime: accountData.sync,
357
+
358
+ lastError:
359
+ accountData.state === 'connected' || !accountData.lastErrorState || !Object.keys(accountData.lastErrorState).length
360
+ ? null
361
+ : accountData.lastErrorState
362
+ }))
47
363
  };
48
364
 
49
365
  return list;
50
366
  }
51
367
 
52
368
  async getMailboxInfo(path) {
53
- let redisKey = BigInt('0x' + crypto.createHash('sha1').update(normalizePath(path)).digest('hex')).toString(36);
369
+ let redisKey = BigInt('0x' + crypto.createHash(MAILBOX_HASH).update(normalizePath(path)).digest('hex')).toString(36);
54
370
 
55
- let data = await this.redis.hgetall(`iam:${this.account}:h:${redisKey}`);
371
+ let data = await this.redis.hgetall(`${REDIS_PREFIX}iam:${this.account}:h:${redisKey}`);
56
372
  if (!data || !Object.keys(data).length) {
57
373
  return {};
58
374
  }
@@ -70,49 +386,175 @@ class Account {
70
386
  }
71
387
 
72
388
  getAccountKey() {
73
- return `iad:${this.account}`;
389
+ return `${REDIS_PREFIX}iad:${this.account}`;
74
390
  }
75
391
 
76
392
  getMailboxListKey() {
77
- return `ial:${this.account}`;
393
+ return `${REDIS_PREFIX}ial:${this.account}`;
394
+ }
395
+
396
+ getMailboxHashKey() {
397
+ return `${REDIS_PREFIX}iah:${this.account}`;
398
+ }
399
+
400
+ getLogKey() {
401
+ // this format ensures that the key is deleted when user is removed
402
+ return `${REDIS_PREFIX}iam:${this.account}:g`;
403
+ }
404
+
405
+ getExternalQueueKey() {
406
+ return `${REDIS_PREFIX}iap:${this.account}`;
78
407
  }
79
408
 
80
409
  unserializeAccountData(accountData) {
81
- let result = {};
410
+ const result = {};
411
+
412
+ const counters = {};
82
413
 
83
414
  Object.keys(accountData).forEach(key => {
415
+ if (key.startsWith('_')) {
416
+ return; //ignore
417
+ }
418
+
419
+ let countMatch = key.match(/^stats:count:([^:]+):([^:]+)/);
420
+ if (countMatch) {
421
+ const [, type, counter] = countMatch;
422
+ if (!counters[type]) {
423
+ counters[type] = {};
424
+ }
425
+
426
+ if (!counters[type][counter]) {
427
+ counters[type][counter] = 0;
428
+ }
429
+
430
+ counters[type][counter] = Number(accountData[key]);
431
+ return;
432
+ }
433
+
84
434
  switch (key) {
85
435
  case 'notifyFrom':
436
+ case 'syncFrom':
437
+ case 'lastWatch':
86
438
  // Date object
87
439
  if (accountData[key]) {
88
- let date = new Date(accountData[key]);
89
- if (date.toString() !== 'Invalid Date') {
90
- result[key] = date;
440
+ if (accountData[key] === 'null') {
441
+ result[key] = null;
442
+ } else {
443
+ let date = new Date(accountData[key]);
444
+ if (date.toString() !== 'Invalid Date') {
445
+ result[key] = date;
446
+ }
91
447
  }
92
448
  }
93
449
  break;
94
450
 
451
+ // boolean values
95
452
  case 'copy':
96
- result[key] = accountData[key] === 'true' ? true : false;
453
+ case 'logs':
454
+ if (accountData[key] && accountData[key] !== 'null') {
455
+ result[key] = accountData[key] === 'true' ? true : false;
456
+ }
457
+ break;
458
+
459
+ case 'path':
460
+ if (!accountData[key] || !accountData[key].length) {
461
+ break;
462
+ }
463
+ if (accountData[key].length > 1 && (/^"|"$/.test(accountData[key]) || /^\[|\]$/.test(accountData[key]))) {
464
+ // seems like JSON array or string
465
+ try {
466
+ let value = JSON.parse(accountData[key]);
467
+ if (value === null) {
468
+ break;
469
+ }
470
+ result[key] = value;
471
+ } catch (err) {
472
+ this.logger.error({ msg: 'Failed to parse input from Redis', key, value: accountData[key], err });
473
+ }
474
+ } else {
475
+ // regular string
476
+ result[key] = accountData[key];
477
+ }
478
+ break;
479
+
480
+ // generic JSON values
481
+ case 'webhooksCustomHeaders':
482
+ case 'subconnections':
483
+ case 'watchResponse':
484
+ case 'watchFailure':
485
+ try {
486
+ let value = JSON.parse(accountData[key]);
487
+ if (value === null) {
488
+ break;
489
+ }
490
+ result[key] = value;
491
+ } catch (err) {
492
+ this.logger.error({ msg: 'Failed to parse input from Redis', key, value: accountData[key], err });
493
+ }
494
+ break;
495
+
496
+ case 'outlookSubscription':
497
+ try {
498
+ let value = JSON.parse(accountData[key]);
499
+ if (value === null) {
500
+ break;
501
+ }
502
+
503
+ if (value.expirationDateTime) {
504
+ let expirationDateTime = new Date(value.expirationDateTime);
505
+ if (expirationDateTime.toString() === 'Invalid Date') {
506
+ expirationDateTime = null;
507
+ }
508
+ value.expirationDateTime = expirationDateTime;
509
+ }
510
+
511
+ result[key] = value;
512
+ } catch (err) {
513
+ this.logger.error({ msg: 'Failed to parse input from Redis', key, value: accountData[key], err });
514
+ }
97
515
  break;
98
516
 
99
517
  case 'imap':
100
518
  case 'smtp':
519
+ case 'imapServerInfo':
520
+ case 'smtpServerEhlo':
101
521
  case 'oauth2':
102
522
  case 'lastErrorState':
523
+ case 'syncError':
524
+ case 'smtpStatus':
525
+ case 'webhookErrorFlag':
103
526
  try {
104
- result[key] = JSON.parse(accountData[key]);
105
- if (result[key].expires) {
106
- let date = new Date(result[key].expires);
107
- if (date.toString() !== 'Invalid Date') {
108
- result[key].expires = date;
527
+ let value = JSON.parse(accountData[key]);
528
+ if (value === null) {
529
+ break;
530
+ }
531
+ result[key] = value;
532
+ for (let subKey of ['created', 'expires', 'generated']) {
533
+ if (result[key][subKey]) {
534
+ let dateVal = /^[0-9]+$/.test(result[key][subKey]) ? Number(result[key][subKey]) : result[key][subKey];
535
+ let date = new Date(dateVal);
536
+ if (date.toString() !== 'Invalid Date') {
537
+ result[key][subKey] = date;
538
+ }
539
+ }
540
+ }
541
+ for (let subKey of Object.keys(result[key])) {
542
+ if (result[key][subKey] === null) {
543
+ delete result[key][subKey];
109
544
  }
110
545
  }
111
546
  } catch (err) {
112
- this.logger.error({ msg: 'Failed to parse input from Redis', key, err });
547
+ this.logger.error({ msg: 'Failed to parse input from Redis', key, value: accountData[key], err });
113
548
  }
114
549
  break;
115
550
 
551
+ // number values
552
+ case 'connections':
553
+ case 'listRegistry':
554
+ case 'runIndex':
555
+ result[key] = Number(accountData[key]) || 0;
556
+ break;
557
+
116
558
  default:
117
559
  result[key] = accountData[key];
118
560
  break;
@@ -149,6 +591,17 @@ class Account {
149
591
  result.path = '*';
150
592
  }
151
593
 
594
+ if (typeof result.account === 'undefined') {
595
+ result.account = null;
596
+ }
597
+
598
+ // falls back to oauth2 username as the email address if email address is not set
599
+ if (!result.email && result?.oauth2?.auth?.user && isEmail(result?.oauth2?.auth?.user)) {
600
+ result.email = result.oauth2.auth.user;
601
+ }
602
+
603
+ result.counters = counters;
604
+
152
605
  return result;
153
606
  }
154
607
 
@@ -156,14 +609,91 @@ class Account {
156
609
  let result = {};
157
610
 
158
611
  Object.keys(accountData).forEach(key => {
612
+ if (key.startsWith('_')) {
613
+ // ignore
614
+ return;
615
+ }
159
616
  switch (key) {
160
617
  case 'notifyFrom':
618
+ case 'syncFrom':
619
+ case 'lastWatch':
161
620
  // Date object
162
621
  if (accountData[key] === 'now') {
163
622
  result[key] = new Date().toISOString();
164
- } else if (accountData[key] && typeof accountData[key] === 'object' && accountData[key].toString() !== 'Invalid Date') {
623
+ } else if (
624
+ accountData[key] &&
625
+ typeof accountData[key] === 'object' &&
626
+ typeof accountData[key].toISOString === 'function' &&
627
+ accountData[key].toString() !== 'Invalid Date'
628
+ ) {
165
629
  result[key] = accountData[key].toISOString();
630
+ } else if (typeof accountData[key] === 'string') {
631
+ let date = new Date(accountData[key]);
632
+ if (date.toString() !== 'Invalid Date') {
633
+ result[key] = date.toISOString();
634
+ }
635
+ } else if (accountData[key] === null) {
636
+ result[key] = 'null';
637
+ }
638
+ break;
639
+
640
+ case 'path':
641
+ if (
642
+ accountData[key] === null ||
643
+ accountData[key] === '*' ||
644
+ (Array.isArray(accountData[key]) && (accountData[key].length === 0 || (accountData[key].length === 1 && accountData[key][0] === '*')))
645
+ ) {
646
+ result[key] = '';
647
+ } else {
648
+ try {
649
+ result[key] = JSON.stringify(accountData[key]);
650
+ } catch (err) {
651
+ this.logger.error({ msg: 'Failed to stringify input for Redis', key, err });
652
+ }
653
+ }
654
+ break;
655
+
656
+ // generic JSON values
657
+ case 'subconnections':
658
+ case 'webhooksCustomHeaders':
659
+ case 'watchResponse':
660
+ case 'watchFailure':
661
+ try {
662
+ let value = JSON.stringify(accountData[key]);
663
+
664
+ result[key] = value;
665
+ } catch (err) {
666
+ this.logger.error({ msg: 'Failed to stringify input for Redis', key, err });
667
+ }
668
+
669
+ break;
670
+
671
+ case 'outlookSubscription':
672
+ try {
673
+ let objValue = Object.assign({}, accountData[key]);
674
+
675
+ if (['number', 'string'].includes(typeof objValue.expirationDateTime)) {
676
+ objValue.expirationDateTime = new Date(objValue.expirationDateTime);
677
+ }
678
+
679
+ if (
680
+ objValue.expirationDateTime &&
681
+ typeof objValue.expirationDateTime === 'object' &&
682
+ typeof objValue.expirationDateTime.toISOString === 'function' &&
683
+ objValue.expirationDateTime.toString() !== 'Invalid Date'
684
+ ) {
685
+ objValue.expirationDateTime = objValue.expirationDateTime.toISOString();
686
+ } else if (objValue.expirationDateTime) {
687
+ objValue.expirationDateTime = null;
688
+ }
689
+
690
+ let value = JSON.stringify(objValue);
691
+
692
+ result[key] = value;
693
+ } catch (err) {
694
+ this.logger.error({ msg: 'Failed to stringify input for Redis', key, err });
166
695
  }
696
+
167
697
  break;
168
698
 
169
699
  case 'imap':
@@ -192,12 +722,14 @@ class Account {
192
722
  }
193
723
  }
194
724
 
195
- if (
196
- accountData[key].expires &&
197
- typeof accountData[key].expires === 'object' &&
198
- accountData[key].expires.toString() !== 'Invalid Date'
199
- ) {
200
- connectData.expires = accountData[key].expires.toISOString();
725
+ for (let subKey of ['created', 'expires', 'generated']) {
726
+ if (
727
+ accountData[key][subKey] &&
728
+ typeof accountData[key][subKey] === 'object' &&
729
+ accountData[key][subKey].toString() !== 'Invalid Date'
730
+ ) {
731
+ connectData[subKey] = accountData[key][subKey].toISOString();
732
+ }
201
733
  }
202
734
 
203
735
  result[key] = JSON.stringify(connectData);
@@ -206,8 +738,28 @@ class Account {
206
738
  }
207
739
  break;
208
740
 
741
+ case 'webhooks':
742
+ if (typeof accountData[key] !== 'undefined' && accountData[key] !== null && typeof accountData[key].toString === 'function') {
743
+ result[key] = accountData[key].toString();
744
+ if (!result[key]) {
745
+ // clear potential error flag
746
+ result.webhookErrorFlag = '{}';
747
+ }
748
+ }
749
+ break;
750
+
751
+ // boolean values
752
+ case 'copy':
753
+ case 'logs':
754
+ if (typeof accountData[key] === 'boolean') {
755
+ result[key] = accountData[key].toString();
756
+ } else if (accountData[key] === null) {
757
+ result[key] = '';
758
+ }
759
+ break;
760
+
209
761
  default:
210
- if (accountData[key] !== undefined && typeof accountData[key].toString === 'function') {
762
+ if (typeof accountData[key] !== 'undefined' && accountData[key] !== null && typeof accountData[key].toString === 'function') {
211
763
  result[key] = accountData[key].toString();
212
764
  }
213
765
  break;
@@ -217,7 +769,7 @@ class Account {
217
769
  return result;
218
770
  }
219
771
 
220
- async loadAccountData(account, requireValid) {
772
+ async loadAccountData(account, requireValid, runIndex) {
221
773
  if (!this.account || (account && account !== this.account)) {
222
774
  let message = 'Invalid account ID';
223
775
  let error = Boom.boomify(new Error(message), { statusCode: 400 });
@@ -233,23 +785,76 @@ class Account {
233
785
  }
234
786
 
235
787
  let accountData = this.unserializeAccountData(result);
236
- if (requireValid && accountData.state !== 'connected') {
788
+
789
+ if (accountData.oauth2 && accountData.oauth2.provider) {
790
+ let app = await oauth2Apps.get(accountData.oauth2.provider);
791
+ if (app) {
792
+ if (app.baseScopes === 'api') {
793
+ accountData.isApi = true;
794
+ }
795
+
796
+ // Check if this is a send-only OAuth2 API account (Gmail or Outlook, regardless of baseScopes setting)
797
+ if (this.isSendOnlyAccount(accountData, app)) {
798
+ accountData.isApi = true;
799
+ accountData.sendOnly = true;
800
+ }
801
+
802
+ accountData._app = app;
803
+ }
804
+ } else if (accountData.oauth2 && accountData.oauth2.auth && accountData.oauth2.auth.delegatedAccount) {
805
+ let delegatedAccount = await resolveDelegatedAccount(this.redis, accountData.account);
806
+ if (delegatedAccount) {
807
+ accountData.delegatedAccount = delegatedAccount;
808
+ let delegatedAccountRow = await this.redis.hgetall(`${REDIS_PREFIX}iad:${delegatedAccount}`);
809
+ let delegatedAccountData = this.unserializeAccountData(delegatedAccountRow);
810
+ if (delegatedAccountData.oauth2 && delegatedAccountData.oauth2.provider) {
811
+ let app = await oauth2Apps.get(delegatedAccountData.oauth2.provider);
812
+ if (app) {
813
+ accountData._app = app;
814
+ if (app.baseScopes === 'api') {
815
+ accountData.isApi = true;
816
+ }
817
+ }
818
+ }
819
+ }
820
+ }
821
+
822
+ accountData.state =
823
+ !runIndex || runIndex <= accountData.runIndex || ['init', 'unset'].includes(accountData.state) || accountData.isApi ? accountData.state : 'init';
824
+
825
+ if (requireValid && !['connected', 'connecting', 'syncing'].includes(accountData.state)) {
237
826
  let err;
238
827
  switch (accountData.state) {
239
828
  case 'init':
240
829
  err = new Error('Requested account is not yet initialized');
830
+ err.code = 'NotYetConnected';
241
831
  break;
832
+ /*
833
+ // Check disabled for the following states - allow commands to go through.
834
+ // A secondary IMAP connection is opened if possible.
835
+ */
836
+ /*
242
837
  case 'connecting':
838
+ case 'syncing':
243
839
  err = new Error('Requested account is not yet connected');
840
+ err.code = 'NotYetConnected';
244
841
  break;
842
+ */
245
843
  case 'authenticationError':
246
844
  err = new Error('Requested account can not be authenticated');
845
+ err.code = 'AuthenticationFails';
247
846
  break;
248
847
  case 'connectError':
249
848
  err = new Error('Can not establish server connection for requested account');
849
+ err.code = 'ConnectionError';
850
+ break;
851
+ case 'unset':
852
+ err = new Error('Syncing is disabled for the requested account');
853
+ err.code = 'NotSyncing';
250
854
  break;
251
855
  default:
252
856
  err = new Error('Requested account currently not available');
857
+ err.code = 'NoAvailable';
253
858
  break;
254
859
  }
255
860
 
@@ -257,6 +862,9 @@ class Account {
257
862
  if (accountData.state) {
258
863
  error.output.payload.state = accountData.state;
259
864
  }
865
+ if (err.code) {
866
+ error.output.payload.code = err.code;
867
+ }
260
868
  throw error;
261
869
  }
262
870
 
@@ -266,32 +874,161 @@ class Account {
266
874
  async update(accountData) {
267
875
  let oldAccountData = await this.loadAccountData(accountData.account);
268
876
 
269
- let result = await this.redis.hmset(this.getAccountKey(), this.serializeAccountData(accountData));
877
+ if (accountData.oauth2?.provider) {
878
+ // check if this OAuth2 provider exists
879
+ let oauth2App = await oauth2Apps.get(accountData.oauth2.provider);
880
+ if (!oauth2App) {
881
+ let message = 'Invalid or missing OAuth2 provider';
882
+ let error = Boom.boomify(new Error(message), { statusCode: 400 });
883
+ throw error;
884
+ }
885
+ }
886
+
887
+ let removeProvider;
888
+ let addProvider;
889
+
890
+ if (accountData.oauth2?.provider) {
891
+ addProvider = accountData.oauth2.provider;
892
+ if (oldAccountData.oauth2?.provider && oldAccountData.oauth2.provider !== accountData.oauth2.provider) {
893
+ removeProvider = oldAccountData.oauth2.provider;
894
+ }
895
+ }
896
+
897
+ for (let subKey of ['imap', 'smtp', 'oauth2']) {
898
+ if (!accountData[subKey] || typeof accountData[subKey] !== 'object') {
899
+ continue;
900
+ }
901
+ let partial = accountData[subKey].partial;
902
+ delete accountData[subKey].partial;
903
+ if (!partial) {
904
+ continue;
905
+ }
906
+
907
+ // merge old and new values
908
+ if (!oldAccountData[subKey]) {
909
+ // nothing to merge
910
+ continue;
911
+ }
912
+
913
+ mergeObjects(accountData[subKey], oldAccountData[subKey]);
914
+ }
915
+
916
+ let pipeline = this.redis.multi().hmset(this.getAccountKey(), this.serializeAccountData(accountData));
917
+
918
+ if (addProvider && !LEGACY_KEYS.includes(addProvider)) {
919
+ pipeline = pipeline.sadd(`${REDIS_PREFIX}oapp:a:${addProvider}`, this.account);
920
+ if (accountData._oldOAuth2User && accountData._oldOAuth2User.toLowerCase() !== accountData.oauth2.auth?.user?.toLowerCase()) {
921
+ pipeline = pipeline.hdel(`${REDIS_PREFIX}oapp:h:${addProvider}`, accountData._oldOAuth2User.toLowerCase());
922
+ }
923
+ if (accountData.oauth2?.auth?.user) {
924
+ // check if already exists
925
+ let existingAppBinding = await this.redis.hget(
926
+ `${REDIS_PREFIX}oapp:h:${accountData.oauth2.provider}`,
927
+ accountData.oauth2.auth?.user?.toLowerCase()
928
+ );
929
+ if (existingAppBinding && existingAppBinding !== this.account) {
930
+ let existingAccount;
931
+ try {
932
+ existingAccount = await this.loadAccountData(existingAppBinding);
933
+ } catch (err) {
934
+ // account not found
935
+ }
936
+
937
+ if (existingAccount?.oauth2?.auth?.user === accountData.oauth2.auth?.user) {
938
+ let message = 'Another account for the same OAuth2 user already exists';
939
+ let error = Boom.boomify(new Error(message), { statusCode: 400 });
940
+ error.output.payload.code = 'AccountAlreadyExists';
941
+ error.output.payload.existingAccount = existingAppBinding;
942
+ throw error;
943
+ }
944
+ }
945
+ pipeline = pipeline.hset(`${REDIS_PREFIX}oapp:h:${addProvider}`, accountData.oauth2?.auth?.user?.toLowerCase(), this.account);
946
+ }
947
+ }
948
+
949
+ if (removeProvider) {
950
+ pipeline = pipeline.srem(`${REDIS_PREFIX}oapp:a:${removeProvider}`, this.account);
951
+ if (oldAccountData.oauth2?.auth?.user) {
952
+ pipeline = pipeline.hdel(`${REDIS_PREFIX}oapp:h:${removeProvider}`, oldAccountData.oauth2?.auth?.user?.toLowerCase());
953
+ }
954
+ }
955
+
956
+ let [[err, result]] = await pipeline.exec();
957
+ if (err) {
958
+ throw err;
959
+ }
960
+
270
961
  if (!result || result !== 'OK') {
271
962
  let message = 'Something went wrong';
272
963
  let error = Boom.boomify(new Error(message), { statusCode: 500 });
273
964
  throw error;
274
965
  }
275
966
 
276
- if ('imap' in accountData) {
277
- // if partial update, then skip check
967
+ let reconnectRequested = false;
968
+
969
+ if ('imap' in accountData && !reconnectRequested) {
278
970
  try {
279
971
  deepStrictEqual(oldAccountData.imap, accountData.imap);
280
- strictEqual(oldAccountData.path || '*', accountData.path || '*');
281
972
  } catch (err) {
282
973
  // changes detected!
283
- this.logger.info({ msg: 'IMAP configuration changed for account', account: this.account });
284
- await this.call({ cmd: 'update', account: this.account });
974
+ reconnectRequested = true;
285
975
  }
286
976
  }
287
977
 
978
+ if ('path' in accountData && !reconnectRequested) {
979
+ try {
980
+ strictEqual(JSON.stringify(oldAccountData.path || '*'), JSON.stringify(accountData.path || '*'));
981
+ } catch (err) {
982
+ // changes detected!
983
+ reconnectRequested = true;
984
+ }
985
+ }
986
+
987
+ if ('subconnections' in accountData && !reconnectRequested) {
988
+ try {
989
+ deepStrictEqual(oldAccountData.subconnections, accountData.subconnections);
990
+ } catch (err) {
991
+ // changes detected!
992
+ reconnectRequested = true;
993
+ }
994
+ }
995
+
996
+ if (reconnectRequested) {
997
+ // changes detected!
998
+ this.logger.info({ msg: 'IMAP configuration changed for account', account: this.account });
999
+ await this.call({
1000
+ cmd: 'update',
1001
+ account: this.account,
1002
+ timeout: this.timeout
1003
+ });
1004
+ }
1005
+
288
1006
  return {
289
1007
  account: this.account
290
1008
  };
291
1009
  }
292
1010
 
1011
+ async genId() {
1012
+ let id;
1013
+ let retries = 0;
1014
+ while (retries++ < 20) {
1015
+ id = nanoid();
1016
+ let alreadyExists = await this.redis.hexists(`${REDIS_PREFIX}iad:${id}`, 'account');
1017
+ if (alreadyExists) {
1018
+ id = false;
1019
+ } else {
1020
+ break;
1021
+ }
1022
+ }
1023
+ return id;
1024
+ }
1025
+
293
1026
  async create(accountData) {
294
1027
  this.account = accountData.account;
1028
+ if (this.account === null) {
1029
+ // auogenerate ID
1030
+ this.account = accountData.account = await this.genId();
1031
+ }
295
1032
 
296
1033
  if (!this.account) {
297
1034
  let message = 'Invalid account ID';
@@ -299,13 +1036,64 @@ class Account {
299
1036
  throw error;
300
1037
  }
301
1038
 
302
- let result = await this.redis
1039
+ let oauth2App;
1040
+ if (accountData.oauth2 && accountData.oauth2.provider) {
1041
+ // check if this OAuth2 provider exists
1042
+ oauth2App = await oauth2Apps.get(accountData.oauth2.provider);
1043
+ if (!oauth2App) {
1044
+ let message = 'Invalid or missing OAuth2 provider';
1045
+ let error = Boom.boomify(new Error(message), { statusCode: 400 });
1046
+ throw error;
1047
+ }
1048
+ }
1049
+
1050
+ if (!accountData.imapIndexer && (accountData.imap || (oauth2App && (!oauth2App.baseScopes || oauth2App.baseScopes === 'imap')))) {
1051
+ accountData.imapIndexer = (await settings.get('imapIndexer')) || 'full';
1052
+ }
1053
+
1054
+ const runIndex = await this.call({
1055
+ cmd: 'runIndex'
1056
+ });
1057
+
1058
+ let pipeline = this.redis
303
1059
  .multi()
304
1060
  .hgetall(this.getAccountKey())
305
1061
  .hmset(this.getAccountKey(), this.serializeAccountData(accountData))
306
1062
  .hsetnx(this.getAccountKey(), 'state', 'init')
307
- .sadd('ia:accounts', this.account)
308
- .exec();
1063
+ .hsetnx(this.getAccountKey(), 'runIndex', runIndex.toString())
1064
+ .hsetnx(this.getAccountKey(), `state:count:connected`, '0')
1065
+ .sadd(`${REDIS_PREFIX}ia:accounts`, this.account);
1066
+
1067
+ if (accountData.oauth2 && accountData.oauth2.provider) {
1068
+ pipeline = pipeline.sadd(`${REDIS_PREFIX}oapp:a:${accountData.oauth2.provider}`, this.account);
1069
+ if (accountData.oauth2.auth?.user) {
1070
+ // check if already exists
1071
+ let existingAppBinding = await this.redis.hget(
1072
+ `${REDIS_PREFIX}oapp:h:${accountData.oauth2.provider}`,
1073
+ accountData.oauth2.auth?.user?.toLowerCase()
1074
+ );
1075
+ if (existingAppBinding && existingAppBinding !== this.account) {
1076
+ let existingAccount;
1077
+ try {
1078
+ existingAccount = await this.loadAccountData(existingAppBinding);
1079
+ } catch (err) {
1080
+ // account not found
1081
+ }
1082
+
1083
+ if (existingAccount?.oauth2?.auth?.user === accountData.oauth2.auth?.user) {
1084
+ let message = 'Another account for the same OAuth2 user already exists';
1085
+ let error = Boom.boomify(new Error(message), { statusCode: 400 });
1086
+ error.output.payload.code = 'AccountAlreadyExists';
1087
+ error.output.payload.existingAccount = existingAppBinding;
1088
+ throw error;
1089
+ }
1090
+ }
1091
+
1092
+ pipeline = pipeline.hset(`${REDIS_PREFIX}oapp:h:${accountData.oauth2.provider}`, accountData.oauth2.auth?.user?.toLowerCase(), this.account);
1093
+ }
1094
+ }
1095
+
1096
+ let result = await pipeline.exec();
309
1097
 
310
1098
  if (!result || !result[1] || result[1][0] || result[1][1] !== 'OK') {
311
1099
  let message = 'Something went wrong';
@@ -317,38 +1105,75 @@ class Account {
317
1105
  if (result[0][1] && result[0][1].account) {
318
1106
  // existing user
319
1107
  state = 'existing';
320
- await this.call({ cmd: 'update', account: this.account });
1108
+
1109
+ try {
1110
+ let oldAccountData = this.unserializeAccountData(result[0][1]);
1111
+ if (oldAccountData?.oauth2?.provider && oldAccountData?.oauth2?.auth?.user && oldAccountData.oauth2.provider !== accountData.oauth2?.provider) {
1112
+ // remove previous entry if provider was updated
1113
+ await this.redis.hdel(`${REDIS_PREFIX}oapp:h:${oldAccountData.oauth2.provider}`, oldAccountData.oauth2.auth?.user?.toLowerCase());
1114
+ }
1115
+ } catch (err) {
1116
+ // ignore
1117
+ }
1118
+
1119
+ await this.call({
1120
+ cmd: 'update',
1121
+ account: this.account,
1122
+ timeout: this.timeout
1123
+ });
321
1124
  } else {
322
1125
  state = 'new';
323
- await this.call({ cmd: 'new', account: this.account });
1126
+ await this.call({
1127
+ cmd: 'new',
1128
+ account: this.account,
1129
+ timeout: this.timeout
1130
+ });
324
1131
  }
325
1132
 
326
1133
  return { account: this.account, state };
327
1134
  }
328
1135
 
329
1136
  async delete() {
330
- await this.loadAccountData(this.account);
1137
+ let accountData = await this.loadAccountData(this.account);
1138
+
1139
+ const dateKeyTdy = new Date().toISOString().substring(0, 10).replace(/-/g, '');
1140
+ const dateKeyYdy = new Date(Date.now() - 24 * 3600 * 1000).toISOString().substring(0, 10).replace(/-/g, '');
1141
+
1142
+ const tombstoneTdy = `${REDIS_PREFIX}tomb:${this.account}:${dateKeyTdy}`;
1143
+ const tombstoneYdy = `${REDIS_PREFIX}tomb:${this.account}:${dateKeyYdy}`;
331
1144
 
332
- let result = await this.redis
1145
+ let pipeline = this.redis
333
1146
  .multi()
334
1147
  .del(this.getAccountKey())
335
- .keys(`iam:${this.account}:*`)
336
- .srem('ia:accounts', this.account)
337
- .del(`ial:${this.account}`) // mailbox list
338
- .del(`iah:${this.account}`) // mailbox list for ID references
339
- .del(`iar:b:${this.account}`) // bounce list
340
- .del(`iaq:${this.account}`) // delayed message queue
341
- .exec();
342
-
343
- if (result && result[1] && result[1][1] && result[1][1].length) {
344
- // delete all mailbox specific keys
345
- let run = await this.redis.multi();
346
- for (let key of result[1][1]) {
347
- run = run.del(key);
1148
+ .srem(`${REDIS_PREFIX}ia:accounts`, this.account)
1149
+ .del(this.getMailboxListKey()) // mailbox list
1150
+ .del(this.getMailboxHashKey()) // mailbox list for ID references
1151
+ .del(`${REDIS_PREFIX}iar:b:${this.account}`) // bounce list
1152
+ .del(`${REDIS_PREFIX}iar:s:${this.account}`) // seen messages list
1153
+ .del(`${REDIS_PREFIX}iaq:${this.account}`) // delayed message queue
1154
+ .del(`${REDIS_PREFIX}iat:${this.account}`) // access tokens
1155
+ .del(`${REDIS_PREFIX}iac:${this.account}`) // cache hash
1156
+ .del(`${REDIS_PREFIX}iap:${this.account}`) // external event notification queue
1157
+
1158
+ .del(tombstoneTdy)
1159
+ .del(tombstoneYdy)
1160
+
1161
+ .del(`${REDIS_PREFIX}tpl:${this.account}:i`) // stored templates index
1162
+ .del(`${REDIS_PREFIX}tpl:${this.account}:c`); // stored templates index
1163
+
1164
+ if (accountData.oauth2?.provider) {
1165
+ pipeline = pipeline.srem(`${REDIS_PREFIX}oapp:a:${accountData.oauth2.provider}`, this.account);
1166
+ if (accountData.oauth2?.auth?.user) {
1167
+ pipeline = pipeline.hdel(`${REDIS_PREFIX}oapp:h:${accountData.oauth2.provider}`, accountData.oauth2.auth?.user?.toLowerCase());
348
1168
  }
349
- await run.exec();
350
1169
  }
351
1170
 
1171
+ let result = await pipeline.exec();
1172
+
1173
+ // scan and delete keys
1174
+ // should we wait though? might take a lot of time
1175
+ await redisScanDelete(this.redis, this.logger, `${REDIS_PREFIX}iam:${this.account}:*`);
1176
+
352
1177
  if (!result || !result[0] || !result[0][1]) {
353
1178
  return {
354
1179
  account: this.account,
@@ -356,7 +1181,35 @@ class Account {
356
1181
  };
357
1182
  }
358
1183
 
359
- await this.call({ cmd: 'delete', account: this.account });
1184
+ try {
1185
+ let queueKeep = (await settings.get('queueKeep')) || true;
1186
+ let serviceUrl = (await settings.get('serviceUrl')) || null;
1187
+
1188
+ let payload = {
1189
+ serviceUrl,
1190
+ account: this.account,
1191
+ date: new Date().toISOString(),
1192
+ event: ACCOUNT_DELETED_NOTIFY
1193
+ };
1194
+
1195
+ await this.documentsQueue.add(ACCOUNT_DELETED_NOTIFY, payload, {
1196
+ removeOnComplete: queueKeep,
1197
+ removeOnFail: queueKeep,
1198
+ attempts: 10,
1199
+ backoff: {
1200
+ type: 'exponential',
1201
+ delay: 5000
1202
+ }
1203
+ });
1204
+ } catch (err) {
1205
+ this.logger.error({ msg: 'Failed to add entry to documents queue', err });
1206
+ }
1207
+
1208
+ await this.call({
1209
+ cmd: 'delete',
1210
+ account: this.account,
1211
+ timeout: this.timeout
1212
+ });
360
1213
 
361
1214
  return {
362
1215
  account: this.account,
@@ -370,7 +1223,16 @@ class Account {
370
1223
  const { port1, port2 } = new MessageChannel();
371
1224
  const stream = new MessagePortReadable(port1);
372
1225
 
373
- let streamCreated = await this.call({ cmd: 'getRawMessage', account: this.account, message, port: port2 }, [port2]);
1226
+ let streamCreated = await this.call(
1227
+ {
1228
+ cmd: 'getRawMessage',
1229
+ account: this.account,
1230
+ message,
1231
+ timeout: this.timeout,
1232
+ port: port2
1233
+ },
1234
+ [port2]
1235
+ );
374
1236
 
375
1237
  if (streamCreated && streamCreated.headers) {
376
1238
  stream.headers = streamCreated.headers;
@@ -385,8 +1247,17 @@ class Account {
385
1247
  const { port1, port2 } = new MessageChannel();
386
1248
  const stream = new MessagePortReadable(port1);
387
1249
 
388
- let streamCreated = await this.call({ cmd: 'getAttachment', account: this.account, attachment, port: port2 }, [port2]);
389
-
1250
+ let streamCreated = await this.call(
1251
+ {
1252
+ cmd: 'getAttachment',
1253
+ account: this.account,
1254
+ attachment,
1255
+ timeout: this.timeout,
1256
+ port: port2
1257
+ },
1258
+ [port2]
1259
+ );
1260
+
390
1261
  if (streamCreated && streamCreated.headers) {
391
1262
  stream.headers = streamCreated.headers;
392
1263
  }
@@ -394,169 +1265,1522 @@ class Account {
394
1265
  return stream;
395
1266
  }
396
1267
 
397
- async getMailboxListing() {
398
- await this.loadAccountData(this.account, true);
1268
+ async getMailboxListing(query) {
1269
+ let accountData = await this.loadAccountData(this.account, false);
399
1270
 
400
- let result = [];
1271
+ let mailboxListing;
1272
+ if (await this.isApiClient(accountData)) {
1273
+ mailboxListing = await this.listMailboxes(query);
1274
+
1275
+ if (mailboxListing && mailboxListing.error) {
1276
+ let error = Boom.boomify(new Error(mailboxListing.error), { statusCode: mailboxListing.statusCode || 500 });
1277
+ if (mailboxListing.code) {
1278
+ error.output.payload.code = mailboxListing.code;
1279
+ }
1280
+ throw error;
1281
+ }
1282
+
1283
+ // just pass through, do nothing
1284
+ return mailboxListing;
1285
+ } else if (accountData.state === 'connected' || query.counters) {
1286
+ // run LIST
1287
+ mailboxListing = await this.listMailboxes(query);
1288
+ if (mailboxListing && mailboxListing.error) {
1289
+ let error = Boom.boomify(new Error(mailboxListing.error), { statusCode: mailboxListing.statusCode || 500 });
1290
+ if (mailboxListing.code) {
1291
+ error.output.payload.code = mailboxListing.code;
1292
+ }
1293
+ throw error;
1294
+ }
1295
+ } else if (accountData.state === 'unset') {
1296
+ // account has not been set up yet
1297
+ let error = Boom.boomify(new Error('Syncing is disabled for the requested account'), { statusCode: 503 });
1298
+ if (accountData.state) {
1299
+ error.output.payload.state = accountData.state;
1300
+ }
1301
+ error.output.payload.code = 'NotSyncing';
1302
+ throw error;
1303
+ } else if (accountData.state === 'init' || !(await this.redis.exists(this.getMailboxListKey()))) {
1304
+ // account has not been set up yet
1305
+ let error = Boom.boomify(new Error('Requested account is not yet initialized'), { statusCode: 503 });
1306
+ if (accountData.state) {
1307
+ error.output.payload.state = accountData.state;
1308
+ }
1309
+ error.output.payload.code = 'NotYetConnected';
1310
+ throw error;
1311
+ }
1312
+
1313
+ let mailboxes = [];
401
1314
  let storedListing = await this.redis.hgetallBuffer(this.getMailboxListKey());
402
1315
 
403
1316
  for (let path of Object.keys(storedListing || {})) {
404
1317
  try {
405
1318
  let decoded = msgpack.decode(storedListing[path]);
406
- if (decoded.delimiter && decoded.path.indexOf(decoded.delimiter) >= 0) {
1319
+
1320
+ if (decoded.path && decoded.delimiter && decoded.path.indexOf(decoded.delimiter) >= 0) {
407
1321
  decoded.parentPath = decoded.path.substr(0, decoded.path.lastIndexOf(decoded.delimiter));
408
1322
  }
409
- delete decoded.delimiter;
410
- result.push(Object.assign(decoded, await this.getMailboxInfo(path)));
1323
+
1324
+ let listedMailboxInfo;
1325
+ if (mailboxListing) {
1326
+ listedMailboxInfo = mailboxListing.find(entry => entry.path === path);
1327
+ if (listedMailboxInfo && listedMailboxInfo.status) {
1328
+ delete listedMailboxInfo.status.path;
1329
+ }
1330
+ }
1331
+
1332
+ mailboxes.push(
1333
+ Object.assign(
1334
+ decoded,
1335
+ await this.getMailboxInfo(path),
1336
+ listedMailboxInfo && listedMailboxInfo.status
1337
+ ? {
1338
+ status: listedMailboxInfo.status
1339
+ }
1340
+ : {}
1341
+ )
1342
+ );
411
1343
  } catch (err) {
412
1344
  // should not happen
413
- this.logger.error(err);
1345
+ if (logger.notifyError) {
1346
+ logger.notifyError(err, event => {
1347
+ if (this.account) {
1348
+ event.setUser(this.account);
1349
+ }
1350
+
1351
+ event.addMetadata('ee', {
1352
+ path,
1353
+ mailboxListing: typeof mailboxListing
1354
+ });
1355
+ });
1356
+ }
1357
+
1358
+ let message = 'Failed to process stored mailbox listing';
1359
+ this.logger.error({ msg: message, path, mailboxListing: typeof mailboxListing, account: this.account, err });
1360
+ let error = Boom.boomify(new Error(message), { statusCode: 503 });
1361
+ error.output.payload.code = err.code;
1362
+ throw error;
414
1363
  }
415
1364
  }
416
1365
 
417
- return result;
1366
+ return mailboxes;
418
1367
  }
419
1368
 
420
1369
  async updateMessage(message, updates) {
421
1370
  await this.loadAccountData(this.account, true);
422
1371
 
423
- return await this.call({ cmd: 'updateMessage', account: this.account, message, updates });
1372
+ return await this.call({
1373
+ cmd: 'updateMessage',
1374
+ account: this.account,
1375
+ message,
1376
+ updates,
1377
+ timeout: this.timeout
1378
+ });
424
1379
  }
425
1380
 
426
- async moveMessage(message, target) {
1381
+ async updateMessages(path, search, updates) {
427
1382
  await this.loadAccountData(this.account, true);
428
- return await this.call({ cmd: 'moveMessage', account: this.account, message, target });
1383
+
1384
+ return await this.call({
1385
+ cmd: 'updateMessages',
1386
+ account: this.account,
1387
+ path,
1388
+ search,
1389
+ updates,
1390
+ timeout: this.timeout
1391
+ });
1392
+ }
1393
+
1394
+ async listMailboxes(query) {
1395
+ let options = {};
1396
+ if (query && query.counters) {
1397
+ options.statusQuery = {
1398
+ messages: true,
1399
+ unseen: true
1400
+ };
1401
+ }
1402
+ let x = await this.call({
1403
+ cmd: 'listMailboxes',
1404
+ account: this.account,
1405
+ options,
1406
+ timeout: this.timeout
1407
+ });
1408
+
1409
+ return x;
1410
+ }
1411
+
1412
+ async moveMessage(message, target, options) {
1413
+ await this.loadAccountData(this.account, true);
1414
+ return await this.call({
1415
+ cmd: 'moveMessage',
1416
+ account: this.account,
1417
+ message,
1418
+ target,
1419
+ options,
1420
+ timeout: this.timeout
1421
+ });
1422
+ }
1423
+
1424
+ async moveMessages(source, search, target) {
1425
+ await this.loadAccountData(this.account, true);
1426
+ return await this.call({
1427
+ cmd: 'moveMessages',
1428
+ account: this.account,
1429
+ source,
1430
+ search,
1431
+ target,
1432
+ timeout: this.timeout
1433
+ });
429
1434
  }
430
1435
 
431
1436
  async deleteMessage(message, force) {
432
1437
  await this.loadAccountData(this.account, true);
433
1438
 
434
- return await this.call({ cmd: 'deleteMessage', account: this.account, message, force });
1439
+ return await this.call({
1440
+ cmd: 'deleteMessage',
1441
+ account: this.account,
1442
+ message,
1443
+ force,
1444
+ timeout: this.timeout
1445
+ });
1446
+ }
1447
+
1448
+ async deleteMessages(path, search, force) {
1449
+ await this.loadAccountData(this.account, true);
1450
+
1451
+ return await this.call({
1452
+ cmd: 'deleteMessages',
1453
+ account: this.account,
1454
+ path,
1455
+ search,
1456
+ force,
1457
+ timeout: this.timeout
1458
+ });
1459
+ }
1460
+
1461
+ async getQuota() {
1462
+ await this.loadAccountData(this.account, true);
1463
+
1464
+ return await this.call({
1465
+ cmd: 'getQuota',
1466
+ account: this.account,
1467
+ timeout: this.timeout
1468
+ });
435
1469
  }
436
1470
 
437
1471
  async createMailbox(path) {
438
1472
  await this.loadAccountData(this.account, true);
439
1473
 
440
- return await this.call({ cmd: 'createMailbox', account: this.account, path });
1474
+ return await this.call({
1475
+ cmd: 'createMailbox',
1476
+ account: this.account,
1477
+ path,
1478
+ timeout: this.timeout
1479
+ });
1480
+ }
1481
+
1482
+ async modifyMailbox(path, newPath, subscribed) {
1483
+ await this.loadAccountData(this.account, true);
1484
+
1485
+ return await this.call({
1486
+ cmd: 'modifyMailbox',
1487
+ account: this.account,
1488
+ path,
1489
+ newPath,
1490
+ subscribed,
1491
+ timeout: this.timeout
1492
+ });
441
1493
  }
442
1494
 
443
1495
  async deleteMailbox(path) {
444
1496
  await this.loadAccountData(this.account, true);
445
1497
 
446
- return await this.call({ cmd: 'deleteMailbox', account: this.account, path });
1498
+ return await this.call({
1499
+ cmd: 'deleteMailbox',
1500
+ account: this.account,
1501
+ path,
1502
+ timeout: this.timeout
1503
+ });
447
1504
  }
448
1505
 
449
1506
  async getText(text, options) {
1507
+ if (options.documentStore && (await settings.get('documentStoreEnabled'))) {
1508
+ await this.loadAccountData(this.account, false);
1509
+
1510
+ const { index, client } = this.esClient;
1511
+
1512
+ let buf = Buffer.from(text, 'base64url');
1513
+ let message = buf.subarray(0, 8).toString('base64url');
1514
+
1515
+ let getResult = await client.get({
1516
+ index,
1517
+ id: `${this.account}:${message}`
1518
+ });
1519
+
1520
+ this.logger.trace({
1521
+ msg: 'Executed ES query',
1522
+ query: { type: 'get', index, id: `${this.account}:${message}` },
1523
+ results: getResult && getResult._source ? 1 : 0
1524
+ });
1525
+
1526
+ if (!getResult || !getResult._source) {
1527
+ let message = 'Requested message was not found';
1528
+ let error = Boom.boomify(new Error(message), { statusCode: 404 });
1529
+ throw error;
1530
+ }
1531
+
1532
+ let messageData = getResult._source;
1533
+ let response = {};
1534
+
1535
+ response.hasMore = false;
1536
+ for (let textType of Object.keys(messageData.text || {})) {
1537
+ if (['plain', 'html'].includes(textType) && (options.textType === '*' || options.textType === textType)) {
1538
+ if (options.maxBytes && messageData.text[textType].length > options.maxBytes) {
1539
+ response[textType] = messageData.text[textType].substring(0, options.maxBytes);
1540
+ response.hasMore = true;
1541
+ } else {
1542
+ response[textType] = messageData.text[textType];
1543
+ }
1544
+ }
1545
+ }
1546
+
1547
+ return response;
1548
+ }
1549
+
450
1550
  await this.loadAccountData(this.account, true);
451
1551
 
452
- return await this.call({ cmd: 'getText', account: this.account, text, options });
1552
+ return await this.call({
1553
+ cmd: 'getText',
1554
+ account: this.account,
1555
+ text,
1556
+ options,
1557
+ timeout: this.timeout
1558
+ });
453
1559
  }
454
1560
 
455
1561
  async getMessage(message, options) {
1562
+ if (options.webSafeHtml) {
1563
+ options.textType = '*';
1564
+ options.embedAttachedImages = true;
1565
+ options.preProcessHtml = true;
1566
+ }
1567
+
1568
+ if (options.documentStore && (await settings.get('documentStoreEnabled'))) {
1569
+ await this.loadAccountData(this.account, false);
1570
+
1571
+ const { index, client } = this.esClient;
1572
+
1573
+ const reqOpts = {
1574
+ index,
1575
+ id: `${this.account}:${message}`,
1576
+ _source_excludes: 'preview,seemsLikeNew,account,created,updateTime'
1577
+ };
1578
+
1579
+ switch (options.textType) {
1580
+ case '*':
1581
+ break;
1582
+ case 'html':
1583
+ reqOpts._source_excludes += ',text.plain';
1584
+ break;
1585
+ case 'plain':
1586
+ reqOpts._source_excludes += ',text.html,text._generatedHtml';
1587
+ break;
1588
+ default:
1589
+ reqOpts._source_excludes += ',text.plain,text.html,text._generatedHtml';
1590
+ }
1591
+
1592
+ let getResult = await client.get(reqOpts);
1593
+
1594
+ this.logger.trace({
1595
+ msg: 'Executed ES query',
1596
+ query: { type: 'get', index, id: `${this.account}:${message}` },
1597
+ results: getResult && getResult._source ? 1 : 0
1598
+ });
1599
+
1600
+ if (!getResult || !getResult._source) {
1601
+ let message = 'Requested message was not found';
1602
+ let error = Boom.boomify(new Error(message), { statusCode: 404 });
1603
+ throw error;
1604
+ }
1605
+
1606
+ let messageData = getResult._source;
1607
+
1608
+ // restore headers and text object as per the API response
1609
+ let headersObj = {};
1610
+ for (let { key, value } of messageData.headers) {
1611
+ headersObj[key] = value;
1612
+ }
1613
+ messageData.headers = headersObj;
1614
+
1615
+ if (messageData.text && (messageData.text.html || messageData.text.plain)) {
1616
+ messageData.text.hasMore = false;
1617
+
1618
+ for (let textType of Object.keys(messageData.text || {})) {
1619
+ if (['plain', 'html'].includes(textType) && options.maxBytes && messageData.text[textType].length > options.maxBytes) {
1620
+ messageData.text[textType] = messageData.text[textType].substring(0, options.maxBytes);
1621
+ messageData.text.hasMore = true;
1622
+ }
1623
+ }
1624
+ }
1625
+
1626
+ for (let key of ['unseen', 'flagged', 'answered', 'draft']) {
1627
+ if (messageData[key] === false) {
1628
+ delete messageData[key];
1629
+ }
1630
+ }
1631
+
1632
+ if (options.preProcessHtml && messageData.text && (messageData.text.html || messageData.text.plain)) {
1633
+ // If available, use the cached version
1634
+ messageData.text.html =
1635
+ messageData.text._generatedHtml ||
1636
+ mimeHtml({
1637
+ html: messageData.text.html,
1638
+ text: messageData.text.plain
1639
+ });
1640
+ messageData.text.webSafe = true;
1641
+ messageData.text._cachedWebSafe = !!messageData.text._generatedHtml;
1642
+ }
1643
+
1644
+ if (options.embedAttachedImages && messageData.text && messageData.text.html && messageData.attachments && messageData.attachments.length) {
1645
+ let attachmentsToDownload = [];
1646
+
1647
+ // first pass, find attachments to inline
1648
+ messageData.text.html.replace(/\bcid:([^"'\s>]+)/g, (fullMatch, cidMatch) => {
1649
+ let attachment = messageData.attachments.find(attachment => [`<${cidMatch}>`, cidMatch].includes(attachment.contentId));
1650
+ if (attachment && !attachment.content) {
1651
+ attachmentsToDownload.push(attachment);
1652
+ }
1653
+ });
1654
+
1655
+ // download large inline attachments not stored in ES
1656
+ for (let attachment of attachmentsToDownload) {
1657
+ try {
1658
+ let downloadStream = await this.getAttachment(attachment.id);
1659
+ if (downloadStream) {
1660
+ let content = await download(downloadStream);
1661
+ this.logger.trace({ msg: 'Fetched attachment content', account: this.account, attachment, size: content.length });
1662
+ attachment.content = content.toString('base64');
1663
+ }
1664
+ } catch (err) {
1665
+ this.logger.error({ msg: 'Failed to fetch attachment content', account: this.account, attachment, err });
1666
+ }
1667
+ }
1668
+
1669
+ // second pass, replace placeholders with inline attachments
1670
+ messageData.text.html = messageData.text.html.replace(/\bcid:([^"'\s>]+)/g, (fullMatch, cidMatch) => {
1671
+ let attachment = messageData.attachments.find(attachment => [`<${cidMatch}>`, cidMatch].includes(attachment.contentId));
1672
+ if (attachment && attachment.content) {
1673
+ return `data:${attachment.contentType || 'application/octet-stream'};base64,${attachment.content}`;
1674
+ }
1675
+ return fullMatch;
1676
+ });
1677
+ }
1678
+
1679
+ if (messageData.text && messageData.text._generatedHtml) {
1680
+ // remove cached pre-processed HTML from output
1681
+ delete messageData.text._generatedHtml;
1682
+ }
1683
+
1684
+ // Add event file content if the attachment exists
1685
+ if (messageData.calendarEvents) {
1686
+ for (let calendarEvent of messageData.calendarEvents) {
1687
+ if (!calendarEvent.content && calendarEvent.attachment) {
1688
+ let attachment = messageData.attachments && messageData.attachments.find(attachment => attachment.id === calendarEvent.attachment);
1689
+ if (attachment && attachment.content) {
1690
+ calendarEvent.encoding = 'base64';
1691
+ calendarEvent.content = attachment.content;
1692
+ }
1693
+ }
1694
+ }
1695
+ }
1696
+
1697
+ if (messageData.attachments) {
1698
+ for (let attachment of messageData.attachments) {
1699
+ delete attachment.content;
1700
+ }
1701
+ }
1702
+
1703
+ if (options.markAsSeen && (!messageData.flags || !messageData.flags.includes('\\Seen'))) {
1704
+ // mark message as seen
1705
+ messageData.flags.push('\\Seen');
1706
+ // do not wait until the update is completed, return immediately
1707
+ this.updateMessage(message, { flags: { add: ['\\Seen'] } }).catch(err => {
1708
+ this.logger.error({ msg: 'Failed to mark message as Seen', message, err });
1709
+ });
1710
+ }
1711
+
1712
+ if (messageData.specialUse && !messageData.messageSpecialUse) {
1713
+ for (let specialUseTag of ['\\Junk', '\\Sent', '\\Trash', '\\Inbox', '\\Drafts']) {
1714
+ if (messageData.specialUse === specialUseTag || (messageData.labels && messageData.labels.includes(specialUseTag))) {
1715
+ messageData.messageSpecialUse = specialUseTag;
1716
+ break;
1717
+ }
1718
+ }
1719
+ }
1720
+
1721
+ return messageData;
1722
+ }
1723
+
456
1724
  await this.loadAccountData(this.account, true);
457
1725
 
458
- let messageData = await this.call({ cmd: 'getMessage', account: this.account, message, options });
1726
+ let messageData = await this.call({
1727
+ cmd: 'getMessage',
1728
+ account: this.account,
1729
+ message,
1730
+ options,
1731
+ timeout: this.timeout
1732
+ });
1733
+
459
1734
  if (!messageData) {
460
1735
  let message = 'Requested message was not found';
461
1736
  let error = Boom.boomify(new Error(message), { statusCode: 404 });
462
1737
  throw error;
463
1738
  }
1739
+
464
1740
  return messageData;
465
1741
  }
466
1742
 
467
1743
  async listMessages(query) {
468
- await this.loadAccountData(this.account, true);
1744
+ if (query.documentStore && (await settings.get('documentStoreEnabled'))) {
1745
+ await this.loadAccountData(this.account, false);
1746
+
1747
+ const { index, client } = this.esClient;
1748
+
1749
+ let inboxData = await this.redis.hgetBuffer(this.getMailboxListKey(), 'INBOX');
1750
+ let delimiter;
1751
+ if (inboxData) {
1752
+ try {
1753
+ inboxData = msgpack.decode(inboxData);
1754
+ delimiter = inboxData.delimiter;
1755
+ } catch (err) {
1756
+ delimiter = '/'; // hope for the best
1757
+ inboxData = false;
1758
+ }
1759
+ }
469
1760
 
470
- let path = normalizePath(query.path);
471
- let encodedMailboxData = await this.redis.hgetBuffer(this.getMailboxListKey(), path);
472
- if (!encodedMailboxData) {
473
- let message = 'Mailbox record was not found';
474
- let error = Boom.boomify(new Error(message), { statusCode: 404 });
475
- error.output.payload.path = query.path;
476
- throw error;
1761
+ inboxData = inboxData || {
1762
+ path: 'INBOX',
1763
+ delimiter
1764
+ };
1765
+
1766
+ inboxData.specialUse = inboxData.specialUse || '\\Inbox';
1767
+
1768
+ let path = normalizePath(query.path, delimiter);
1769
+ let mailboxData = path === 'INBOX' ? inboxData : false;
1770
+ if (!mailboxData) {
1771
+ mailboxData = await this.redis.hgetBuffer(this.getMailboxListKey(), path);
1772
+ if (mailboxData) {
1773
+ try {
1774
+ mailboxData = msgpack.decode(mailboxData);
1775
+ } catch (err) {
1776
+ mailboxData = false;
1777
+ }
1778
+ }
1779
+ }
1780
+
1781
+ let searchQuery = {
1782
+ bool: {
1783
+ must: [
1784
+ {
1785
+ term: {
1786
+ account: this.account
1787
+ }
1788
+ }
1789
+ ]
1790
+ }
1791
+ };
1792
+
1793
+ searchQuery.bool.must.push({
1794
+ bool: {
1795
+ should: [
1796
+ {
1797
+ term: {
1798
+ path
1799
+ }
1800
+ },
1801
+ {
1802
+ term: {
1803
+ labels: (mailboxData && mailboxData.specialUse) || path
1804
+ }
1805
+ }
1806
+ ],
1807
+ minimum_should_match: 1
1808
+ }
1809
+ });
1810
+
1811
+ let page = Number(query.page) || 0;
1812
+ let pageSize = Math.abs(Number(query.pageSize) || 20);
1813
+
1814
+ if (page < 0) {
1815
+ page = 0;
1816
+ }
1817
+
1818
+ let searchResult = await client.search({
1819
+ index,
1820
+ size: pageSize,
1821
+ from: pageSize * page,
1822
+ query: searchQuery,
1823
+ sort: { uid: 'desc' },
1824
+ _source_excludes: 'headers,text.plain,text.html,text._generatedHtml,seemsLikeNew,attachments.content,summary,riskAssessment,updateTime'
1825
+ });
1826
+
1827
+ this.logger.trace({
1828
+ msg: 'Executed ES query',
1829
+ query: { type: 'search', index, size: pageSize, from: pageSize * page, query: searchQuery, sort: { uid: 'desc' } },
1830
+ results: searchResult.hits.total.value
1831
+ });
1832
+
1833
+ let response = {
1834
+ total: searchResult.hits.total.value,
1835
+ page,
1836
+ pages: Math.max(Math.ceil(searchResult.hits.total.value / pageSize), 1),
1837
+ messages: searchResult.hits.hits.map(entry => {
1838
+ let messageData = entry._source;
1839
+
1840
+ // normalize as per the API response
1841
+
1842
+ for (let key of ['unseen', 'flagged', 'answered', 'draft']) {
1843
+ if (messageData[key] === false) {
1844
+ messageData[key] = undefined;
1845
+ }
1846
+ }
1847
+
1848
+ for (let key of ['account', 'created', 'specialUse']) {
1849
+ if (messageData[key]) {
1850
+ messageData[key] = undefined;
1851
+ }
1852
+ }
1853
+
1854
+ return messageData;
1855
+ })
1856
+ };
1857
+
1858
+ return response;
477
1859
  }
478
1860
 
479
- // mailbox seems to exist, so call parent to resolve open connection
480
- return await this.call(Object.assign({ cmd: 'listMessages', account: this.account }, query));
1861
+ await this.loadAccountData(this.account, true);
1862
+
1863
+ return await this.call(
1864
+ Object.assign(
1865
+ {
1866
+ cmd: 'listMessages',
1867
+ account: this.account
1868
+ },
1869
+ query,
1870
+ { timeout: this.timeout }
1871
+ )
1872
+ );
481
1873
  }
482
1874
 
483
- async buildContacts() {
1875
+ async searchMessages(query, searchOpts) {
1876
+ searchOpts = searchOpts || {};
1877
+ if (query.documentStore && (await settings.get('documentStoreEnabled'))) {
1878
+ if (!searchOpts.unified) {
1879
+ await this.loadAccountData(this.account, false);
1880
+ }
1881
+
1882
+ const { index, client } = this.esClient;
1883
+
1884
+ let searchQuery = {
1885
+ bool: {
1886
+ must: []
1887
+ }
1888
+ };
1889
+
1890
+ if (this.account) {
1891
+ searchQuery.bool.must.push({
1892
+ term: {
1893
+ account: this.account
1894
+ }
1895
+ });
1896
+ }
1897
+
1898
+ if (searchOpts.unified && query.accounts && query.accounts.length) {
1899
+ searchQuery.bool.must.push({
1900
+ bool: {
1901
+ should: query.accounts.map(account => ({
1902
+ term: {
1903
+ account
1904
+ }
1905
+ })),
1906
+ minimum_should_match: 1
1907
+ }
1908
+ });
1909
+ }
1910
+
1911
+ if (query.path) {
1912
+ let inboxData = await this.redis.hgetBuffer(this.getMailboxListKey(), 'INBOX');
1913
+ let delimiter;
1914
+
1915
+ if (inboxData) {
1916
+ try {
1917
+ inboxData = msgpack.decode(inboxData);
1918
+ delimiter = inboxData.delimiter;
1919
+ } catch (err) {
1920
+ delimiter = '/'; // hope for the best
1921
+ inboxData = false;
1922
+ }
1923
+ }
1924
+
1925
+ inboxData = inboxData || {
1926
+ path: 'INBOX',
1927
+ delimiter
1928
+ };
1929
+
1930
+ inboxData.specialUse = inboxData.specialUse || '\\Inbox';
1931
+
1932
+ let path = normalizePath(query.path, delimiter);
1933
+ let mailboxData = path === 'INBOX' ? inboxData : false;
1934
+ if (!mailboxData) {
1935
+ mailboxData = await this.redis.hgetBuffer(this.getMailboxListKey(), path);
1936
+ if (mailboxData) {
1937
+ try {
1938
+ mailboxData = msgpack.decode(mailboxData);
1939
+ } catch (err) {
1940
+ mailboxData = false;
1941
+ }
1942
+ }
1943
+ }
1944
+
1945
+ searchQuery.bool.must.push({
1946
+ bool: {
1947
+ should: [
1948
+ {
1949
+ term: {
1950
+ path
1951
+ }
1952
+ },
1953
+ {
1954
+ term: {
1955
+ labels: (mailboxData && mailboxData.specialUse) || path
1956
+ }
1957
+ }
1958
+ ],
1959
+ minimum_should_match: 1
1960
+ }
1961
+ });
1962
+ }
1963
+
1964
+ if (searchOpts.unified && query.paths && query.paths.length) {
1965
+ searchQuery.bool.must.push({
1966
+ bool: {
1967
+ should: query.paths.flatMap(path => {
1968
+ let res = [
1969
+ {
1970
+ term: {
1971
+ path
1972
+ }
1973
+ },
1974
+ {
1975
+ term: {
1976
+ labels: path
1977
+ }
1978
+ },
1979
+ {
1980
+ term: {
1981
+ messageSpecialUse: path
1982
+ }
1983
+ }
1984
+ ];
1985
+
1986
+ if (/^inbox$/i.test(path)) {
1987
+ res.push({
1988
+ term: {
1989
+ messageSpecialUse: '\\Inbox'
1990
+ }
1991
+ });
1992
+ }
1993
+
1994
+ return res;
1995
+ }),
1996
+ minimum_should_match: 1
1997
+ }
1998
+ });
1999
+ }
2000
+
2001
+ for (let key of ['answered', 'deleted', 'draft', 'unseen', 'flagged']) {
2002
+ if (typeof query.search[key] === 'boolean') {
2003
+ searchQuery.bool.must.push({
2004
+ term: {
2005
+ [key]: query.search[key]
2006
+ }
2007
+ });
2008
+ }
2009
+ }
2010
+
2011
+ if (typeof query.search.seen === 'boolean') {
2012
+ searchQuery.bool.must.push({
2013
+ term: {
2014
+ unseen: !query.search.seen
2015
+ }
2016
+ });
2017
+ }
2018
+
2019
+ for (let key of ['from', 'to', 'cc', 'bcc']) {
2020
+ if (query.search[key]) {
2021
+ searchQuery.bool.must.push({
2022
+ bool: {
2023
+ should: [
2024
+ {
2025
+ match: {
2026
+ [`${key}.name`]: {
2027
+ query: query.search[key],
2028
+ operator: 'and'
2029
+ }
2030
+ }
2031
+ },
2032
+ {
2033
+ term: {
2034
+ [`${key}.address`]: query.search[key]
2035
+ }
2036
+ }
2037
+ ],
2038
+ minimum_should_match: 1
2039
+ }
2040
+ });
2041
+ }
2042
+ }
2043
+
2044
+ if (query.search.uid) {
2045
+ let uidEntries = unpackUIDRangeForSearch(query.search.uid);
2046
+ if (uidEntries && uidEntries.length) {
2047
+ let mustList = [];
2048
+ for (let entry of uidEntries) {
2049
+ if (typeof entry === 'number') {
2050
+ mustList.push({
2051
+ match: {
2052
+ uid: {
2053
+ query: entry,
2054
+ operator: 'and'
2055
+ }
2056
+ }
2057
+ });
2058
+ } else if (typeof entry === 'object') {
2059
+ mustList.push({
2060
+ range: {
2061
+ uid: entry
2062
+ }
2063
+ });
2064
+ }
2065
+ }
2066
+
2067
+ if (mustList.length) {
2068
+ searchQuery.bool.must.push({
2069
+ bool: {
2070
+ should: mustList,
2071
+ minimum_should_match: 1
2072
+ }
2073
+ });
2074
+ }
2075
+ }
2076
+ }
2077
+
2078
+ for (let key of ['emailId', 'threadId']) {
2079
+ if (query.search[key]) {
2080
+ searchQuery.bool.must.push({
2081
+ term: {
2082
+ [key]: query.search[key]
2083
+ }
2084
+ });
2085
+ }
2086
+ }
2087
+
2088
+ if (query.search.subject) {
2089
+ searchQuery.bool.must.push({
2090
+ match: {
2091
+ subject: {
2092
+ query: query.search.subject,
2093
+ operator: 'and'
2094
+ }
2095
+ }
2096
+ });
2097
+ }
2098
+
2099
+ if (query.search.body) {
2100
+ searchQuery.bool.must.push({
2101
+ bool: {
2102
+ should: [
2103
+ {
2104
+ match: {
2105
+ 'text.plain': {
2106
+ query: query.search.body,
2107
+ operator: 'and'
2108
+ }
2109
+ }
2110
+ },
2111
+ {
2112
+ match: {
2113
+ 'text.html': {
2114
+ query: query.search.body,
2115
+ operator: 'and'
2116
+ }
2117
+ }
2118
+ }
2119
+ ],
2120
+ minimum_should_match: 1
2121
+ }
2122
+ });
2123
+ }
2124
+
2125
+ let dateMatch = {};
2126
+
2127
+ for (let key of ['before', 'sentBefore']) {
2128
+ if (query.search[key]) {
2129
+ dateMatch.lte = query.search[key];
2130
+ }
2131
+ }
2132
+
2133
+ for (let key of ['since', 'sentSince']) {
2134
+ if (query.search[key]) {
2135
+ dateMatch.gte = query.search[key];
2136
+ }
2137
+ }
2138
+
2139
+ if (Object.keys(dateMatch).length) {
2140
+ searchQuery.bool.must.push({
2141
+ range: { date: dateMatch }
2142
+ });
2143
+ }
2144
+
2145
+ let sizeMatch = {};
2146
+
2147
+ if (query.search.larger) {
2148
+ dateMatch.gte = query.search.larger;
2149
+ }
2150
+
2151
+ if (query.search.smaller) {
2152
+ dateMatch.lte = query.search.smaller;
2153
+ }
2154
+
2155
+ if (Object.keys(sizeMatch).length) {
2156
+ searchQuery.bool.must.push({
2157
+ range: { size: sizeMatch }
2158
+ });
2159
+ }
2160
+
2161
+ // headers, nested query
2162
+ if (Object.keys(query.search.header || {}).length) {
2163
+ Object.keys(query.search.header).forEach(header => {
2164
+ searchQuery.bool.must.push({
2165
+ nested: {
2166
+ path: 'headers',
2167
+ query: {
2168
+ bool: {
2169
+ must: [
2170
+ {
2171
+ term: {
2172
+ 'headers.key': header.toLowerCase()
2173
+ }
2174
+ },
2175
+ {
2176
+ match: {
2177
+ 'headers.value': {
2178
+ query: (query.search.header[header] || '').toString(),
2179
+ operator: 'and'
2180
+ }
2181
+ }
2182
+ }
2183
+ ]
2184
+ }
2185
+ }
2186
+ }
2187
+ });
2188
+ });
2189
+ }
2190
+
2191
+ if (query.documentQuery) {
2192
+ searchQuery.bool.must.push(query.documentQuery);
2193
+ }
2194
+
2195
+ let page = Number(query.page) || 0;
2196
+ let pageSize = Math.abs(Number(query.pageSize) || 20);
2197
+
2198
+ if (page < 0) {
2199
+ page = 0;
2200
+ }
2201
+
2202
+ let searchResult = await client.search({
2203
+ index,
2204
+ size: pageSize,
2205
+ from: pageSize * page,
2206
+ query: searchQuery,
2207
+ sort: { [!searchOpts.unified ? 'uid' : 'date']: 'desc' },
2208
+ _source_excludes: 'headers,text.plain,text.html,text._generatedHtml,seemsLikeNew,attachments.content,summary,riskAssessment,updateTime'
2209
+ });
2210
+
2211
+ this.logger.trace({
2212
+ msg: 'Executed ES query',
2213
+ query: { type: 'search', index, size: pageSize, from: pageSize * page, query: searchQuery, sort: { uid: 'desc' } },
2214
+ results: searchResult.hits.total.value
2215
+ });
2216
+
2217
+ let response = {
2218
+ total: searchResult.hits.total.value,
2219
+ page,
2220
+ pages: Math.max(Math.ceil(searchResult.hits.total.value / pageSize), 1)
2221
+ };
2222
+
2223
+ if (query.exposeQuery) {
2224
+ response.documentStoreQuery = query;
2225
+ }
2226
+
2227
+ if (query.accounts) {
2228
+ response.accounts = query.accounts;
2229
+ }
2230
+
2231
+ if (query.paths) {
2232
+ response.paths = query.paths;
2233
+ }
2234
+
2235
+ response.messages = searchResult.hits.hits.map(entry => {
2236
+ let messageData = entry._source;
2237
+
2238
+ // normalize as per the API response
2239
+
2240
+ for (let key of ['unseen', 'flagged', 'answered', 'draft']) {
2241
+ if (messageData[key] === false) {
2242
+ messageData[key] = undefined;
2243
+ }
2244
+ }
2245
+
2246
+ for (let key of ['created', 'specialUse'].concat(!searchOpts.unified ? 'account' : [])) {
2247
+ if (messageData[key]) {
2248
+ messageData[key] = undefined;
2249
+ }
2250
+ }
2251
+
2252
+ return messageData;
2253
+ });
2254
+
2255
+ return response;
2256
+ }
2257
+
484
2258
  await this.loadAccountData(this.account, true);
485
- return await this.call({ cmd: 'buildContacts', account: this.account, timeout: 5 * 60 * 1000 });
2259
+
2260
+ return await this.call(
2261
+ Object.assign(
2262
+ {
2263
+ cmd: 'listMessages',
2264
+ account: this.account
2265
+ },
2266
+ query,
2267
+ { timeout: this.timeout }
2268
+ )
2269
+ );
486
2270
  }
487
2271
 
488
2272
  async uploadMessage(data) {
489
2273
  await this.loadAccountData(this.account, true);
490
2274
 
491
- let messageData = await this.call({ cmd: 'uploadMessage', account: this.account, data });
2275
+ let messageData = await this.call({
2276
+ cmd: 'uploadMessage',
2277
+ account: this.account,
2278
+ data,
2279
+ timeout: this.timeout
2280
+ });
492
2281
  return messageData;
493
2282
  }
494
2283
 
495
2284
  async submitMessage(data) {
496
- await this.loadAccountData(this.account, true);
2285
+ await this.loadAccountData(this.account, false);
497
2286
 
498
2287
  let messageData = await this.call(
499
2288
  {
500
2289
  cmd: 'submitMessage',
501
2290
  account: this.account,
502
- data
2291
+ data,
2292
+ // extended wait period when sending emails
2293
+ timeout: Math.max(this.timeout, 10 * 60 * 1000)
503
2294
  }
504
2295
  //typeof data.raw === 'object' ? [data.raw] : []
505
2296
  );
2297
+
506
2298
  return messageData;
507
2299
  }
508
2300
 
509
- async queueMessage(data) {
510
- await this.loadAccountData(this.account, true);
2301
+ async queueMessage(data, meta) {
2302
+ await this.loadAccountData(this.account, false);
511
2303
 
512
2304
  let messageData = await this.call(
513
2305
  {
514
2306
  cmd: 'queueMessage',
515
2307
  account: this.account,
516
- data
2308
+ data,
2309
+ meta,
2310
+ timeout: this.timeout
517
2311
  }
518
2312
  //typeof data.raw === 'object' ? [data.raw] : []
519
2313
  );
520
2314
  return messageData;
521
2315
  }
522
2316
 
523
- async requestReconnect(data) {
2317
+ async listSignatures(query) {
524
2318
  await this.loadAccountData(this.account, true);
525
2319
 
2320
+ return await this.call(
2321
+ Object.assign(
2322
+ {
2323
+ cmd: 'listSignatures',
2324
+ account: this.account
2325
+ },
2326
+ query,
2327
+ { timeout: this.timeout }
2328
+ )
2329
+ );
2330
+ }
2331
+
2332
+ async requestReconnect(data) {
2333
+ await this.loadAccountData(this.account, false);
2334
+
526
2335
  if (data.reconnect) {
527
- await this.call({ cmd: 'update', account: this.account });
2336
+ await this.call({
2337
+ cmd: 'reconnect',
2338
+ account: this.account,
2339
+ timeout: this.timeout
2340
+ });
2341
+ return true;
2342
+ }
2343
+ return false;
2344
+ }
2345
+
2346
+ async requestSync(data) {
2347
+ await this.loadAccountData(this.account, true);
2348
+
2349
+ if (data.sync) {
2350
+ await this.call({
2351
+ cmd: 'sync',
2352
+ account: this.account,
2353
+ timeout: this.timeout
2354
+ });
528
2355
  return true;
529
2356
  }
530
2357
  return false;
531
2358
  }
532
2359
 
533
- async renewAccessToken(oauthKeys) {
2360
+ async flush(data) {
2361
+ await this.loadAccountData(this.account);
2362
+
2363
+ if (!data.flush) {
2364
+ return false;
2365
+ }
2366
+
2367
+ // use a global lock to decrease Redis scanning operations
2368
+ let lockKey = ['flush' /*, this.account*/].join(':');
2369
+
2370
+ let lock = this.getLock();
2371
+ let flushLock;
2372
+
2373
+ try {
2374
+ flushLock = await lock.acquireLock(lockKey, 30 * 60 * 1000);
2375
+ if (!flushLock.success) {
2376
+ this.logger.error({ msg: 'Failed to get lock', lockKey });
2377
+
2378
+ let error = Boom.boomify(new Error('One flush operation at a time allowed, try again later'), { statusCode: 429 });
2379
+ error.output.payload.code = 'LockFail';
2380
+ throw error;
2381
+ }
2382
+ } catch (err) {
2383
+ this.logger.error({ msg: 'Failed to get lock', lockKey, err });
2384
+ if (Boom.isBoom) {
2385
+ throw err;
2386
+ }
2387
+ let error = Boom.boomify(new Error('Failed to get flush lock, try again later'), { statusCode: 500 });
2388
+ if (err.code) {
2389
+ error.output.payload.code = err.code || 'LockFail';
2390
+ }
2391
+ throw error;
2392
+ }
2393
+
2394
+ try {
2395
+ await this.call({
2396
+ cmd: 'pause',
2397
+ account: this.account,
2398
+ timeout: this.timeout
2399
+ });
2400
+
2401
+ let notifyFrom = data.notifyFrom && data.notifyFrom !== 'now' ? data.notifyFrom : new Date();
2402
+ let imapIndexer = data.imapIndexer;
2403
+
2404
+ const dateKeyTdy = new Date().toISOString().substring(0, 10).replace(/-/g, '');
2405
+ const dateKeyYdy = new Date(Date.now() - 24 * 3600 * 1000).toISOString().substring(0, 10).replace(/-/g, '');
2406
+
2407
+ const tombstoneTdy = `${REDIS_PREFIX}tomb:${this.account}:${dateKeyTdy}`;
2408
+ const tombstoneYdy = `${REDIS_PREFIX}tomb:${this.account}:${dateKeyYdy}`;
2409
+
2410
+ try {
2411
+ let pipeline = this.redis
2412
+ .multi()
2413
+ // start syncing new messages from current time
2414
+ .hset(this.getAccountKey(), 'notifyFrom', notifyFrom.toISOString())
2415
+ // mark connection count to 0 to trigger `accountInitialized` event
2416
+ .hset(this.getAccountKey(), `state:count:connected`, '0')
2417
+ .del(this.getMailboxListKey()) // mailbox list
2418
+ // Note: mailbox ID hash (iah:) and listRegistry are intentionally NOT deleted
2419
+ // to preserve message ID stability across reconnections. Message IDs depend on
2420
+ // the specific mailbox ID entries, so deleting them would invalidate existing IDs.
2421
+ .del(`${REDIS_PREFIX}iar:b:${this.account}`) // bounce list
2422
+ .del(`${REDIS_PREFIX}iar:s:${this.account}`) // seen messages list
2423
+ .del(tombstoneTdy)
2424
+ .del(tombstoneYdy);
2425
+
2426
+ if (imapIndexer) {
2427
+ pipeline = pipeline.hset(this.getAccountKey(), 'imapIndexer', imapIndexer);
2428
+ }
2429
+
2430
+ if (data.syncFrom || data.syncFrom === null) {
2431
+ pipeline = pipeline.hset(this.getAccountKey(), 'syncFrom', data.syncFrom ? data.syncFrom.toISOString() : 'null');
2432
+ }
2433
+
2434
+ await pipeline.exec();
2435
+
2436
+ // scan and delete keys
2437
+ await redisScanDelete(this.redis, this.logger, `${REDIS_PREFIX}iam:${this.account}:*`);
2438
+
2439
+ if (await settings.get('documentStoreEnabled')) {
2440
+ // Flush ElasticSearch index for this account
2441
+ const { index, client } = this.esClient;
2442
+ if (!client) {
2443
+ return;
2444
+ }
2445
+
2446
+ let deleteResult = {};
2447
+ let deletedCount = 0;
2448
+
2449
+ let filterQuery = {
2450
+ match: {
2451
+ account: this.account
2452
+ }
2453
+ };
2454
+
2455
+ for (let indexName of [index, `${index}.threads`, `${index}.embeddings`]) {
2456
+ try {
2457
+ deleteResult[indexName] = await client.deleteByQuery({
2458
+ index: indexName,
2459
+ query: filterQuery
2460
+ });
2461
+ deletedCount += deleteResult[indexName].deleted || 0;
2462
+ } catch (err) {
2463
+ logger.error({
2464
+ msg: 'Failed to delete account emails from index',
2465
+ action: 'flush',
2466
+ code: 'document_delete_account_error',
2467
+ index: indexName,
2468
+ request: filterQuery,
2469
+ err
2470
+ });
2471
+ if (indexName === index) {
2472
+ throw err;
2473
+ }
2474
+ }
2475
+ }
2476
+
2477
+ logger.trace({
2478
+ msg: 'Deleted account emails from index',
2479
+ action: 'flush',
2480
+ code: 'document_delete_account',
2481
+ deletedCount,
2482
+ deleteResult
2483
+ });
2484
+ }
2485
+
2486
+ return true;
2487
+ } finally {
2488
+ let finalize = async () => {
2489
+ // Wait a bit before resuming. Just to be sure all pending processes have been completed.
2490
+ await new Promise(r => setTimeout(r, 5 * 1000));
2491
+ await this.call({
2492
+ cmd: 'resume',
2493
+ account: this.account,
2494
+ timeout: this.timeout
2495
+ });
2496
+ };
2497
+ finalize().catch(err => {
2498
+ this.logger.error({ msg: 'Failed to finish flushing', account: this.account, err });
2499
+ });
2500
+ }
2501
+ } finally {
2502
+ await lock.releaseLock(flushLock);
2503
+ }
2504
+ }
2505
+
2506
+ async renewAccessToken(oauth2Opts) {
2507
+ let lockKey = ['oauth', this.account].join(':');
2508
+
534
2509
  let accountData = await this.loadAccountData(this.account, false);
535
- let oAuth2Client = new OAuth2Client(oauthKeys.clientId, oauthKeys.clientSecret, oauthKeys.redirectUrl);
536
- oAuth2Client.setCredentials({
537
- access_token: accountData.oauth2.accessToken,
538
- refresh_token: accountData.oauth2.refreshToken,
539
- scope: accountData.oauth2.scope || 'https://mail.google.com/',
540
- token_type: accountData.oauth2.tokenType || 'Bearer',
541
- expiry_date: accountData.oauth2.expires
542
- });
2510
+ if (accountData.oauth2?.expires > new Date(Date.now() + 30 * 1000)) {
2511
+ return accountData;
2512
+ }
543
2513
 
544
- let r = await oAuth2Client.refreshAccessTokenAsync();
545
- if (!r || !r.credentials) {
546
- throw new Error('Failed to renew access tokens');
2514
+ let lock = this.getLock();
2515
+ let renewLock;
2516
+
2517
+ try {
2518
+ renewLock = await lock.waitAcquireLock(lockKey, 5 * 60 * 1000, 1 * 60 * 1000);
2519
+ if (!renewLock.success) {
2520
+ this.logger.error({ msg: 'Failed to get lock', lockKey });
2521
+ throw new Error('Failed to get renewal lock');
2522
+ }
2523
+ } catch (err) {
2524
+ this.logger.error({ msg: 'Failed to get lock', lockKey, err });
2525
+ let error = Boom.boomify(new Error('Failed to get renewal lock'), { statusCode: 500 });
2526
+ if (err.code) {
2527
+ error.output.payload.code = err.code || 'LockFail';
2528
+ }
2529
+ throw error;
547
2530
  }
548
2531
 
549
- this.logger.info({ msg: 'Renewed oauth tokens' });
2532
+ try {
2533
+ let accountData = await this.loadAccountData(this.account, false);
2534
+
2535
+ // check if the token was already renewed
2536
+ if (
2537
+ accountData.oauth2 &&
2538
+ accountData.oauth2.accessToken &&
2539
+ accountData.oauth2.expires &&
2540
+ accountData.oauth2.expires > new Date(Date.now() + 30 * 1000)
2541
+ ) {
2542
+ this.logger.info({
2543
+ msg: 'OAuth2 access token renewed while locked',
2544
+ action: 'ensureAccessToken',
2545
+ error: null,
2546
+ user: accountData.oauth2.auth.user,
2547
+ expires: accountData.oauth2.expires,
2548
+ scopes: accountData.oauth2.scope,
2549
+ oauth2App: accountData.oauth2.provider
2550
+ });
2551
+ return accountData;
2552
+ }
550
2553
 
551
- accountData.oauth2 = Object.assign(accountData.oauth2 || {}, {
552
- accessToken: r.credentials.access_token,
553
- expires: new Date(r.credentials.expiry_date).toISOString()
554
- });
2554
+ const oAuth2Client = await oauth2Apps.getClient(accountData.oauth2.provider, oauth2Opts);
2555
+
2556
+ let r = await oAuth2Client.refreshToken({
2557
+ refreshToken: accountData.oauth2.refreshToken,
2558
+ // user is needed if it's a service account
2559
+ user: accountData.oauth2.auth.user
2560
+ });
2561
+
2562
+ if (r.tokenRequest?.userFlag) {
2563
+ await this.update({ account: accountData.account, oauth2: { partial: true, userFlag: r.tokenRequest?.userFlag } });
2564
+ }
2565
+
2566
+ if (!r.access_token) {
2567
+ throw new Error('Failed to renew token');
2568
+ }
2569
+
2570
+ let now = new Date();
555
2571
 
556
- await this.update({ account: accountData.account, oauth2: accountData.oauth2 });
2572
+ let updates = {
2573
+ accessToken: r.access_token,
2574
+ expires: new Date(now.getTime() + r.expires_in * 1000).toISOString(),
2575
+ generated: now.toISOString()
2576
+ };
2577
+
2578
+ if (r.refresh_token) {
2579
+ updates.refreshToken = r.refresh_token;
2580
+ updates.refreshTokenGenerated = now.toISOString();
2581
+ }
2582
+
2583
+ if (r.scope) {
2584
+ updates.scope = r.scope.split(/\s+/);
2585
+ }
2586
+
2587
+ accountData.oauth2 = Object.assign(accountData.oauth2 || {}, updates);
2588
+ delete accountData.oauth2.userFlag;
2589
+
2590
+ this.logger.info({
2591
+ msg: 'Renewed OAuth2 access token',
2592
+ action: 'ensureAccessToken',
2593
+ error: null,
2594
+ user: accountData.oauth2.auth.user,
2595
+ expires: updates.expires,
2596
+ scopes: updates.scope,
2597
+ oauth2App: accountData.oauth2.provider
2598
+ });
2599
+
2600
+ await this.update({ account: accountData.account, oauth2: accountData.oauth2 });
2601
+
2602
+ return accountData;
2603
+ } catch (err) {
2604
+ this.logger.info({
2605
+ msg: 'Failed to renew OAuth2 access token',
2606
+ action: 'ensureAccessToken',
2607
+ error: err,
2608
+ response: err.tokenRequest && err.tokenRequest.response,
2609
+ flag: err.tokenRequest && err.tokenRequest.flag
2610
+ });
2611
+
2612
+ if (err.tokenRequest?.userFlag) {
2613
+ await this.update({ account: this.account, oauth2: { partial: true, userFlag: err.tokenRequest?.userFlag } });
2614
+ }
557
2615
 
2616
+ throw err;
2617
+ } finally {
2618
+ await lock.releaseLock(renewLock);
2619
+ }
2620
+ }
2621
+
2622
+ async invalidateAccessToken() {
2623
+ let accountData = await this.loadAccountData(this.account, false);
2624
+ if (accountData.oauth2) {
2625
+ accountData.oauth2.expires = new Date(Date.now() - 24 * 3600 * 1000).toISOString();
2626
+ await this.update({ account: accountData.account, oauth2: accountData.oauth2 });
2627
+ this.logger.info({ msg: 'Invalidated the OAuth2 access token', expires: accountData.oauth2.expires });
2628
+ }
558
2629
  return accountData;
559
2630
  }
2631
+
2632
+ async getActiveAccessTokenData() {
2633
+ // throws if account does not exist
2634
+ let accountData = await this.loadAccountData(this.account);
2635
+ if (!accountData.oauth2 || !accountData.oauth2.auth || !accountData.oauth2.auth.user || !accountData.oauth2.provider) {
2636
+ let error = Boom.boomify(new Error('Not an OAuth2 account'), { statusCode: 403 });
2637
+ error.output.payload.code = 'AccountNotOAuth2';
2638
+ throw error;
2639
+ }
2640
+
2641
+ if (accountData.oauth2.useAuthServer) {
2642
+ // resolve credentials
2643
+
2644
+ let authData = await resolveCredentials(this.account, 'api');
2645
+
2646
+ return {
2647
+ account: accountData.account,
2648
+ user: authData.user,
2649
+ accessToken: authData.accessToken,
2650
+ provider: accountData.oauth2.auth.provider,
2651
+ registeredScopes: accountData.oauth2.scope,
2652
+ cached: false
2653
+ };
2654
+ }
2655
+
2656
+ let now = Date.now();
2657
+ let accessToken;
2658
+ let cached = false;
2659
+
2660
+ if (!accountData.oauth2.accessToken || !accountData.oauth2.expires || accountData.oauth2.expires < new Date(now + 30 * 1000)) {
2661
+ // renew access token
2662
+ try {
2663
+ await this.renewAccessToken();
2664
+ accountData = await this.loadAccountData(this.account);
2665
+ accessToken = accountData.oauth2.accessToken;
2666
+ } catch (err) {
2667
+ let error = Boom.boomify(err, { statusCode: 403 });
2668
+ error.output.payload.code = 'OauthRenewError';
2669
+ error.output.payload.authenticationFailed = true;
2670
+ if (err.tokenRequest) {
2671
+ error.output.payload.tokenRequest = err.tokenRequest;
2672
+ }
2673
+ throw error;
2674
+ }
2675
+ } else {
2676
+ accessToken = accountData.oauth2.accessToken;
2677
+ cached = true;
2678
+ }
2679
+
2680
+ return {
2681
+ account: accountData.account,
2682
+ user: accountData.oauth2.auth.user,
2683
+ accessToken,
2684
+ provider: accountData.oauth2.auth.provider,
2685
+ registeredScopes: accountData.oauth2.scope,
2686
+ expires:
2687
+ accountData.oauth2.expires && typeof accountData.oauth2.expires.toISOString === 'function'
2688
+ ? accountData.oauth2.expires.toISOString()
2689
+ : accountData.oauth2.expires,
2690
+ cached
2691
+ };
2692
+ }
2693
+
2694
+ async isApiClient(accountData) {
2695
+ if (accountData.oauth2?.auth?.delegatedAccount) {
2696
+ let delegatedAccount = await resolveDelegatedAccount(this.redis, accountData.account);
2697
+ if (delegatedAccount) {
2698
+ accountData.delegatedAccount = delegatedAccount;
2699
+ let delegatedAccountRow = await this.redis.hgetall(`${REDIS_PREFIX}iad:${delegatedAccount}`);
2700
+ let delegatedAccountData = this.unserializeAccountData(delegatedAccountRow);
2701
+ if (delegatedAccountData?.oauth2?.provider) {
2702
+ let app = await oauth2Apps.get(delegatedAccountData.oauth2.provider);
2703
+ return app?.baseScopes === 'api';
2704
+ } else {
2705
+ return false;
2706
+ }
2707
+ }
2708
+ }
2709
+
2710
+ if (accountData.oauth2?.provider) {
2711
+ let app = await oauth2Apps.get(accountData.oauth2.provider);
2712
+ return app?.baseScopes === 'api';
2713
+ }
2714
+ return false;
2715
+ }
2716
+
2717
+ async oauth2Request(url, method, payload, options) {
2718
+ let accountData = await this.loadAccountData(this.account);
2719
+ if (!accountData.oauth2 || !accountData.oauth2.auth || !accountData.oauth2.auth.user || !accountData.oauth2.provider) {
2720
+ let error = Boom.boomify(new Error('Not an OAuth2 account'), { statusCode: 403 });
2721
+ error.output.payload.code = 'AccountNotOAuth2';
2722
+ throw error;
2723
+ }
2724
+
2725
+ let oAuth2Client = await oauth2Apps.getClient(accountData.oauth2.provider, {
2726
+ logger: this.logger
2727
+ });
2728
+
2729
+ const tokenData = await this.getActiveAccessTokenData();
2730
+ if (!tokenData?.accessToken) {
2731
+ let error = Boom.boomify(new Error('Can not provide access token'), { statusCode: 403 });
2732
+ error.output.payload.code = 'MissingAccessToken';
2733
+ throw error;
2734
+ }
2735
+
2736
+ if (oAuth2Client.apiBase) {
2737
+ let urlObj = new URL(url, oAuth2Client.apiBase);
2738
+ url = urlObj.href;
2739
+ }
2740
+
2741
+ return await oAuth2Client.request(tokenData?.accessToken, url, method, payload, options);
2742
+ }
2743
+
2744
+ async pushQueueEvent(event) {
2745
+ let evObj = {
2746
+ t: new Date(),
2747
+ e: event
2748
+ };
2749
+ try {
2750
+ let res = await this.redis.lpush(this.getExternalQueueKey(), msgpack.encode(evObj));
2751
+
2752
+ this.logger.debug({
2753
+ msg: 'MS Graph subscription event',
2754
+ type: 'event',
2755
+ event,
2756
+ res
2757
+ });
2758
+ } catch (err) {
2759
+ this.logger.error({ msg: 'Failed to insert event to account queue', event, err });
2760
+ }
2761
+
2762
+ try {
2763
+ await this.call({ cmd: 'externalNotify', accounts: [this.account] });
2764
+ } catch (err) {
2765
+ logger.error({ msg: 'Failed to notify about queue changes', err });
2766
+ }
2767
+ }
2768
+
2769
+ async pullQueueEvent() {
2770
+ let event;
2771
+ try {
2772
+ let evBuf = await this.redis.rpopBuffer(this.getExternalQueueKey());
2773
+ if (evBuf) {
2774
+ let { t, e } = msgpack.decode(evBuf);
2775
+ event = e;
2776
+ this.logger.debug({ msg: 'Processing event from the account queue', event, queueTime: t, queueDelay: (Date.now() - t.getTime()) / 1000 });
2777
+ }
2778
+ } catch (err) {
2779
+ this.logger.error({ msg: 'Failed to pull event from the account queue', err });
2780
+ }
2781
+
2782
+ return event || null;
2783
+ }
560
2784
  }
561
2785
 
562
2786
  module.exports = { Account };