mirror of
https://github.com/paperless-ngx/paperless-ngx.git
synced 2025-10-23 23:09:08 -04:00
Merge remote-tracking branch 'origin/dev'
This commit is contained in:
commit
c2bede40c7
@ -704,6 +704,20 @@ but could result in missing text content.
|
|||||||
this value if you are certain your documents are not malicious and
|
this value if you are certain your documents are not malicious and
|
||||||
you need the text which was not OCRed
|
you need the text which was not OCRed
|
||||||
|
|
||||||
|
#### [`PAPERLESS_OCR_COLOR_CONVERSION_STRATEGY=<RGB>`](#PAPERLESS_OCR_COLOR_CONVERSION_STRATEGY) {#PAPERLESS_OCR_COLOR_CONVERSION_STRATEGY}
|
||||||
|
|
||||||
|
: Controls the Ghostscript color conversion strategy when creating the archive file. This setting
|
||||||
|
will only be utilized if the output is a version of PDF/A.
|
||||||
|
|
||||||
|
Valid options are CMYK, Gray, LeaveColorUnchanged, RGB or UseDeviceIndependentColor.
|
||||||
|
|
||||||
|
You can find more on the settings [here](https://ghostscript.readthedocs.io/en/latest/VectorDevices.html#color-conversion-and-management) in the Ghostscript documentation.
|
||||||
|
|
||||||
|
!!! warning
|
||||||
|
|
||||||
|
Utilizing some of the options may result in errors when creating archive
|
||||||
|
files from PDFs.
|
||||||
|
|
||||||
#### [`PAPERLESS_OCR_USER_ARGS=<json>`](#PAPERLESS_OCR_USER_ARGS) {#PAPERLESS_OCR_USER_ARGS}
|
#### [`PAPERLESS_OCR_USER_ARGS=<json>`](#PAPERLESS_OCR_USER_ARGS) {#PAPERLESS_OCR_USER_ARGS}
|
||||||
|
|
||||||
: OCRmyPDF offers many more options. Use this parameter to specify any
|
: OCRmyPDF offers many more options. Use this parameter to specify any
|
||||||
|
@ -192,4 +192,24 @@ describe('ShareLinksDropdownComponent', () => {
|
|||||||
component.share(link)
|
component.share(link)
|
||||||
// expect(navigatorSpy).toHaveBeenCalledWith({ url: component.getShareUrl(link) })
|
// expect(navigatorSpy).toHaveBeenCalledWith({ url: component.getShareUrl(link) })
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should correctly generate share URLs', () => {
|
||||||
|
environment.apiBaseUrl = 'http://example.com/api/'
|
||||||
|
expect(component.getShareUrl({ slug: '123abc123' } as any)).toEqual(
|
||||||
|
'http://example.com/share/123abc123'
|
||||||
|
)
|
||||||
|
environment.apiBaseUrl = 'http://example.domainwithapiinit.com/api/'
|
||||||
|
expect(component.getShareUrl({ slug: '123abc123' } as any)).toEqual(
|
||||||
|
'http://example.domainwithapiinit.com/share/123abc123'
|
||||||
|
)
|
||||||
|
environment.apiBaseUrl = 'http://example.domainwithapiinit.com:1234/api/'
|
||||||
|
expect(component.getShareUrl({ slug: '123abc123' } as any)).toEqual(
|
||||||
|
'http://example.domainwithapiinit.com:1234/share/123abc123'
|
||||||
|
)
|
||||||
|
environment.apiBaseUrl =
|
||||||
|
'http://example.domainwithapiinit.com:1234/subpath/api/'
|
||||||
|
expect(component.getShareUrl({ slug: '123abc123' } as any)).toEqual(
|
||||||
|
'http://example.domainwithapiinit.com:1234/subpath/share/123abc123'
|
||||||
|
)
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
@ -80,7 +80,10 @@ export class ShareLinksDropdownComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
|
|
||||||
getShareUrl(link: PaperlessShareLink): string {
|
getShareUrl(link: PaperlessShareLink): string {
|
||||||
return `${environment.apiBaseUrl.replace('api', 'share')}${link.slug}`
|
const apiURL = new URL(environment.apiBaseUrl)
|
||||||
|
return `${apiURL.origin}${apiURL.pathname.replace(/\/api\/$/, '/share/')}${
|
||||||
|
link.slug
|
||||||
|
}`
|
||||||
}
|
}
|
||||||
|
|
||||||
getDaysRemaining(link: PaperlessShareLink): string {
|
getDaysRemaining(link: PaperlessShareLink): string {
|
||||||
|
@ -5,7 +5,7 @@ export const environment = {
|
|||||||
apiBaseUrl: document.baseURI + 'api/',
|
apiBaseUrl: document.baseURI + 'api/',
|
||||||
apiVersion: '3',
|
apiVersion: '3',
|
||||||
appTitle: 'Paperless-ngx',
|
appTitle: 'Paperless-ngx',
|
||||||
version: '2.0.0',
|
version: '2.0.0-dev',
|
||||||
webSocketHost: window.location.host,
|
webSocketHost: window.location.host,
|
||||||
webSocketProtocol: window.location.protocol == 'https:' ? 'wss:' : 'ws:',
|
webSocketProtocol: window.location.protocol == 'https:' ? 'wss:' : 'ws:',
|
||||||
webSocketBaseUrl: base_url.pathname + 'ws/',
|
webSocketBaseUrl: base_url.pathname + 'ws/',
|
||||||
|
@ -3019,7 +3019,7 @@
|
|||||||
<context context-type="sourcefile">src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts</context>
|
<context context-type="sourcefile">src/app/components/common/edit-dialog/custom-field-edit-dialog/custom-field-edit-dialog.component.ts</context>
|
||||||
<context context-type="linenumber">39</context>
|
<context context-type="linenumber">39</context>
|
||||||
</context-group>
|
</context-group>
|
||||||
<target state="translated">Neues benutzerdefinidertes Feld erstellen</target>
|
<target state="translated">Neues benutzerdefiniertes Feld erstellen</target>
|
||||||
</trans-unit>
|
</trans-unit>
|
||||||
<trans-unit id="8751213029607178010" datatype="html">
|
<trans-unit id="8751213029607178010" datatype="html">
|
||||||
<source>Edit custom field</source>
|
<source>Edit custom field</source>
|
||||||
|
@ -0,0 +1,22 @@
|
|||||||
|
# Generated by Django 4.2.7 on 2023-11-30 14:29
|
||||||
|
|
||||||
|
import multiselectfield.db.fields
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
dependencies = [
|
||||||
|
("documents", "1040_customfield_customfieldinstance_and_more"),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name="consumptiontemplate",
|
||||||
|
name="sources",
|
||||||
|
field=multiselectfield.db.fields.MultiSelectField(
|
||||||
|
choices=[(1, "Consume Folder"), (2, "Api Upload"), (3, "Mail Fetch")],
|
||||||
|
default="1,2,3",
|
||||||
|
max_length=5,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
]
|
@ -754,7 +754,7 @@ class ConsumptionTemplate(models.Model):
|
|||||||
order = models.IntegerField(_("order"), default=0)
|
order = models.IntegerField(_("order"), default=0)
|
||||||
|
|
||||||
sources = MultiSelectField(
|
sources = MultiSelectField(
|
||||||
max_length=3,
|
max_length=5,
|
||||||
choices=DocumentSourceChoices.choices,
|
choices=DocumentSourceChoices.choices,
|
||||||
default=f"{DocumentSource.ConsumeFolder},{DocumentSource.ApiUpload},{DocumentSource.MailFetch}",
|
default=f"{DocumentSource.ConsumeFolder},{DocumentSource.ApiUpload},{DocumentSource.MailFetch}",
|
||||||
)
|
)
|
||||||
|
@ -3,7 +3,7 @@ msgstr ""
|
|||||||
"Project-Id-Version: paperless-ngx\n"
|
"Project-Id-Version: paperless-ngx\n"
|
||||||
"Report-Msgid-Bugs-To: \n"
|
"Report-Msgid-Bugs-To: \n"
|
||||||
"POT-Creation-Date: 2023-11-10 09:02-0800\n"
|
"POT-Creation-Date: 2023-11-10 09:02-0800\n"
|
||||||
"PO-Revision-Date: 2023-11-13 01:31\n"
|
"PO-Revision-Date: 2023-11-29 08:50\n"
|
||||||
"Last-Translator: \n"
|
"Last-Translator: \n"
|
||||||
"Language-Team: Russian\n"
|
"Language-Team: Russian\n"
|
||||||
"Language: ru_RU\n"
|
"Language: ru_RU\n"
|
||||||
@ -409,7 +409,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: documents/models.py:454
|
#: documents/models.py:454
|
||||||
msgid "owner is"
|
msgid "owner is"
|
||||||
msgstr ""
|
msgstr "владелец"
|
||||||
|
|
||||||
#: documents/models.py:455
|
#: documents/models.py:455
|
||||||
msgid "has owner in"
|
msgid "has owner in"
|
||||||
@ -417,7 +417,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: documents/models.py:456
|
#: documents/models.py:456
|
||||||
msgid "does not have owner"
|
msgid "does not have owner"
|
||||||
msgstr ""
|
msgstr "не имеет владельца"
|
||||||
|
|
||||||
#: documents/models.py:457
|
#: documents/models.py:457
|
||||||
msgid "does not have owner in"
|
msgid "does not have owner in"
|
||||||
@ -521,7 +521,7 @@ msgstr "пользователь"
|
|||||||
|
|
||||||
#: documents/models.py:679
|
#: documents/models.py:679
|
||||||
msgid "note"
|
msgid "note"
|
||||||
msgstr ""
|
msgstr "заметка"
|
||||||
|
|
||||||
#: documents/models.py:680
|
#: documents/models.py:680
|
||||||
msgid "notes"
|
msgid "notes"
|
||||||
@ -529,15 +529,15 @@ msgstr "заметки"
|
|||||||
|
|
||||||
#: documents/models.py:688
|
#: documents/models.py:688
|
||||||
msgid "Archive"
|
msgid "Archive"
|
||||||
msgstr ""
|
msgstr "Архив"
|
||||||
|
|
||||||
#: documents/models.py:689
|
#: documents/models.py:689
|
||||||
msgid "Original"
|
msgid "Original"
|
||||||
msgstr ""
|
msgstr "Оригинал"
|
||||||
|
|
||||||
#: documents/models.py:700
|
#: documents/models.py:700
|
||||||
msgid "expiration"
|
msgid "expiration"
|
||||||
msgstr ""
|
msgstr "Срок действия"
|
||||||
|
|
||||||
#: documents/models.py:707
|
#: documents/models.py:707
|
||||||
msgid "slug"
|
msgid "slug"
|
||||||
@ -645,7 +645,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: documents/models.py:887
|
#: documents/models.py:887
|
||||||
msgid "URL"
|
msgid "URL"
|
||||||
msgstr ""
|
msgstr "Ссылка"
|
||||||
|
|
||||||
#: documents/models.py:888
|
#: documents/models.py:888
|
||||||
msgid "Date"
|
msgid "Date"
|
||||||
@ -661,7 +661,7 @@ msgstr ""
|
|||||||
|
|
||||||
#: documents/models.py:891
|
#: documents/models.py:891
|
||||||
msgid "Float"
|
msgid "Float"
|
||||||
msgstr ""
|
msgstr "С плавающей точкой"
|
||||||
|
|
||||||
#: documents/models.py:892
|
#: documents/models.py:892
|
||||||
msgid "Monetary"
|
msgid "Monetary"
|
||||||
@ -669,15 +669,15 @@ msgstr ""
|
|||||||
|
|
||||||
#: documents/models.py:904
|
#: documents/models.py:904
|
||||||
msgid "data type"
|
msgid "data type"
|
||||||
msgstr ""
|
msgstr "Тип данных"
|
||||||
|
|
||||||
#: documents/models.py:912
|
#: documents/models.py:912
|
||||||
msgid "custom field"
|
msgid "custom field"
|
||||||
msgstr ""
|
msgstr "Пользовательское поле"
|
||||||
|
|
||||||
#: documents/models.py:913
|
#: documents/models.py:913
|
||||||
msgid "custom fields"
|
msgid "custom fields"
|
||||||
msgstr ""
|
msgstr "Пользовательские поля"
|
||||||
|
|
||||||
#: documents/models.py:973
|
#: documents/models.py:973
|
||||||
msgid "custom field instance"
|
msgid "custom field instance"
|
||||||
|
@ -864,6 +864,11 @@ OCR_MAX_IMAGE_PIXELS: Optional[int] = None
|
|||||||
if os.environ.get("PAPERLESS_OCR_MAX_IMAGE_PIXELS") is not None:
|
if os.environ.get("PAPERLESS_OCR_MAX_IMAGE_PIXELS") is not None:
|
||||||
OCR_MAX_IMAGE_PIXELS: int = int(os.environ.get("PAPERLESS_OCR_MAX_IMAGE_PIXELS"))
|
OCR_MAX_IMAGE_PIXELS: int = int(os.environ.get("PAPERLESS_OCR_MAX_IMAGE_PIXELS"))
|
||||||
|
|
||||||
|
OCR_COLOR_CONVERSION_STRATEGY = os.getenv(
|
||||||
|
"PAPERLESS_OCR_COLOR_CONVERSION_STRATEGY",
|
||||||
|
"RGB",
|
||||||
|
)
|
||||||
|
|
||||||
OCR_USER_ARGS = os.getenv("PAPERLESS_OCR_USER_ARGS", "{}")
|
OCR_USER_ARGS = os.getenv("PAPERLESS_OCR_USER_ARGS", "{}")
|
||||||
|
|
||||||
# GNUPG needs a home directory for some reason
|
# GNUPG needs a home directory for some reason
|
||||||
|
@ -186,6 +186,11 @@ class RasterisedDocumentParser(DocumentParser):
|
|||||||
"progress_bar": False,
|
"progress_bar": False,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if "pdfa" in ocrmypdf_args["output_type"]:
|
||||||
|
ocrmypdf_args[
|
||||||
|
"color_conversion_strategy"
|
||||||
|
] = settings.OCR_COLOR_CONVERSION_STRATEGY
|
||||||
|
|
||||||
if settings.OCR_MODE == "force" or safe_fallback:
|
if settings.OCR_MODE == "force" or safe_fallback:
|
||||||
ocrmypdf_args["force_ocr"] = True
|
ocrmypdf_args["force_ocr"] = True
|
||||||
elif settings.OCR_MODE in ["skip", "skip_noarchive"]:
|
elif settings.OCR_MODE in ["skip", "skip_noarchive"]:
|
||||||
@ -244,6 +249,10 @@ class RasterisedDocumentParser(DocumentParser):
|
|||||||
f"no DPI information is present in this image and "
|
f"no DPI information is present in this image and "
|
||||||
f"OCR_IMAGE_DPI is not set.",
|
f"OCR_IMAGE_DPI is not set.",
|
||||||
)
|
)
|
||||||
|
if ocrmypdf_args["image_dpi"] < 70: # pragma: no cover
|
||||||
|
self.log.warning(
|
||||||
|
f"Image DPI of {ocrmypdf_args['image_dpi']} is low, OCR may fail",
|
||||||
|
)
|
||||||
|
|
||||||
if settings.OCR_USER_ARGS and not safe_fallback:
|
if settings.OCR_USER_ARGS and not safe_fallback:
|
||||||
try:
|
try:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user