diff --git a/README.md b/README.md
index 9ae9ba9d..41197b3c 100644
--- a/README.md
+++ b/README.md
@@ -1,5 +1,5 @@
-
Stirling-PDF
-
+
+Stirling-PDF
[![Docker Pulls](https://img.shields.io/docker/pulls/frooodle/s-pdf)](https://hub.docker.com/r/frooodle/s-pdf)
[![Discord](https://img.shields.io/discord/1068636748814483718?label=Discord)](https://discord.gg/Cn8pWhQRxZ)
@@ -19,16 +19,17 @@ All files and PDFs exist either exclusively on the client side, reside in server
![stirling-home](images/stirling-home.jpg)
## Features
+
- Dark mode support.
- Custom download options (see [here](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/images/settings.png) for example)
- Parallel file processing and downloads
- API for integration with external scripts
- Optional Login and Authentication support (see [here](https://github.com/Stirling-Tools/Stirling-PDF/tree/main#login-authentication) for documentation)
-
## **PDF Features**
### **Page Operations**
+
- View and modify PDFs - View multi page PDFs with custom viewing sorting and searching. Plus on page edit features like annotate, draw and adding text and images. (Using PDF.js with Joxit and Liberation.Liberation fonts)
- Full interactive GUI for merging/splitting/rotating/moving PDFs and their pages.
- Merge multiple PDFs together into a single resultant file.
@@ -45,6 +46,7 @@ All files and PDFs exist either exclusively on the client side, reside in server
- Convert PDF to a single page.
### **Conversion Operations**
+
- Convert PDFs to and from images.
- Convert any common file to PDF (using LibreOffice).
- Convert PDF to Word/Powerpoint/Others (using LibreOffice).
@@ -53,6 +55,7 @@ All files and PDFs exist either exclusively on the client side, reside in server
- Markdown to PDF.
### **Security & Permissions**
+
- Add and remove passwords.
- Change/set PDF Permissions.
- Add watermark(s).
@@ -61,6 +64,7 @@ All files and PDFs exist either exclusively on the client side, reside in server
- Auto-redact text.
### **Other Operations**
+
- Add/Generate/Write signatures.
- Repair PDFs.
- Detect and remove blank pages.
@@ -77,11 +81,11 @@ All files and PDFs exist either exclusively on the client side, reside in server
- Flatten PDFs.
- Get all information on a PDF to view or export as JSON.
-
For a overview of the tasks and the technology each uses please view [Endpoint-groups.md](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/Endpoint-groups.md)
Demo of the app is available [here](https://stirlingpdf.io). username: demo, password: demo
## Technologies used
+
- Spring Boot + Thymeleaf
- [PDFBox](https://github.com/apache/pdfbox/tree/trunk)
- [LibreOffice](https://www.libreoffice.org/discover/libreoffice/) for advanced conversions
@@ -94,9 +98,11 @@ Demo of the app is available [here](https://stirlingpdf.io). username: demo, pas
## How to use
### Locally
+
Please view https://github.com/Stirling-Tools/Stirling-PDF/blob/main/LocalRunGuide.md
### Docker / Podman
+
https://hub.docker.com/r/frooodle/s-pdf
Stirling PDF has 2 different versions, a Full version and ultra-Lite version. Depending on the types of features you use you may want a smaller image to save on space.
@@ -106,6 +112,7 @@ For people that don't mind about space optimization just use the latest tag.
![Docker Image Size (tag)](https://img.shields.io/docker/image-size/frooodle/s-pdf/latest-ultra-lite?label=Stirling-PDF%20Ultra-Lite)
Docker Run
+
```bash
docker run -d \
-p 8080:8080 \
@@ -122,7 +129,9 @@ docker run -d \
-v /location/of/customFiles:/customFiles \
```
+
Docker Compose
+
```yaml
version: '3.3'
services:
@@ -143,54 +152,60 @@ services:
Note: Podman is CLI-compatible with Docker, so simply replace "docker" with "podman".
## Enable OCR/Compression feature
+
Please view https://github.com/Stirling-Tools/Stirling-PDF/blob/main/HowToUseOCR.md
## Supported Languages
Stirling PDF currently supports 26!
-- English (English) (en_GB)
-- English (US) (en_US)
-- Arabic (العربية) (ar_AR)
-- German (Deutsch) (de_DE)
-- French (Français) (fr_FR)
-- Spanish (Español) (es_ES)
-- Simplified Chinese (简体中文) (zh_CN)
-- Traditional Chinese (繁體中文) (zh_TW)
-- Catalan (Català) (ca_CA)
-- Italian (Italiano) (it_IT)
-- Swedish (Svenska) (sv_SE)
-- Polish (Polski) (pl_PL)
-- Romanian (Română) (ro_RO)
-- Korean (한국어) (ko_KR)
-- Portuguese Brazilian (Português) (pt_BR)
-- Russian (Русский) (ru_RU)
-- Basque (Euskara) (eu_ES)
-- Japanese (日本語) (ja_JP)
-- Dutch (Nederlands) (nl_NL)
-- Greek (el_GR)
-- Turkish (Türkçe) (tr_TR)
-- Indonesia (Bahasa Indonesia) (id_ID)
-- Hindi (हिंदी) (hi_IN)
-- Hungarian (Magyar) (hu_HU)
-- Bulgarian (Български) (bg_BG)
-- Sebian Latin alphabet (Srpski) (sr_LATN_RS)
+
+| Language | Progress |
+| ------------------------------------------- | -------------------------------------- |
+| English (English) (en_GB) | ![100%](https://geps.dev/progress/100) |
+| English (US) (en_US) | ![100%](https://geps.dev/progress/100) |
+| Arabic (العربية) (ar_AR) | ![58%](https://geps.dev/progress/58) |
+| German (Deutsch) (de_DE) | ![95%](https://geps.dev/progress/95) |
+| French (Français) (fr_FR) | ![99%](https://geps.dev/progress/99) |
+| Spanish (Español) (es_ES) | ![95%](https://geps.dev/progress/95) |
+| Simplified Chinese (简体中文) (zh_CN) | ![99%](https://geps.dev/progress/99) |
+| Traditional Chinese (繁體中文) (zh_TW) | ![97%](https://geps.dev/progress/97) |
+| Catalan (Català) (ca_CA) | ![65%](https://geps.dev/progress/65) |
+| Italian (Italiano) (it_IT) | ![99%](https://geps.dev/progress/99) |
+| Swedish (Svenska) (sv_SE) | ![58%](https://geps.dev/progress/58) |
+| Polish (Polski) (pl_PL) | ![60%](https://geps.dev/progress/60) |
+| Romanian (Română) (ro_RO) | ![58%](https://geps.dev/progress/58) |
+| Korean (한국어) (ko_KR) | ![92%](https://geps.dev/progress/92) |
+| Portuguese Brazilian (Português) (pt_BR) | ![74%](https://geps.dev/progress/74) |
+| Russian (Русский) (ru_RU) | ![92%](https://geps.dev/progress/92) |
+| Basque (Euskara) (eu_ES) | ![76%](https://geps.dev/progress/76) |
+| Japanese (日本語) (ja_JP) | ![92%](https://geps.dev/progress/92) |
+| Dutch (Nederlands) (nl_NL) | ![92%](https://geps.dev/progress/92) |
+| Greek (Ελληνικά) (el_GR) | ![77%](https://geps.dev/progress/77) |
+| Turkish (Türkçe) (tr_TR) | ![82%](https://geps.dev/progress/82) |
+| Indonesia (Bahasa Indonesia) (id_ID) | ![87%](https://geps.dev/progress/87) |
+| Hindi (हिंदी) (hi_IN) | ![88%](https://geps.dev/progress/88) |
+| Hungarian (Magyar) (hu_HU) | ![87%](https://geps.dev/progress/87) |
+| Bulgarian (Български) (bg_BG) | ![82%](https://geps.dev/progress/82) |
+| Sebian Latin alphabet (Srpski) (sr_LATN_RS) | ![89%](https://geps.dev/progress/89) |
## Contributing (creating issues, translations, fixing bugs, etc.)
Please see our [Contributing Guide](CONTRIBUTING.md)!
## Customisation
+
Stirling PDF allows easy customization of the app.
Includes things like
+
- Custom application name
- Custom slogans, icons, images, and even custom HTML (via file overrides)
-
There are two options for this, either using the generated settings file ``settings.yml``
This file is located in the ``/configs`` directory and follows standard YAML formatting
Environment variables are also supported and would override the settings file
For example in the settings.yml you have
+
```yaml
system:
defaultLocale: 'en-US'
@@ -199,6 +214,7 @@ system:
To have this via an environment variable you would have ``SYSTEM_DEFAULTLOCALE``
The Current list of settings is
+
```yaml
security:
enableLogin: false # set to 'true' to enable login
@@ -221,24 +237,31 @@ endpoints:
metrics:
enabled: true # 'true' to enable Info APIs endpoints (view http://localhost:8080/swagger-ui/index.html#/API to learn more), 'false' to disable
```
+
### Extra notes
+
- Endpoints. Currently, the endpoints ENDPOINTS_TO_REMOVE and GROUPS_TO_REMOVE can include comma separate lists of endpoints and groups to disable as example ENDPOINTS_TO_REMOVE=img-to-pdf,remove-pages would disable both image-to-pdf and remove pages, GROUPS_TO_REMOVE=LibreOffice Would disable all things that use LibreOffice. You can see a list of all endpoints and groups [here](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/Endpoint-groups.md)
- customStaticFilePath. Customise static files such as the app logo by placing files in the /customFiles/static/ directory. An example of customising app logo is placing a /customFiles/static/favicon.svg to override current SVG. This can be used to change any images/icons/css/fonts/js etc in Stirling-PDF
### Environment only parameters
+
- ``SYSTEM_ROOTURIPATH`` ie set to ``/pdf-app`` to Set the application's root URI to ``localhost:8080/pdf-app``
- ``SYSTEM_CONNECTIONTIMEOUTMINUTES`` to set custom connection timeout values
- ``DOCKER_ENABLE_SECURITY`` to tell docker to download security jar (required as true for auth login)
- ``INSTALL_BOOK_AND_ADVANCED_HTML_OPS`` to download calibre onto stirling-pdf enabling pdf to/from book and advanced html conversion
## API
+
For those wanting to use Stirling-PDFs backend API to link with their own custom scripting to edit PDFs you can view all existing API documentation
[here](https://app.swaggerhub.com/apis-docs/Stirling-Tools/Stirling-PDF/) or navigate to /swagger-ui/index.html of your stirling-pdf instance for your versions documentation (Or by following the API button in your settings of Stirling-PDF)
## Login authentication
+
![stirling-login](images/login-light.png)
+
### Prerequisites:
+
- User must have the folder ./configs volumed within docker so that it is retained during updates.
- Docker uses must download the security jar version by setting ``DOCKER_ENABLE_SECURITY`` to ``true`` in environment variables.
- Then either enable login via the settings.yml file or via setting ``SECURITY_ENABLE_LOGIN`` to ``true``
@@ -257,6 +280,7 @@ For API usage you must provide a header with 'X-API-Key' and the associated API
## FAQ
### Q1: What are your planned features?
+
- Progress bar/Tracking
- Full custom logic pipelines to combine multiple operations together.
- Folder support with auto scanning to perform operations on
@@ -266,7 +290,9 @@ For API usage you must provide a header with 'X-API-Key' and the associated API
- Fill forms manually or automatically
### Q2: Why is my application downloading .htm files?
+
This is an issue caused commonly by your NGINX configuration. The default file upload size for NGINX is 1MB, you need to add the following in your Nginx sites-available file. ``client_max_body_size SIZE;`` Where "SIZE" is 50M for example for 50MB files.
### Q3: Why is my download timing out
+
NGINX has timeout values by default so if you are running Stirling-PDF behind NGINX you may need to set a timeout value such as adding the config ``proxy_read_timeout 3600;``
diff --git a/scripts/counter_translation.py b/scripts/counter_translation.py
new file mode 100644
index 00000000..c7f57575
--- /dev/null
+++ b/scripts/counter_translation.py
@@ -0,0 +1,122 @@
+"""A script to update language progress status in README.md based on
+properties file comparison.
+
+This script compares default properties file with others in a directory to
+determine language progress.
+It then updates README.md based on provided progress list.
+
+Author: Ludy87
+
+Example:
+ To use this script, simply run it from command line:
+ $ python counter_translation.py
+"""
+import os
+import glob
+import re
+from typing import List, Tuple
+
+
+def write_readme(progress_list: List[Tuple[str, int]]) -> None:
+ """
+ Updates the progress status in the README.md file based
+ on the provided progress list.
+
+ Parameters:
+ progress_list (List[Tuple[str, int]]): A list of tuples containing
+ language and progress percentage.
+
+ Returns:
+ None
+ """
+ with open("README.md", "r", encoding="utf-8") as file:
+ content = file.read()
+
+ lines = content.split("\n")
+ for i, line in enumerate(lines[2:], start=2):
+ for progress in progress_list:
+ language, value = progress
+ if language in line:
+ match = re.search(r"\!\[(\d+(\.\d+)?)%\]\(.*\)", line)
+ if match:
+ lines[i] = line.replace(
+ match.group(0),
+ f"![{value}%](https://geps.dev/progress/{value})",
+ )
+
+ new_content = "\n".join(lines)
+
+ with open("README.md", "w", encoding="utf-8") as file:
+ file.write(new_content)
+
+
+def compare_files(default_file_path, files_directory) -> List[Tuple[str, int]]:
+ """
+ Compares the default properties file with other
+ properties files in the directory.
+
+ Parameters:
+ default_file_path (str): The path to the default properties file.
+ files_directory (str): The directory containing other properties files.
+
+ Returns:
+ List[Tuple[str, int]]: A list of tuples containing
+ language and progress percentage.
+ """
+ file_paths = glob.glob(os.path.join(files_directory, "messages_*.properties"))
+ num_lines = sum(1 for _ in open(default_file_path, encoding="utf-8"))
+
+ result_list = []
+
+ for file_path in file_paths:
+ language = (
+ os.path.basename(file_path)
+ .split("messages_", 1)[1]
+ .split(".properties", 1)[0]
+ )
+
+ fails = 0
+ if "en_GB" in language or "en_US" in language:
+ result_list.append(("en_GB", 100))
+ result_list.append(("en_US", 100))
+ continue
+
+ with open(default_file_path, "r", encoding="utf-8") as default_file, open(
+ file_path, "r", encoding="utf-8"
+ ) as file:
+ for _ in range(5):
+ next(default_file)
+ try:
+ next(file)
+ except StopIteration:
+ fails = num_lines
+
+ for _, (line_default, line_file) in enumerate(
+ zip(default_file, file), start=6
+ ):
+ try:
+ if (
+ line_default.split("=", 1)[1].strip()
+ == line_file.split("=", 1)[1].strip()
+ ):
+ fails += 1
+ except IndexError:
+ pass
+
+ result_list.append(
+ (
+ language,
+ int((num_lines - fails) * 100 / num_lines),
+ )
+ )
+
+ unique_data = list(set(result_list))
+ unique_data.sort(key=lambda x: x[1], reverse=True)
+
+ return unique_data
+
+
+if __name__ == "__main__":
+ directory = os.path.join(os.getcwd(), "src", "main", "resources")
+ reference_file = os.path.join(directory, "messages_en_GB.properties")
+ write_readme(compare_files(reference_file, directory))