From be6506da08b4eecb812a4f190e7ae51215fab744 Mon Sep 17 00:00:00 2001 From: shamoon <4887959+shamoon@users.noreply.github.com> Date: Sat, 30 Sep 2023 18:17:42 -0700 Subject: [PATCH] Feature: password reset (#4289) --- docs/configuration.md | 29 ++ docs/usage.md | 5 + src/documents/context_processors.py | 8 + src/documents/static/signin.css | 25 +- .../templates/registration/login.html | 5 + .../registration/password_reset_complete.html | 45 ++ .../registration/password_reset_confirm.html | 70 +++ .../registration/password_reset_done.html | 44 ++ .../registration/password_reset_form.html | 58 +++ src/locale/en_US/LC_MESSAGES/django.po | 415 +++++++++++------- src/paperless/settings.py | 13 + 11 files changed, 544 insertions(+), 173 deletions(-) create mode 100644 src/documents/context_processors.py create mode 100644 src/documents/templates/registration/password_reset_complete.html create mode 100644 src/documents/templates/registration/password_reset_confirm.html create mode 100644 src/documents/templates/registration/password_reset_done.html create mode 100644 src/documents/templates/registration/password_reset_form.html diff --git a/docs/configuration.md b/docs/configuration.md index 88c8a5094..45a004b40 100644 --- a/docs/configuration.md +++ b/docs/configuration.md @@ -1301,3 +1301,32 @@ started by the container. v1.9.2. A one-time migration is performed for users who have this setting set. This setting is always ignored if the corresponding frontend setting has been set. + +## Email sending + +Setting an SMTP server for the backend will allow you to reset your +password. All of these options come from their similarly-named [Django settings](https://docs.djangoproject.com/en/4.2/ref/settings/#email-host) + +#### [`PAPERLESS_EMAIL_HOST=`](#PAPERLESS_EMAIL_HOST) {#PAPERLESS_EMAIL_HOST} + +: Defaults to 'localhost'. + +#### [`PAPERLESS_EMAIL_PORT=`](#PAPERLESS_EMAIL_PORT) {#PAPERLESS_EMAIL_PORT} + +: Defaults to port 25. + +#### [`PAPERLESS_EMAIL_HOST_USER=`](#PAPERLESS_EMAIL_HOST_USER) {#PAPERLESS_EMAIL_HOST_USER} + +: Defaults to ''. + +#### [`PAPERLESS_EMAIL_HOST_PASSWORD=`](#PAPERLESS_EMAIL_HOST_PASSWORD) {#PAPERLESS_EMAIL_HOST_PASSWORD} + +: Defaults to ''. + +#### [`PAPERLESS_EMAIL_USE_TLS=`](#PAPERLESS_EMAIL_USE_TLS) {#PAPERLESS_EMAIL_USE_TLS} + +: Defaults to false. + +#### [`PAPERLESS_EMAIL_USE_SSL=`](#PAPERLESS_EMAIL_USE_SSL) {#PAPERLESS_EMAIL_USE_SSL} + +: Defaults to false. diff --git a/docs/usage.md b/docs/usage.md index 340a1f2c3..aad4976f8 100644 --- a/docs/usage.md +++ b/docs/usage.md @@ -261,6 +261,11 @@ These can be found under Settings > Users & Groups, assuming the user has access as a member of a group those permissions will be inherited and this is reflected in the UI. Explicit permissions can be granted to limit access to certain parts of the UI (and corresponding API endpoints). +### Password reset + +In order to enable the password reset feature you will need to setup an SMTP backend, see +[`PAPERLESS_EMAIL_HOST`](/configuration#PAPERLESS_EMAIL_HOST) + ## Consumption templates Consumption templates were introduced in v2.0 and allow for finer control over what metadata (tags, doc diff --git a/src/documents/context_processors.py b/src/documents/context_processors.py new file mode 100644 index 000000000..90c856aeb --- /dev/null +++ b/src/documents/context_processors.py @@ -0,0 +1,8 @@ +from django.conf import settings as django_settings + + +def settings(request): + return { + "EMAIL_ENABLED": django_settings.EMAIL_HOST != "localhost" + or django_settings.EMAIL_HOST_USER != "", + } diff --git a/src/documents/static/signin.css b/src/documents/static/signin.css index 7115adae2..02f460c7c 100644 --- a/src/documents/static/signin.css +++ b/src/documents/static/signin.css @@ -2,32 +2,43 @@ body { --bs-body-bg: #f5f5f5; --bs-link-color-rgb: 23, 84, 31; /* #17541f */ --bs-link-hover-color-rgb: 15, 56, 20; + --pngx-primary: #17541f; + --pngx-primary-hover: #0f3614; + --pngx-primary-active: #0c2c10; } .form-control { --bs-body-bg: #fff; } -.btn { - --bs-btn-bg: #17541f; - --bs-btn-border-color: #17541f; - --bs-btn-hover-bg: #0f3614; +.btn.btn-primary { + --bs-btn-bg: var(--pngx-primary); + --bs-btn-border-color: var(--pngx-primary); + --bs-btn-hover-bg: var(--pngx-primary-hover); --bs-btn-hover-border-color: #0c2c10; - --bs-btn-active-bg: #0c2c10; + --bs-btn-active-bg: var(--pngx-primary-active); --bs-btn-active-border-color: #09220d; } +.btn-link { + --bs-btn-color: var(--pngx-primary); + --bs-btn-hover-color: var(--pngx-primary-hover); + --bs-btn-active-color: var(--pngx-primary-active); +} + .form-signin { max-width: 330px; } -#inputUsername { +#inputUsername, +#inputPassword1 { margin-bottom: -1px; border-bottom-right-radius: 0; border-bottom-left-radius: 0; } -#inputPassword { +#inputPassword, +#inputPassword2 { border-top-left-radius: 0; border-top-right-radius: 0; } diff --git a/src/documents/templates/registration/login.html b/src/documents/templates/registration/login.html index c4c936182..ba57013ac 100644 --- a/src/documents/templates/registration/login.html +++ b/src/documents/templates/registration/login.html @@ -65,6 +65,11 @@
+ {% if EMAIL_ENABLED %} + + {% endif %} diff --git a/src/documents/templates/registration/password_reset_complete.html b/src/documents/templates/registration/password_reset_complete.html new file mode 100644 index 000000000..d9b0a3b72 --- /dev/null +++ b/src/documents/templates/registration/password_reset_complete.html @@ -0,0 +1,45 @@ + + +{% load static %} +{% load i18n %} + + + + + + + + + + {% translate "Paperless-ngx reset password complete" %} + + + + + + +
+ +

{% translate "Password reset complete." %}

+ {% url 'login' as login_url %} +

{% blocktranslate %}Your new password has been set. You can now log in{% endblocktranslate %}.

+
+ + diff --git a/src/documents/templates/registration/password_reset_confirm.html b/src/documents/templates/registration/password_reset_confirm.html new file mode 100644 index 000000000..8f24212a7 --- /dev/null +++ b/src/documents/templates/registration/password_reset_confirm.html @@ -0,0 +1,70 @@ + + +{% load static %} +{% load i18n %} + + + + + + + + + + {% translate "Paperless-ngx reset password confirmation" %} + + + + + + + + + diff --git a/src/documents/templates/registration/password_reset_done.html b/src/documents/templates/registration/password_reset_done.html new file mode 100644 index 000000000..b798ee324 --- /dev/null +++ b/src/documents/templates/registration/password_reset_done.html @@ -0,0 +1,44 @@ + + +{% load static %} +{% load i18n %} + + + + + + + + + + {% translate "Paperless-ngx reset password sent" %} + + + + + + +
+ +

{% translate "Check your inbox." %}

+

{% translate "We've emailed you instructions for setting your password. You should receive the email shortly!" %}

+
+ + diff --git a/src/documents/templates/registration/password_reset_form.html b/src/documents/templates/registration/password_reset_form.html new file mode 100644 index 000000000..9bdc22943 --- /dev/null +++ b/src/documents/templates/registration/password_reset_form.html @@ -0,0 +1,58 @@ + + +{% load static %} +{% load i18n %} + + + + + + + + + + {% translate "Paperless-ngx reset password request" %} + + + + + + + + + diff --git a/src/locale/en_US/LC_MESSAGES/django.po b/src/locale/en_US/LC_MESSAGES/django.po index bb1fe876e..c1358185a 100644 --- a/src/locale/en_US/LC_MESSAGES/django.po +++ b/src/locale/en_US/LC_MESSAGES/django.po @@ -2,7 +2,7 @@ msgid "" msgstr "" "Project-Id-Version: paperless-ngx\n" "Report-Msgid-Bugs-To: \n" -"POT-Creation-Date: 2023-09-19 14:30-0700\n" +"POT-Creation-Date: 2023-09-28 10:56-0700\n" "PO-Revision-Date: 2022-02-17 04:17\n" "Last-Translator: \n" "Language-Team: English\n" @@ -21,425 +21,425 @@ msgstr "" msgid "Documents" msgstr "" -#: documents/models.py:36 documents/models.py:731 +#: documents/models.py:33 documents/models.py:731 msgid "owner" msgstr "" -#: documents/models.py:53 +#: documents/models.py:50 msgid "None" msgstr "" -#: documents/models.py:54 +#: documents/models.py:51 msgid "Any word" msgstr "" -#: documents/models.py:55 +#: documents/models.py:52 msgid "All words" msgstr "" -#: documents/models.py:56 +#: documents/models.py:53 msgid "Exact match" msgstr "" -#: documents/models.py:57 +#: documents/models.py:54 msgid "Regular expression" msgstr "" -#: documents/models.py:58 +#: documents/models.py:55 msgid "Fuzzy word" msgstr "" -#: documents/models.py:59 +#: documents/models.py:56 msgid "Automatic" msgstr "" -#: documents/models.py:62 documents/models.py:402 documents/models.py:755 +#: documents/models.py:59 documents/models.py:399 documents/models.py:749 #: paperless_mail/models.py:18 paperless_mail/models.py:93 msgid "name" msgstr "" -#: documents/models.py:64 +#: documents/models.py:61 msgid "match" msgstr "" -#: documents/models.py:67 +#: documents/models.py:64 msgid "matching algorithm" msgstr "" -#: documents/models.py:72 +#: documents/models.py:69 msgid "is insensitive" msgstr "" -#: documents/models.py:95 documents/models.py:147 +#: documents/models.py:92 documents/models.py:144 msgid "correspondent" msgstr "" -#: documents/models.py:96 +#: documents/models.py:93 msgid "correspondents" msgstr "" -#: documents/models.py:100 +#: documents/models.py:97 msgid "color" msgstr "" -#: documents/models.py:103 +#: documents/models.py:100 msgid "is inbox tag" msgstr "" -#: documents/models.py:106 +#: documents/models.py:103 msgid "" "Marks this tag as an inbox tag: All newly consumed documents will be tagged " "with inbox tags." msgstr "" -#: documents/models.py:112 +#: documents/models.py:109 msgid "tag" msgstr "" -#: documents/models.py:113 documents/models.py:185 +#: documents/models.py:110 documents/models.py:182 msgid "tags" msgstr "" -#: documents/models.py:118 documents/models.py:167 +#: documents/models.py:115 documents/models.py:164 msgid "document type" msgstr "" -#: documents/models.py:119 +#: documents/models.py:116 msgid "document types" msgstr "" -#: documents/models.py:124 +#: documents/models.py:121 msgid "path" msgstr "" -#: documents/models.py:129 documents/models.py:156 +#: documents/models.py:126 documents/models.py:153 msgid "storage path" msgstr "" -#: documents/models.py:130 +#: documents/models.py:127 msgid "storage paths" msgstr "" -#: documents/models.py:137 +#: documents/models.py:134 msgid "Unencrypted" msgstr "" -#: documents/models.py:138 +#: documents/models.py:135 msgid "Encrypted with GNU Privacy Guard" msgstr "" -#: documents/models.py:159 +#: documents/models.py:156 msgid "title" msgstr "" -#: documents/models.py:171 documents/models.py:645 +#: documents/models.py:168 documents/models.py:645 msgid "content" msgstr "" -#: documents/models.py:174 +#: documents/models.py:171 msgid "" "The raw, text-only data of the document. This field is primarily used for " "searching." msgstr "" -#: documents/models.py:179 +#: documents/models.py:176 msgid "mime type" msgstr "" -#: documents/models.py:189 +#: documents/models.py:186 msgid "checksum" msgstr "" -#: documents/models.py:193 +#: documents/models.py:190 msgid "The checksum of the original document." msgstr "" -#: documents/models.py:197 +#: documents/models.py:194 msgid "archive checksum" msgstr "" -#: documents/models.py:202 +#: documents/models.py:199 msgid "The checksum of the archived document." msgstr "" -#: documents/models.py:205 documents/models.py:385 documents/models.py:651 +#: documents/models.py:202 documents/models.py:382 documents/models.py:651 #: documents/models.py:689 msgid "created" msgstr "" -#: documents/models.py:208 +#: documents/models.py:205 msgid "modified" msgstr "" -#: documents/models.py:215 +#: documents/models.py:212 msgid "storage type" msgstr "" -#: documents/models.py:223 +#: documents/models.py:220 msgid "added" msgstr "" -#: documents/models.py:230 +#: documents/models.py:227 msgid "filename" msgstr "" -#: documents/models.py:236 +#: documents/models.py:233 msgid "Current filename in storage" msgstr "" -#: documents/models.py:240 +#: documents/models.py:237 msgid "archive filename" msgstr "" -#: documents/models.py:246 +#: documents/models.py:243 msgid "Current archive filename in storage" msgstr "" -#: documents/models.py:250 +#: documents/models.py:247 msgid "original filename" msgstr "" -#: documents/models.py:256 +#: documents/models.py:253 msgid "The original name of the file when it was uploaded" msgstr "" -#: documents/models.py:263 +#: documents/models.py:260 msgid "archive serial number" msgstr "" -#: documents/models.py:273 +#: documents/models.py:270 msgid "The position of this document in your physical document archive." msgstr "" -#: documents/models.py:279 documents/models.py:662 documents/models.py:716 +#: documents/models.py:276 documents/models.py:662 documents/models.py:716 msgid "document" msgstr "" -#: documents/models.py:280 +#: documents/models.py:277 msgid "documents" msgstr "" -#: documents/models.py:368 +#: documents/models.py:365 msgid "debug" msgstr "" -#: documents/models.py:369 +#: documents/models.py:366 msgid "information" msgstr "" -#: documents/models.py:370 +#: documents/models.py:367 msgid "warning" msgstr "" -#: documents/models.py:371 paperless_mail/models.py:293 +#: documents/models.py:368 paperless_mail/models.py:293 msgid "error" msgstr "" -#: documents/models.py:372 +#: documents/models.py:369 msgid "critical" msgstr "" -#: documents/models.py:375 +#: documents/models.py:372 msgid "group" msgstr "" -#: documents/models.py:377 +#: documents/models.py:374 msgid "message" msgstr "" -#: documents/models.py:380 +#: documents/models.py:377 msgid "level" msgstr "" -#: documents/models.py:389 +#: documents/models.py:386 msgid "log" msgstr "" -#: documents/models.py:390 +#: documents/models.py:387 msgid "logs" msgstr "" -#: documents/models.py:399 documents/models.py:464 +#: documents/models.py:396 documents/models.py:461 msgid "saved view" msgstr "" -#: documents/models.py:400 +#: documents/models.py:397 msgid "saved views" msgstr "" -#: documents/models.py:405 +#: documents/models.py:402 msgid "show on dashboard" msgstr "" -#: documents/models.py:408 +#: documents/models.py:405 msgid "show in sidebar" msgstr "" -#: documents/models.py:412 +#: documents/models.py:409 msgid "sort field" msgstr "" -#: documents/models.py:417 +#: documents/models.py:414 msgid "sort reverse" msgstr "" -#: documents/models.py:422 +#: documents/models.py:419 msgid "title contains" msgstr "" -#: documents/models.py:423 +#: documents/models.py:420 msgid "content contains" msgstr "" -#: documents/models.py:424 +#: documents/models.py:421 msgid "ASN is" msgstr "" -#: documents/models.py:425 +#: documents/models.py:422 msgid "correspondent is" msgstr "" -#: documents/models.py:426 +#: documents/models.py:423 msgid "document type is" msgstr "" -#: documents/models.py:427 +#: documents/models.py:424 msgid "is in inbox" msgstr "" -#: documents/models.py:428 +#: documents/models.py:425 msgid "has tag" msgstr "" -#: documents/models.py:429 +#: documents/models.py:426 msgid "has any tag" msgstr "" -#: documents/models.py:430 +#: documents/models.py:427 msgid "created before" msgstr "" -#: documents/models.py:431 +#: documents/models.py:428 msgid "created after" msgstr "" -#: documents/models.py:432 +#: documents/models.py:429 msgid "created year is" msgstr "" -#: documents/models.py:433 +#: documents/models.py:430 msgid "created month is" msgstr "" -#: documents/models.py:434 +#: documents/models.py:431 msgid "created day is" msgstr "" -#: documents/models.py:435 +#: documents/models.py:432 msgid "added before" msgstr "" -#: documents/models.py:436 +#: documents/models.py:433 msgid "added after" msgstr "" -#: documents/models.py:437 +#: documents/models.py:434 msgid "modified before" msgstr "" -#: documents/models.py:438 +#: documents/models.py:435 msgid "modified after" msgstr "" -#: documents/models.py:439 +#: documents/models.py:436 msgid "does not have tag" msgstr "" -#: documents/models.py:440 +#: documents/models.py:437 msgid "does not have ASN" msgstr "" -#: documents/models.py:441 +#: documents/models.py:438 msgid "title or content contains" msgstr "" -#: documents/models.py:442 +#: documents/models.py:439 msgid "fulltext query" msgstr "" -#: documents/models.py:443 +#: documents/models.py:440 msgid "more like this" msgstr "" -#: documents/models.py:444 +#: documents/models.py:441 msgid "has tags in" msgstr "" -#: documents/models.py:445 +#: documents/models.py:442 msgid "ASN greater than" msgstr "" -#: documents/models.py:446 +#: documents/models.py:443 msgid "ASN less than" msgstr "" -#: documents/models.py:447 +#: documents/models.py:444 msgid "storage path is" msgstr "" -#: documents/models.py:448 +#: documents/models.py:445 msgid "has correspondent in" msgstr "" -#: documents/models.py:449 +#: documents/models.py:446 msgid "does not have correspondent in" msgstr "" -#: documents/models.py:450 +#: documents/models.py:447 msgid "has document type in" msgstr "" -#: documents/models.py:451 +#: documents/models.py:448 msgid "does not have document type in" msgstr "" -#: documents/models.py:452 +#: documents/models.py:449 msgid "has storage path in" msgstr "" -#: documents/models.py:453 +#: documents/models.py:450 msgid "does not have storage path in" msgstr "" -#: documents/models.py:454 +#: documents/models.py:451 msgid "owner is" msgstr "" -#: documents/models.py:455 +#: documents/models.py:452 msgid "has owner in" msgstr "" -#: documents/models.py:456 +#: documents/models.py:453 msgid "does not have owner" msgstr "" -#: documents/models.py:457 +#: documents/models.py:454 msgid "does not have owner in" msgstr "" -#: documents/models.py:467 +#: documents/models.py:464 msgid "rule type" msgstr "" -#: documents/models.py:469 +#: documents/models.py:466 msgid "value" msgstr "" -#: documents/models.py:472 +#: documents/models.py:469 msgid "filter rule" msgstr "" -#: documents/models.py:473 +#: documents/models.py:470 msgid "filter rules" msgstr "" @@ -555,114 +555,114 @@ msgstr "" msgid "share links" msgstr "" -#: documents/models.py:744 +#: documents/models.py:745 msgid "Consume Folder" msgstr "" -#: documents/models.py:745 +#: documents/models.py:746 msgid "Api Upload" msgstr "" -#: documents/models.py:746 +#: documents/models.py:747 msgid "Mail Fetch" msgstr "" -#: documents/models.py:752 -msgid "consumption template" -msgstr "" - -#: documents/models.py:753 -msgid "consumption templates" -msgstr "" - -#: documents/models.py:757 paperless_mail/models.py:95 +#: documents/models.py:751 paperless_mail/models.py:95 msgid "order" msgstr "" -#: documents/models.py:766 +#: documents/models.py:760 msgid "filter path" msgstr "" -#: documents/models.py:771 +#: documents/models.py:765 msgid "" "Only consume documents with a path that matches this if specified. Wildcards " "specified as * are allowed. Case insensitive." msgstr "" -#: documents/models.py:778 +#: documents/models.py:772 msgid "filter filename" msgstr "" -#: documents/models.py:783 paperless_mail/models.py:148 +#: documents/models.py:777 paperless_mail/models.py:148 msgid "" "Only consume documents which entirely match this filename if specified. " "Wildcards such as *.pdf or *invoice* are allowed. Case insensitive." msgstr "" -#: documents/models.py:794 +#: documents/models.py:788 msgid "filter documents from this mail rule" msgstr "" -#: documents/models.py:798 +#: documents/models.py:792 msgid "assign title" msgstr "" -#: documents/models.py:803 +#: documents/models.py:797 msgid "" "Assign a document title, can include some placeholders, see documentation." msgstr "" -#: documents/models.py:811 paperless_mail/models.py:204 +#: documents/models.py:805 paperless_mail/models.py:204 msgid "assign this tag" msgstr "" -#: documents/models.py:819 paperless_mail/models.py:212 +#: documents/models.py:813 paperless_mail/models.py:212 msgid "assign this document type" msgstr "" -#: documents/models.py:827 paperless_mail/models.py:226 +#: documents/models.py:821 paperless_mail/models.py:226 msgid "assign this correspondent" msgstr "" -#: documents/models.py:835 +#: documents/models.py:829 msgid "assign this storage path" msgstr "" -#: documents/models.py:844 +#: documents/models.py:838 msgid "assign this owner" msgstr "" -#: documents/models.py:851 +#: documents/models.py:845 msgid "grant view permissions to these users" msgstr "" -#: documents/models.py:858 +#: documents/models.py:852 msgid "grant view permissions to these groups" msgstr "" -#: documents/models.py:865 +#: documents/models.py:859 msgid "grant change permissions to these users" msgstr "" -#: documents/models.py:872 +#: documents/models.py:866 msgid "grant change permissions to these groups" msgstr "" -#: documents/serialisers.py:100 +#: documents/models.py:870 +msgid "consumption template" +msgstr "" + +#: documents/models.py:871 +msgid "consumption templates" +msgstr "" + +#: documents/serialisers.py:98 #, python-format msgid "Invalid regular expression: %(error)s" msgstr "" -#: documents/serialisers.py:375 +#: documents/serialisers.py:373 msgid "Invalid color." msgstr "" -#: documents/serialisers.py:751 +#: documents/serialisers.py:749 #, python-format msgid "File type %(type)s not supported" msgstr "" -#: documents/serialisers.py:848 +#: documents/serialisers.py:846 msgid "Invalid variable detected." msgstr "" @@ -722,123 +722,206 @@ msgstr "" msgid "Sign in" msgstr "" +#: documents/templates/registration/login.html:70 +msgid "Forgot your password?" +msgstr "" + +#: documents/templates/registration/password_reset_complete.html:14 +msgid "Paperless-ngx reset password complete" +msgstr "" + +#: documents/templates/registration/password_reset_complete.html:40 +msgid "Password reset complete." +msgstr "" + +#: documents/templates/registration/password_reset_complete.html:42 +#, python-format +msgid "" +"Your new password has been set. You can now log " +"in" +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:14 +msgid "Paperless-ngx reset password confirmation" +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:42 +msgid "Set a new password." +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:46 +msgid "Passwords did not match or too weak. Try again." +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:49 +msgid "New Password" +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:50 +msgid "Confirm Password" +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:61 +msgid "Change my password" +msgstr "" + +#: documents/templates/registration/password_reset_confirm.html:65 +msgid "request a new password reset" +msgstr "" + +#: documents/templates/registration/password_reset_done.html:14 +msgid "Paperless-ngx reset password sent" +msgstr "" + +#: documents/templates/registration/password_reset_done.html:40 +msgid "Check your inbox." +msgstr "" + +#: documents/templates/registration/password_reset_done.html:41 +msgid "" +"We've emailed you instructions for setting your password. You should receive " +"the email shortly!" +msgstr "" + +#: documents/templates/registration/password_reset_form.html:14 +msgid "Paperless-ngx reset password request" +msgstr "" + +#: documents/templates/registration/password_reset_form.html:41 +msgid "" +"Enter your email address below, and we'll email instructions for setting a " +"new one." +msgstr "" + +#: documents/templates/registration/password_reset_form.html:44 +msgid "An error occurred. Please try again." +msgstr "" + +#: documents/templates/registration/password_reset_form.html:47 +msgid "Email" +msgstr "" + +#: documents/templates/registration/password_reset_form.html:54 +msgid "Send me instructions!" +msgstr "" + #: paperless/apps.py:10 msgid "Paperless" msgstr "" -#: paperless/settings.py:573 +#: paperless/settings.py:574 msgid "English (US)" msgstr "" -#: paperless/settings.py:574 +#: paperless/settings.py:575 msgid "Arabic" msgstr "" -#: paperless/settings.py:575 +#: paperless/settings.py:576 msgid "Afrikaans" msgstr "" -#: paperless/settings.py:576 +#: paperless/settings.py:577 msgid "Belarusian" msgstr "" -#: paperless/settings.py:577 +#: paperless/settings.py:578 msgid "Catalan" msgstr "" -#: paperless/settings.py:578 +#: paperless/settings.py:579 msgid "Czech" msgstr "" -#: paperless/settings.py:579 +#: paperless/settings.py:580 msgid "Danish" msgstr "" -#: paperless/settings.py:580 +#: paperless/settings.py:581 msgid "German" msgstr "" -#: paperless/settings.py:581 +#: paperless/settings.py:582 msgid "Greek" msgstr "" -#: paperless/settings.py:582 +#: paperless/settings.py:583 msgid "English (GB)" msgstr "" -#: paperless/settings.py:583 +#: paperless/settings.py:584 msgid "Spanish" msgstr "" -#: paperless/settings.py:584 +#: paperless/settings.py:585 msgid "Finnish" msgstr "" -#: paperless/settings.py:585 +#: paperless/settings.py:586 msgid "French" msgstr "" -#: paperless/settings.py:586 +#: paperless/settings.py:587 msgid "Italian" msgstr "" -#: paperless/settings.py:587 +#: paperless/settings.py:588 msgid "Luxembourgish" msgstr "" -#: paperless/settings.py:588 +#: paperless/settings.py:589 msgid "Norwegian" msgstr "" -#: paperless/settings.py:589 +#: paperless/settings.py:590 msgid "Dutch" msgstr "" -#: paperless/settings.py:590 +#: paperless/settings.py:591 msgid "Polish" msgstr "" -#: paperless/settings.py:591 +#: paperless/settings.py:592 msgid "Portuguese (Brazil)" msgstr "" -#: paperless/settings.py:592 +#: paperless/settings.py:593 msgid "Portuguese" msgstr "" -#: paperless/settings.py:593 +#: paperless/settings.py:594 msgid "Romanian" msgstr "" -#: paperless/settings.py:594 +#: paperless/settings.py:595 msgid "Russian" msgstr "" -#: paperless/settings.py:595 +#: paperless/settings.py:596 msgid "Slovak" msgstr "" -#: paperless/settings.py:596 +#: paperless/settings.py:597 msgid "Slovenian" msgstr "" -#: paperless/settings.py:597 +#: paperless/settings.py:598 msgid "Serbian" msgstr "" -#: paperless/settings.py:598 +#: paperless/settings.py:599 msgid "Swedish" msgstr "" -#: paperless/settings.py:599 +#: paperless/settings.py:600 msgid "Turkish" msgstr "" -#: paperless/settings.py:600 +#: paperless/settings.py:601 msgid "Ukrainian" msgstr "" -#: paperless/settings.py:601 +#: paperless/settings.py:602 msgid "Chinese Simplified" msgstr "" diff --git a/src/paperless/settings.py b/src/paperless/settings.py index 6d25a53cc..f3cbb6ee2 100644 --- a/src/paperless/settings.py +++ b/src/paperless/settings.py @@ -357,6 +357,7 @@ TEMPLATES = [ "django.template.context_processors.request", "django.contrib.auth.context_processors.auth", "django.contrib.messages.context_processors.messages", + "documents.context_processors.settings", ], }, }, @@ -1002,3 +1003,15 @@ def _get_nltk_language_setting(ocr_lang: str) -> Optional[str]: NLTK_ENABLED: Final[bool] = __get_boolean("PAPERLESS_ENABLE_NLTK", "yes") NLTK_LANGUAGE: Optional[str] = _get_nltk_language_setting(OCR_LANGUAGE) + +############################################################################### +# Email (SMTP) Backend # +############################################################################### + +EMAIL_HOST: Final[str] = os.getenv("PAPERLESS_EMAIL_HOST", "localhost") +EMAIL_PORT: Final[int] = int(os.getenv("PAPERLESS_EMAIL_PORT", 25)) +EMAIL_HOST_USER: Final[str] = os.getenv("PAPERLESS_EMAIL_HOST_USER", "") +EMAIL_HOST_PASSWORD: Final[str] = os.getenv("PAPERLESS_EMAIL_HOST_PASSWORD", "") +EMAIL_USE_TLS: Final[bool] = __get_boolean("PAPERLESS_EMAIL_USE_TLS") +EMAIL_USE_SSL: Final[bool] = __get_boolean("PAPERLESS_EMAIL_USE_SSL") +EMAIL_SUBJECT_PREFIX: Final[str] = "[Paperless-ngx] "