emailengine-app 1.14.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.eslintrc +14 -0
- package/.github/CODE_OF_CONDUCT.md +76 -0
- package/.github/FUNDING.yml +4 -0
- package/.github/ISSUE_TEMPLATE/bug_report.md +38 -0
- package/.github/ISSUE_TEMPLATE/feature_request.md +20 -0
- package/.github/contributing.md +17 -0
- package/.ncurc.js +10 -0
- package/.prettierrc.js +8 -0
- package/Dockerfile +17 -0
- package/Gruntfile.js +16 -0
- package/LICENSE.txt +661 -0
- package/README.md +524 -0
- package/bin/emailengine.js +14 -0
- package/config/default.toml +39 -0
- package/docker-compose.yml +47 -0
- package/encrypt.js +179 -0
- package/examples/api.md +137 -0
- package/examples/auth-server.js +104 -0
- package/getswagger.sh +8 -0
- package/lib/account.js +562 -0
- package/lib/append-list.js +67 -0
- package/lib/bounce-detect.js +380 -0
- package/lib/connection.js +1753 -0
- package/lib/consts.js +22 -0
- package/lib/db.js +72 -0
- package/lib/encrypt.js +100 -0
- package/lib/enum-message-flags.js +6 -0
- package/lib/get-raw-email.js +292 -0
- package/lib/get-secret.js +83 -0
- package/lib/logger.js +35 -0
- package/lib/lua/s-list-accounts.lua +51 -0
- package/lib/lua/z-expunge.lua +20 -0
- package/lib/lua/z-get-by-uid.lua +16 -0
- package/lib/lua/z-get-mailbox-id.lua +15 -0
- package/lib/lua/z-get-mailbox-path.lua +4 -0
- package/lib/lua/z-get.lua +15 -0
- package/lib/lua/z-push.lua +14 -0
- package/lib/lua/z-set.lua +17 -0
- package/lib/mailbox.js +1545 -0
- package/lib/message-port-stream.js +79 -0
- package/lib/schemas.js +311 -0
- package/lib/settings.js +63 -0
- package/lib/tools.js +488 -0
- package/package.json +79 -0
- package/scan.js +111 -0
- package/server.js +672 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.css +3872 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.min.css +7 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-grid.min.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.css +325 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.min.css +8 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap-reboot.min.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap.css +10298 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css +7 -0
- package/static/bootstrap-4.6.0-dist/css/bootstrap.min.css.map +1 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js +7045 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.js.map +1 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js +7 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.bundle.min.js.map +1 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.js +4432 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.js.map +1 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js +7 -0
- package/static/bootstrap-4.6.0-dist/js/bootstrap.min.js.map +1 -0
- package/static/css/callout.css +63 -0
- package/static/css/emailengine.css +33 -0
- package/static/favicon/android-chrome-192x192.png +0 -0
- package/static/favicon/android-chrome-512x512.png +0 -0
- package/static/favicon/apple-touch-icon.png +0 -0
- package/static/favicon/favicon-16x16.png +0 -0
- package/static/favicon/favicon-32x32.png +0 -0
- package/static/favicon/manifest.json +20 -0
- package/static/favicon.ico +0 -0
- package/static/icons/alarm-fill.svg +3 -0
- package/static/icons/alarm.svg +7 -0
- package/static/icons/alert-circle-fill.svg +3 -0
- package/static/icons/alert-circle.svg +4 -0
- package/static/icons/alert-octagon-fill.svg +3 -0
- package/static/icons/alert-octagon.svg +5 -0
- package/static/icons/alert-square-fill.svg +3 -0
- package/static/icons/alert-square.svg +5 -0
- package/static/icons/alert-triangle-fill.svg +3 -0
- package/static/icons/alert-triangle.svg +5 -0
- package/static/icons/archive-fill.svg +3 -0
- package/static/icons/archive.svg +4 -0
- package/static/icons/arrow-bar-bottom.svg +4 -0
- package/static/icons/arrow-bar-left.svg +4 -0
- package/static/icons/arrow-bar-right.svg +4 -0
- package/static/icons/arrow-bar-up.svg +4 -0
- package/static/icons/arrow-clockwise.svg +4 -0
- package/static/icons/arrow-counterclockwise.svg +4 -0
- package/static/icons/arrow-down-left.svg +4 -0
- package/static/icons/arrow-down-right.svg +4 -0
- package/static/icons/arrow-down-short.svg +4 -0
- package/static/icons/arrow-down.svg +4 -0
- package/static/icons/arrow-left-right.svg +5 -0
- package/static/icons/arrow-left-short.svg +4 -0
- package/static/icons/arrow-left.svg +4 -0
- package/static/icons/arrow-repeat.svg +5 -0
- package/static/icons/arrow-right-short.svg +4 -0
- package/static/icons/arrow-right.svg +4 -0
- package/static/icons/arrow-up-down.svg +5 -0
- package/static/icons/arrow-up-left.svg +4 -0
- package/static/icons/arrow-up-right.svg +4 -0
- package/static/icons/arrow-up-short.svg +4 -0
- package/static/icons/arrow-up.svg +4 -0
- package/static/icons/arrows-angle-contract.svg +5 -0
- package/static/icons/arrows-angle-expand.svg +5 -0
- package/static/icons/arrows-collapse.svg +5 -0
- package/static/icons/arrows-expand.svg +5 -0
- package/static/icons/arrows-fullscreen.svg +7 -0
- package/static/icons/at.svg +3 -0
- package/static/icons/award.svg +4 -0
- package/static/icons/backspace-fill.svg +3 -0
- package/static/icons/backspace-reverse-fill.svg +3 -0
- package/static/icons/backspace-reverse.svg +5 -0
- package/static/icons/backspace.svg +5 -0
- package/static/icons/bar-chart-fill.svg +5 -0
- package/static/icons/bar-chart.svg +3 -0
- package/static/icons/battery-charging.svg +5 -0
- package/static/icons/battery-full.svg +4 -0
- package/static/icons/battery.svg +4 -0
- package/static/icons/bell-fill.svg +3 -0
- package/static/icons/bell.svg +4 -0
- package/static/icons/blockquote-left.svg +4 -0
- package/static/icons/blockquote-right.svg +4 -0
- package/static/icons/book-half-fill.svg +4 -0
- package/static/icons/book.svg +4 -0
- package/static/icons/bookmark-fill.svg +3 -0
- package/static/icons/bookmark.svg +3 -0
- package/static/icons/bootstrap-fill.svg +3 -0
- package/static/icons/bootstrap-reboot.svg +3 -0
- package/static/icons/bootstrap.svg +4 -0
- package/static/icons/box-arrow-bottom-left.svg +4 -0
- package/static/icons/box-arrow-bottom-right.svg +4 -0
- package/static/icons/box-arrow-down.svg +5 -0
- package/static/icons/box-arrow-left.svg +5 -0
- package/static/icons/box-arrow-right.svg +5 -0
- package/static/icons/box-arrow-up-left.svg +4 -0
- package/static/icons/box-arrow-up-right.svg +4 -0
- package/static/icons/box-arrow-up.svg +5 -0
- package/static/icons/braces.svg +3 -0
- package/static/icons/brightness-fill-high.svg +4 -0
- package/static/icons/brightness-fill-low.svg +11 -0
- package/static/icons/brightness-high.svg +3 -0
- package/static/icons/brightness-low.svg +11 -0
- package/static/icons/brush.svg +4 -0
- package/static/icons/bucket-fill.svg +4 -0
- package/static/icons/bucket.svg +4 -0
- package/static/icons/building.svg +5 -0
- package/static/icons/bullseye.svg +6 -0
- package/static/icons/calendar-fill.svg +4 -0
- package/static/icons/calendar.svg +4 -0
- package/static/icons/camera-video-fill.svg +4 -0
- package/static/icons/camera-video.svg +4 -0
- package/static/icons/camera.svg +5 -0
- package/static/icons/capslock-fill.svg +3 -0
- package/static/icons/capslock.svg +3 -0
- package/static/icons/chat-fill.svg +3 -0
- package/static/icons/chat.svg +3 -0
- package/static/icons/check-box.svg +4 -0
- package/static/icons/check-circle.svg +4 -0
- package/static/icons/check.svg +3 -0
- package/static/icons/chevron-compact-down.svg +3 -0
- package/static/icons/chevron-compact-left.svg +3 -0
- package/static/icons/chevron-compact-right.svg +3 -0
- package/static/icons/chevron-compact-up.svg +3 -0
- package/static/icons/chevron-down.svg +3 -0
- package/static/icons/chevron-left.svg +3 -0
- package/static/icons/chevron-right.svg +3 -0
- package/static/icons/chevron-up.svg +3 -0
- package/static/icons/circle-fill.svg +3 -0
- package/static/icons/circle-half.svg +3 -0
- package/static/icons/circle-slash.svg +3 -0
- package/static/icons/circle.svg +3 -0
- package/static/icons/clock-fill.svg +3 -0
- package/static/icons/clock.svg +4 -0
- package/static/icons/cloud-download.svg +5 -0
- package/static/icons/cloud-fill.svg +3 -0
- package/static/icons/cloud-upload.svg +5 -0
- package/static/icons/cloud.svg +3 -0
- package/static/icons/code-slash.svg +3 -0
- package/static/icons/code.svg +3 -0
- package/static/icons/columns-gutters.svg +3 -0
- package/static/icons/columns.svg +4 -0
- package/static/icons/command.svg +4 -0
- package/static/icons/compass.svg +5 -0
- package/static/icons/cone-striped.svg +4 -0
- package/static/icons/cone.svg +4 -0
- package/static/icons/controller.svg +5 -0
- package/static/icons/credit-card.svg +5 -0
- package/static/icons/cursor-fill.svg +3 -0
- package/static/icons/cursor.svg +3 -0
- package/static/icons/dash.svg +3 -0
- package/static/icons/diamond-half.svg +3 -0
- package/static/icons/diamond.svg +3 -0
- package/static/icons/display-fill.svg +5 -0
- package/static/icons/display.svg +4 -0
- package/static/icons/document-code.svg +4 -0
- package/static/icons/document-diff.svg +5 -0
- package/static/icons/document-richtext.svg +4 -0
- package/static/icons/document-spreadsheet.svg +5 -0
- package/static/icons/document-text.svg +4 -0
- package/static/icons/document.svg +3 -0
- package/static/icons/documents-alt.svg +4 -0
- package/static/icons/documents.svg +4 -0
- package/static/icons/dot.svg +3 -0
- package/static/icons/download.svg +5 -0
- package/static/icons/egg-fried.svg +4 -0
- package/static/icons/eject-fill.svg +3 -0
- package/static/icons/eject.svg +3 -0
- package/static/icons/envelope-fill.svg +3 -0
- package/static/icons/envelope-open-fill.svg +3 -0
- package/static/icons/envelope-open.svg +5 -0
- package/static/icons/envelope.svg +4 -0
- package/static/icons/eye-fill.svg +4 -0
- package/static/icons/eye-slash-fill.svg +5 -0
- package/static/icons/eye-slash.svg +6 -0
- package/static/icons/eye.svg +4 -0
- package/static/icons/filter.svg +3 -0
- package/static/icons/flag-fill.svg +4 -0
- package/static/icons/flag.svg +4 -0
- package/static/icons/folder-fill.svg +3 -0
- package/static/icons/folder-symlink-fill.svg +3 -0
- package/static/icons/folder-symlink.svg +5 -0
- package/static/icons/folder.svg +4 -0
- package/static/icons/fonts.svg +3 -0
- package/static/icons/forward-fill.svg +3 -0
- package/static/icons/forward.svg +3 -0
- package/static/icons/gear-fill.svg +3 -0
- package/static/icons/gear-wide-connected.svg +4 -0
- package/static/icons/gear-wide.svg +3 -0
- package/static/icons/gear.svg +4 -0
- package/static/icons/geo.svg +5 -0
- package/static/icons/graph-down.svg +5 -0
- package/static/icons/graph-up.svg +5 -0
- package/static/icons/grid-fill.svg +6 -0
- package/static/icons/grid.svg +3 -0
- package/static/icons/hammer.svg +4 -0
- package/static/icons/hash.svg +3 -0
- package/static/icons/heart-fill.svg +3 -0
- package/static/icons/heart.svg +3 -0
- package/static/icons/house-fill.svg +4 -0
- package/static/icons/house.svg +4 -0
- package/static/icons/image-alt.svg +4 -0
- package/static/icons/image-fill.svg +3 -0
- package/static/icons/image.svg +5 -0
- package/static/icons/images.svg +5 -0
- package/static/icons/inbox-fill.svg +4 -0
- package/static/icons/inbox.svg +4 -0
- package/static/icons/inboxes-fill.svg +4 -0
- package/static/icons/inboxes.svg +4 -0
- package/static/icons/info-fill.svg +3 -0
- package/static/icons/info-square-fill.svg +3 -0
- package/static/icons/info-square.svg +5 -0
- package/static/icons/info.svg +5 -0
- package/static/icons/justify-left.svg +3 -0
- package/static/icons/justify-right.svg +3 -0
- package/static/icons/justify.svg +3 -0
- package/static/icons/kanban-fill.svg +3 -0
- package/static/icons/kanban.svg +6 -0
- package/static/icons/laptop.svg +4 -0
- package/static/icons/layout-sidebar-reverse.svg +4 -0
- package/static/icons/layout-sidebar.svg +4 -0
- package/static/icons/layout-split.svg +3 -0
- package/static/icons/list-check.svg +3 -0
- package/static/icons/list-ol.svg +4 -0
- package/static/icons/list-task.svg +5 -0
- package/static/icons/list-ul.svg +3 -0
- package/static/icons/list.svg +3 -0
- package/static/icons/lock-fill.svg +4 -0
- package/static/icons/lock.svg +3 -0
- package/static/icons/map.svg +3 -0
- package/static/icons/mic.svg +4 -0
- package/static/icons/moon.svg +3 -0
- package/static/icons/music-player-fill.svg +4 -0
- package/static/icons/music-player.svg +5 -0
- package/static/icons/option.svg +3 -0
- package/static/icons/outlet.svg +5 -0
- package/static/icons/pause-fill.svg +3 -0
- package/static/icons/pause.svg +3 -0
- package/static/icons/pen.svg +5 -0
- package/static/icons/pencil.svg +4 -0
- package/static/icons/people-fill.svg +3 -0
- package/static/icons/people.svg +3 -0
- package/static/icons/person-fill.svg +3 -0
- package/static/icons/person.svg +3 -0
- package/static/icons/phone-landscape.svg +4 -0
- package/static/icons/phone.svg +4 -0
- package/static/icons/pie-chart-fill.svg +3 -0
- package/static/icons/pie-chart.svg +4 -0
- package/static/icons/play-fill.svg +3 -0
- package/static/icons/play.svg +3 -0
- package/static/icons/plug.svg +4 -0
- package/static/icons/plus.svg +4 -0
- package/static/icons/power.svg +4 -0
- package/static/icons/question-fill.svg +3 -0
- package/static/icons/question-square-fill.svg +3 -0
- package/static/icons/question-square.svg +4 -0
- package/static/icons/question.svg +4 -0
- package/static/icons/reply-all-fill.svg +4 -0
- package/static/icons/reply-all.svg +4 -0
- package/static/icons/reply-fill.svg +3 -0
- package/static/icons/reply.svg +3 -0
- package/static/icons/screwdriver.svg +3 -0
- package/static/icons/search.svg +4 -0
- package/static/icons/shield-fill.svg +3 -0
- package/static/icons/shield-lock-fill.svg +3 -0
- package/static/icons/shield-lock.svg +5 -0
- package/static/icons/shield-shaded.svg +4 -0
- package/static/icons/shield.svg +3 -0
- package/static/icons/shift-fill.svg +3 -0
- package/static/icons/shift.svg +3 -0
- package/static/icons/skip-backward-fill.svg +4 -0
- package/static/icons/skip-backward.svg +3 -0
- package/static/icons/skip-end-fill.svg +5 -0
- package/static/icons/skip-end.svg +4 -0
- package/static/icons/skip-forward-fill.svg +5 -0
- package/static/icons/skip-forward.svg +3 -0
- package/static/icons/skip-start-fill.svg +4 -0
- package/static/icons/skip-start.svg +4 -0
- package/static/icons/speaker.svg +4 -0
- package/static/icons/square-fill.svg +3 -0
- package/static/icons/square-half.svg +3 -0
- package/static/icons/square.svg +3 -0
- package/static/icons/star-fill.svg +3 -0
- package/static/icons/star-half.svg +3 -0
- package/static/icons/star.svg +3 -0
- package/static/icons/stop-fill.svg +3 -0
- package/static/icons/stop.svg +3 -0
- package/static/icons/stopwatch-fill.svg +3 -0
- package/static/icons/stopwatch.svg +5 -0
- package/static/icons/sun.svg +4 -0
- package/static/icons/table.svg +7 -0
- package/static/icons/tablet-landscape.svg +4 -0
- package/static/icons/tablet.svg +4 -0
- package/static/icons/tag-fill.svg +3 -0
- package/static/icons/tag.svg +4 -0
- package/static/icons/terminal-fill.svg +3 -0
- package/static/icons/terminal.svg +4 -0
- package/static/icons/text-center.svg +3 -0
- package/static/icons/text-indent-left.svg +3 -0
- package/static/icons/text-indent-right.svg +3 -0
- package/static/icons/text-left.svg +3 -0
- package/static/icons/text-right.svg +3 -0
- package/static/icons/three-dots-vertical.svg +3 -0
- package/static/icons/three-dots.svg +3 -0
- package/static/icons/toggle-off.svg +3 -0
- package/static/icons/toggle-on.svg +3 -0
- package/static/icons/toggles.svg +4 -0
- package/static/icons/tools.svg +4 -0
- package/static/icons/trash-fill.svg +3 -0
- package/static/icons/trash.svg +4 -0
- package/static/icons/triangle-fill.svg +3 -0
- package/static/icons/triangle-half.svg +3 -0
- package/static/icons/triangle.svg +3 -0
- package/static/icons/trophy.svg +6 -0
- package/static/icons/tv-fill.svg +3 -0
- package/static/icons/tv.svg +3 -0
- package/static/icons/type-bold.svg +3 -0
- package/static/icons/type-h1.svg +3 -0
- package/static/icons/type-h2.svg +3 -0
- package/static/icons/type-h3.svg +3 -0
- package/static/icons/type-italic.svg +3 -0
- package/static/icons/type-strikethrough.svg +4 -0
- package/static/icons/type-underline.svg +4 -0
- package/static/icons/type.svg +3 -0
- package/static/icons/unlock-fill.svg +4 -0
- package/static/icons/unlock.svg +3 -0
- package/static/icons/upload.svg +4 -0
- package/static/icons/volume-down-fill.svg +4 -0
- package/static/icons/volume-down.svg +4 -0
- package/static/icons/volume-mute-fill.svg +4 -0
- package/static/icons/volume-mute.svg +4 -0
- package/static/icons/volume-up-fill.svg +6 -0
- package/static/icons/volume-up.svg +6 -0
- package/static/icons/wallet.svg +3 -0
- package/static/icons/watch.svg +5 -0
- package/static/icons/wifi.svg +5 -0
- package/static/icons/window.svg +5 -0
- package/static/icons/wrench.svg +3 -0
- package/static/icons/x-circle-fill.svg +3 -0
- package/static/icons/x-circle.svg +5 -0
- package/static/icons/x-octagon-fill.svg +3 -0
- package/static/icons/x-octagon.svg +4 -0
- package/static/icons/x-square-fill.svg +3 -0
- package/static/icons/x-square.svg +4 -0
- package/static/icons/x.svg +4 -0
- package/static/index.html +752 -0
- package/static/js/emailengine.js +581 -0
- package/static/js/jquery-3.4.1.slim.min.js +2 -0
- package/static/js/moment-with-locales-2.24.0.min.js +1 -0
- package/static/js/popper.min.js +5 -0
- package/static/logo.png +0 -0
- package/systemd/emailengine.service +89 -0
- package/systemd/nginx-proxy.conf +77 -0
- package/views/error.hbs +2 -0
- package/workers/api.js +2266 -0
- package/workers/arena.js +89 -0
- package/workers/imap.js +611 -0
- package/workers/smtp.js +278 -0
- package/workers/submit.js +214 -0
- package/workers/webhooks.js +134 -0
package/encrypt.js
ADDED
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
require('dotenv').config();
|
|
4
|
+
try {
|
|
5
|
+
process.chdir(__dirname);
|
|
6
|
+
} catch (err) {
|
|
7
|
+
// ignore
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
process.title = 'emailengine-encrypt';
|
|
11
|
+
|
|
12
|
+
const { redis } = require('./lib/db');
|
|
13
|
+
const config = require('wild-config');
|
|
14
|
+
const { encrypt, decrypt, parseEncryptedData } = require('./lib/encrypt');
|
|
15
|
+
const { encryptedKeys } = require('./lib/settings');
|
|
16
|
+
const getSecret = require('./lib/get-secret');
|
|
17
|
+
|
|
18
|
+
const DECRYPT_PASSWORDS = [].concat(config.decrypt || []);
|
|
19
|
+
|
|
20
|
+
async function processSecret(value, encryptSecret) {
|
|
21
|
+
let lastErr = false;
|
|
22
|
+
let decrypted = value;
|
|
23
|
+
|
|
24
|
+
for (let password of DECRYPT_PASSWORDS) {
|
|
25
|
+
try {
|
|
26
|
+
decrypted = decrypt(value, password);
|
|
27
|
+
if (password === encryptSecret) {
|
|
28
|
+
// nothing was changed
|
|
29
|
+
return value;
|
|
30
|
+
}
|
|
31
|
+
break;
|
|
32
|
+
} catch (err) {
|
|
33
|
+
lastErr = err;
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
let parsed = parseEncryptedData(decrypted);
|
|
38
|
+
if (parsed.format !== 'cleartext') {
|
|
39
|
+
// was not able to decrypt
|
|
40
|
+
if (encryptSecret) {
|
|
41
|
+
try {
|
|
42
|
+
decrypted = decrypt(value, encryptSecret);
|
|
43
|
+
// did not throw, so the value is already encrypted with the new password
|
|
44
|
+
return value;
|
|
45
|
+
} catch (err) {
|
|
46
|
+
// ignore
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
throw lastErr || new Error('Could not decrypt encrypted password');
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
if (encryptSecret) {
|
|
54
|
+
// encrypt
|
|
55
|
+
return encrypt(decrypted, encryptSecret);
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
// return plaintext
|
|
59
|
+
return decrypted;
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
async function main() {
|
|
63
|
+
console.error('EmailEngine account encryption tool');
|
|
64
|
+
|
|
65
|
+
const encryptSecret = await getSecret();
|
|
66
|
+
|
|
67
|
+
if (!encryptSecret && !DECRYPT_PASSWORDS.length) {
|
|
68
|
+
console.error('Usage:');
|
|
69
|
+
console.error(' emailengine encrypt --dbs.redis="redis://url" --service.secret="new-pass" --decrypt="old-pass"');
|
|
70
|
+
console.error('Where');
|
|
71
|
+
console.error(' --dbs.redis is a Redis configuration URL');
|
|
72
|
+
console.error(' --service.secret is the secret value to use for encryption.');
|
|
73
|
+
console.error(' Leave empty to remove encryption.');
|
|
74
|
+
console.error(' --decrypt is the old secret value. Not needed if current passwords are not encrypted.');
|
|
75
|
+
console.error(' You can set this value multiple times if accounts are enrypted with different secrets.');
|
|
76
|
+
return;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
// convert settings
|
|
80
|
+
for (let key of encryptedKeys) {
|
|
81
|
+
let value = await redis.hget('settings', key);
|
|
82
|
+
if (value && typeof value === 'string') {
|
|
83
|
+
try {
|
|
84
|
+
let updated = await processSecret(value, encryptSecret);
|
|
85
|
+
if (updated !== value) {
|
|
86
|
+
await redis.hset('settings', key, updated);
|
|
87
|
+
console.log(`${key}: Updated setting value`);
|
|
88
|
+
}
|
|
89
|
+
} catch (err) {
|
|
90
|
+
console.error(`${key}: Failed to process setting value`);
|
|
91
|
+
console.error(err);
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
let updatedAccounts = 0;
|
|
97
|
+
let accounts = await redis.smembers('ia:accounts');
|
|
98
|
+
for (let account of accounts) {
|
|
99
|
+
let accountData = await redis.hgetall(`iad:${account}`);
|
|
100
|
+
if (!accountData) {
|
|
101
|
+
continue;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
let updates = {};
|
|
105
|
+
let updated = false;
|
|
106
|
+
for (let key of ['imap', 'smtp', 'oauth2']) {
|
|
107
|
+
if (!accountData[key]) {
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
try {
|
|
112
|
+
accountData[key] = JSON.parse(accountData[key]);
|
|
113
|
+
} catch (err) {
|
|
114
|
+
console.error(`Failed to parse ${key} for ${account}`);
|
|
115
|
+
console.error(err);
|
|
116
|
+
continue;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
if (!accountData[key]) {
|
|
120
|
+
continue;
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
let changes = false;
|
|
124
|
+
|
|
125
|
+
for (let subKey of ['pass', 'accessToken', 'refreshToken']) {
|
|
126
|
+
if (accountData[key].auth && accountData[key].auth[subKey]) {
|
|
127
|
+
try {
|
|
128
|
+
let value = await processSecret(accountData[key].auth[subKey], encryptSecret);
|
|
129
|
+
if (value !== accountData[key].auth[subKey]) {
|
|
130
|
+
accountData[key].auth[subKey] = value;
|
|
131
|
+
changes = true;
|
|
132
|
+
}
|
|
133
|
+
} catch (err) {
|
|
134
|
+
console.error(`Could not process "${key}.auth.${subKey}" for ${account}. Check decryption secrets.`);
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
for (let subKey of ['accessToken', 'refreshToken']) {
|
|
140
|
+
if (accountData[key] && accountData[key][subKey]) {
|
|
141
|
+
try {
|
|
142
|
+
let value = await processSecret(accountData[key][subKey], encryptSecret);
|
|
143
|
+
if (value !== accountData[key][subKey]) {
|
|
144
|
+
accountData[key][subKey] = value;
|
|
145
|
+
changes = true;
|
|
146
|
+
}
|
|
147
|
+
} catch (err) {
|
|
148
|
+
console.error(`Could not process "${key}.${subKey}" for ${account}. Check decryption secrets.`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
if (changes) {
|
|
154
|
+
updates[key] = JSON.stringify(accountData[key]);
|
|
155
|
+
updated = true;
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
if (updated) {
|
|
160
|
+
let result = await redis.hmset(`iad:${account}`, updates);
|
|
161
|
+
if (result === 'OK') {
|
|
162
|
+
console.log(`${account}: updated`);
|
|
163
|
+
} else {
|
|
164
|
+
console.log(`${account}: Unexpected response from DB: ${result}`);
|
|
165
|
+
}
|
|
166
|
+
updatedAccounts++;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
console.log(`Updated ${updatedAccounts}/${accounts.length} accounts`);
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
main()
|
|
174
|
+
.then(() => process.exit(0))
|
|
175
|
+
.catch(err => {
|
|
176
|
+
console.error(err);
|
|
177
|
+
process.exit(1);
|
|
178
|
+
})
|
|
179
|
+
.finally();
|
package/examples/api.md
ADDED
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
# API
|
|
2
|
+
|
|
3
|
+
```
|
|
4
|
+
curl -XPOST "localhost:3000/v1/account" -H "content-type: application/json" -d '{
|
|
5
|
+
"account": "example",
|
|
6
|
+
"name": "Example",
|
|
7
|
+
"imap": {
|
|
8
|
+
"host": "localhost",
|
|
9
|
+
"port": 9993,
|
|
10
|
+
"secure": true,
|
|
11
|
+
"auth": {
|
|
12
|
+
"user": "myuser2",
|
|
13
|
+
"pass": "verysecret"
|
|
14
|
+
},
|
|
15
|
+
"tls": {
|
|
16
|
+
"rejectUnauthorized": false
|
|
17
|
+
}
|
|
18
|
+
},
|
|
19
|
+
"smtp": {
|
|
20
|
+
"host": "localhost",
|
|
21
|
+
"port": 1025,
|
|
22
|
+
"secure": false,
|
|
23
|
+
"auth": {
|
|
24
|
+
"user": "myuser2",
|
|
25
|
+
"pass": "verysecret"
|
|
26
|
+
},
|
|
27
|
+
"tls": {
|
|
28
|
+
"rejectUnauthorized": false
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
}'
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
```
|
|
35
|
+
curl -XPUT "localhost:3000/account/example" -H "content-type: application/json" -d '{
|
|
36
|
+
"imap": {
|
|
37
|
+
"host": "localhost",
|
|
38
|
+
"port": 9993,
|
|
39
|
+
"secure": true,
|
|
40
|
+
"auth": {
|
|
41
|
+
"user": "myuser2",
|
|
42
|
+
"pass": "verysecret"
|
|
43
|
+
},
|
|
44
|
+
"tls": {
|
|
45
|
+
"rejectUnauthorized": false
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
}'
|
|
49
|
+
```
|
|
50
|
+
|
|
51
|
+
```
|
|
52
|
+
curl -XPOST "localhost:3000/v1/verifyAccount" -H "content-type: application/json" -d '{
|
|
53
|
+
"imap": {
|
|
54
|
+
"host": "localhost",
|
|
55
|
+
"port": 9993,
|
|
56
|
+
"secure": true,
|
|
57
|
+
"auth": {
|
|
58
|
+
"user": "myuser2",
|
|
59
|
+
"pass": "verysecret"
|
|
60
|
+
},
|
|
61
|
+
"tls": {
|
|
62
|
+
"rejectUnauthorized": false
|
|
63
|
+
}
|
|
64
|
+
},
|
|
65
|
+
"smtp": {
|
|
66
|
+
"host": "localhost",
|
|
67
|
+
"port": 1025,
|
|
68
|
+
"secure": false,
|
|
69
|
+
"auth": {
|
|
70
|
+
"user": "myuser2",
|
|
71
|
+
"pass": "verysecret"
|
|
72
|
+
},
|
|
73
|
+
"tls": {
|
|
74
|
+
"rejectUnauthorized": false
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
}'
|
|
78
|
+
```
|
|
79
|
+
|
|
80
|
+
```
|
|
81
|
+
curl -XPOST "localhost:3000/v1/account/pangalink/submit" -H "content-type: application/json" -d '{
|
|
82
|
+
"reference": {
|
|
83
|
+
"message": "AAAAAQAACnA",
|
|
84
|
+
"action": "reply"
|
|
85
|
+
},
|
|
86
|
+
"from": {
|
|
87
|
+
"name": "Pangalink",
|
|
88
|
+
"address": "no-reply@pangalink.net"
|
|
89
|
+
},
|
|
90
|
+
"to": [{
|
|
91
|
+
"name": "Andris Reinman",
|
|
92
|
+
"address": "andris@emailengine.app"
|
|
93
|
+
}],
|
|
94
|
+
"subject": "test kiri",
|
|
95
|
+
"text": "eriti test kiri",
|
|
96
|
+
"html": "<p>eriti test kiri</p>",
|
|
97
|
+
"attachments": [
|
|
98
|
+
{
|
|
99
|
+
"filename": "checkmark.png",
|
|
100
|
+
"content": "iVBORw0KGgoAAAANSUhEUgAAABAAAAAQAQMAAAAlPW0iAAAABlBMVEUAAAD///+l2Z/dAAAAM0lEQVR4nGP4/5/h/1+G/58ZDrAz3D/McH8yw83NDDeNGe4Ug9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC"
|
|
101
|
+
}
|
|
102
|
+
]
|
|
103
|
+
}'
|
|
104
|
+
```
|
|
105
|
+
|
|
106
|
+
```
|
|
107
|
+
curl -XDELETE "localhost:3000/v1/account/example"
|
|
108
|
+
```
|
|
109
|
+
|
|
110
|
+
```
|
|
111
|
+
curl -XGET "localhost:3000/v1/account/example/mailboxes"
|
|
112
|
+
```
|
|
113
|
+
|
|
114
|
+
```
|
|
115
|
+
curl -XGET "localhost:3000/v1/account/example/messages?path=INBOX&page=1"
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
```
|
|
119
|
+
curl -XGET "localhost:3000/v1/account/pangalink/message/AAAAAQAAMlw"
|
|
120
|
+
```
|
|
121
|
+
|
|
122
|
+
```
|
|
123
|
+
curl -XGET "localhost:3000/v1/account/pangalink/message/AAAAAQAAMlw/source"
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
```
|
|
127
|
+
curl -XGET "localhost:3000/v1/account/example/text/AAAAAQAAAeGTkaExkaEykA?textType=html&maxBytes=200"
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
```
|
|
131
|
+
curl -XPUT "localhost:3000/v1/account/pangalink/message/AAAAAQAAMlw" -H "content-type: application/json" -d '{
|
|
132
|
+
"flags": {
|
|
133
|
+
"add": ["test2", "test3"],
|
|
134
|
+
"delete": ["test1"]
|
|
135
|
+
}
|
|
136
|
+
}'
|
|
137
|
+
```
|
|
@@ -0,0 +1,104 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
// This is an example authentication server
|
|
4
|
+
// It provides fixed credentials for an account called "example" and generates OAuth2 access tokens for an account called "oauth-user"
|
|
5
|
+
|
|
6
|
+
const Hapi = require('@hapi/hapi');
|
|
7
|
+
const hapiPino = require('hapi-pino');
|
|
8
|
+
const XOAuth2 = require('nodemailer/lib/xoauth2');
|
|
9
|
+
|
|
10
|
+
// Gmail oauth app credentials. Must have https://mail.google.com scope set
|
|
11
|
+
const OAUTH2_CLIENT_ID = process.env.OAUTH2_CLIENT_ID;
|
|
12
|
+
const OAUTH2_CLIENT_SECRET = process.env.OAUTH2_CLIENT_SECRET;
|
|
13
|
+
|
|
14
|
+
// Single user specific credentials as our demo only provides tokens for a single user
|
|
15
|
+
const USER_ADDRESS = process.env.USER_ADDRESS;
|
|
16
|
+
const USER_REFRESH_TOKEN = process.env.USER_REFRESH_TOKEN;
|
|
17
|
+
|
|
18
|
+
const init = async () => {
|
|
19
|
+
const server = Hapi.server({
|
|
20
|
+
port: 3080,
|
|
21
|
+
host: 'localhost'
|
|
22
|
+
});
|
|
23
|
+
|
|
24
|
+
await server.register({
|
|
25
|
+
plugin: hapiPino,
|
|
26
|
+
options: {
|
|
27
|
+
level: 'trace'
|
|
28
|
+
}
|
|
29
|
+
});
|
|
30
|
+
|
|
31
|
+
server.route({
|
|
32
|
+
method: 'GET',
|
|
33
|
+
path: '/credentials',
|
|
34
|
+
|
|
35
|
+
async handler(request) {
|
|
36
|
+
switch (request.query.account) {
|
|
37
|
+
// account with id "example" uses password based authentication
|
|
38
|
+
case 'example':
|
|
39
|
+
return {
|
|
40
|
+
user: 'myuser2',
|
|
41
|
+
pass: 'verysecret'
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
// account with id "oauth-user" uses OAuth2 tokens
|
|
45
|
+
case 'oauth-user':
|
|
46
|
+
return {
|
|
47
|
+
user: USER_ADDRESS,
|
|
48
|
+
accessToken: await getAccessToken(USER_ADDRESS, USER_REFRESH_TOKEN)
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
return false;
|
|
53
|
+
}
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
await server.start();
|
|
57
|
+
console.log('Authentication Server URL: %s/credentials', server.info.uri);
|
|
58
|
+
};
|
|
59
|
+
|
|
60
|
+
// The following crux re-uses OAuth2 token generation from the Nodemailer package.
|
|
61
|
+
// Normally you'd probably use something like this instead: https://www.npmjs.com/package/google-auth-library
|
|
62
|
+
const tokens = new Map();
|
|
63
|
+
async function getAccessToken(user, refreshToken) {
|
|
64
|
+
// check cache first
|
|
65
|
+
if (tokens.has(user)) {
|
|
66
|
+
let token = tokens.get(user);
|
|
67
|
+
if (token.expires > new Date()) {
|
|
68
|
+
// use cached token
|
|
69
|
+
return token.accessToken;
|
|
70
|
+
}
|
|
71
|
+
// clear expired token
|
|
72
|
+
tokens.delete(user);
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// generate new token
|
|
76
|
+
let token = await new Promise((resolve, reject) => {
|
|
77
|
+
let xoauth = new XOAuth2({
|
|
78
|
+
user,
|
|
79
|
+
clientId: OAUTH2_CLIENT_ID,
|
|
80
|
+
clientSecret: OAUTH2_CLIENT_SECRET,
|
|
81
|
+
refreshToken,
|
|
82
|
+
accessUrl: 'https://accounts.google.com/o/oauth2/token'
|
|
83
|
+
});
|
|
84
|
+
xoauth.generateToken(err => {
|
|
85
|
+
if (err) {
|
|
86
|
+
return reject(err);
|
|
87
|
+
}
|
|
88
|
+
if (!xoauth.accessToken) {
|
|
89
|
+
return reject(new Error('Could not generate new access token'));
|
|
90
|
+
}
|
|
91
|
+
resolve({
|
|
92
|
+
accessToken: xoauth.accessToken,
|
|
93
|
+
expires: xoauth.expires
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
// update cache
|
|
99
|
+
tokens.set(user, token);
|
|
100
|
+
|
|
101
|
+
return token.accessToken;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
init();
|