From 4d47e535b5b3e4000d1d92033cb7cfca68b09565 Mon Sep 17 00:00:00 2001 From: Anthony Stirling <77850077+Frooodle@users.noreply.github.com> Date: Thu, 12 Sep 2024 13:30:05 +0100 Subject: [PATCH 1/5] Update README.md --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 095f645028d..f56fa5cd4e0 100644 --- a/README.md +++ b/README.md @@ -100,6 +100,8 @@ Demo of the app is available [here](https://stirlingpdf.io). - [PDF-LIB.js](https://github.com/Hopding/pdf-lib) ## How to use +### Windows +For windows users download the latest Stirling-PDF.exe from our [release](https://github.com/Stirling-Tools/Stirling-PDF/releases) section or by clicking [here](https://github.com/Stirling-Tools/Stirling-PDF/releases/latest/download/Stirling-PDF.exe) ### Locally From 8c01425eeed94ae40b4903183ae75eb6af4be8a6 Mon Sep 17 00:00:00 2001 From: Anthony Stirling <77850077+Frooodle@users.noreply.github.com> Date: Fri, 13 Sep 2024 16:42:38 +0100 Subject: [PATCH 2/5] Lots of changes (#1889) * Add image support to multi-tool page Related to #278 * changes to support image types * final touches * final touches * final touches Signed-off-by: a * final touches Signed-off-by: a * final touches Signed-off-by: a * final touches Signed-off-by: a * final touches Signed-off-by: a * final touches Signed-off-by: a * final touches Signed-off-by: a * Update translation files (#1888) Signed-off-by: GitHub Action Co-authored-by: GitHub Action --------- Signed-off-by: a Signed-off-by: GitHub Action Co-authored-by: a Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com> Co-authored-by: GitHub Action --- .../docker-compose-latest-fat-security.yml | 1 - ...ocker-compose-latest-security-with-sso.yml | 1 - .../docker-compose-latest-security.yml | 1 - ...ker-compose-latest-ultra-lite-security.yml | 1 - .../docker-compose-latest-ultra-lite.yml | 1 - exampleYmlFiles/docker-compose-latest.yml | 1 - .../software/SPDF/EE/EEAppConfig.java | 25 + .../software/SPDF/SPdfApplication.java | 1 - .../software/SPDF/config/AppConfig.java | 25 + .../SPDF/config/AppUpdateService.java | 2 +- .../software/SPDF/config/MetricsFilter.java | 23 +- .../SPDF/config/PdfMetadataService.java | 139 +++++ .../config/security/AppUpdateAuthService.java | 2 +- .../security/SecurityConfiguration.java | 14 +- .../security/UserAuthenticationFilter.java | 5 +- .../SPDF/config/security/UserService.java | 11 + ...tomOAuth2AuthenticationSuccessHandler.java | 2 +- .../CustomOAuth2LogoutSuccessHandler.java | 2 +- .../oauth2/CustomOAuth2UserService.java | 2 +- .../controller/api/SplitPDFController.java | 6 +- .../api/SplitPdfByChaptersController.java | 6 +- .../SPDF/controller/api/UserController.java | 4 +- .../api/misc/FlattenController.java | 6 +- .../controller/api/misc/ShowJavascript.java | 4 +- .../api/pipeline/UserServiceInterface.java | 2 + .../controller/web/AccountWebController.java | 6 +- .../controller/web/MetricsController.java | 325 +++++----- .../SPDF/model/ApplicationProperties.java | 555 ++---------------- .../SPDF/repository/UserRepository.java | 1 - .../software/SPDF/utils/PdfUtils.java | 27 - .../software/SPDF/utils/RequestUriUtils.java | 23 + src/main/resources/messages_ar_AR.properties | 6 +- src/main/resources/messages_bg_BG.properties | 6 +- src/main/resources/messages_ca_CA.properties | 6 +- src/main/resources/messages_cs_CZ.properties | 6 +- src/main/resources/messages_da_DK.properties | 6 +- src/main/resources/messages_de_DE.properties | 6 +- src/main/resources/messages_el_GR.properties | 6 +- src/main/resources/messages_en_GB.properties | 6 +- src/main/resources/messages_en_US.properties | 6 +- src/main/resources/messages_es_ES.properties | 6 +- src/main/resources/messages_eu_ES.properties | 6 +- src/main/resources/messages_fr_FR.properties | 6 +- src/main/resources/messages_ga_IE.properties | 6 +- src/main/resources/messages_hi_IN.properties | 6 +- src/main/resources/messages_hr_HR.properties | 6 +- src/main/resources/messages_hu_HU.properties | 6 +- src/main/resources/messages_id_ID.properties | 6 +- src/main/resources/messages_it_IT.properties | 6 +- src/main/resources/messages_ja_JP.properties | 6 +- src/main/resources/messages_ko_KR.properties | 6 +- src/main/resources/messages_nl_NL.properties | 6 +- src/main/resources/messages_no_NB.properties | 6 +- src/main/resources/messages_pl_PL.properties | 6 +- src/main/resources/messages_pt_BR.properties | 6 +- src/main/resources/messages_pt_PT.properties | 6 +- src/main/resources/messages_ro_RO.properties | 6 +- src/main/resources/messages_ru_RU.properties | 6 +- src/main/resources/messages_sk_SK.properties | 6 +- .../resources/messages_sr_LATN_RS.properties | 6 +- src/main/resources/messages_sv_SE.properties | 6 +- src/main/resources/messages_th_TH.properties | 6 +- src/main/resources/messages_tr_TR.properties | 6 +- src/main/resources/messages_uk_UA.properties | 6 +- src/main/resources/messages_vi_VN.properties | 6 +- src/main/resources/messages_zh_CN.properties | 6 +- src/main/resources/messages_zh_TW.properties | 6 +- src/main/resources/settings.yml.template | 16 + .../static/js/multitool/PdfContainer.js | 2 +- .../resources/templates/fragments/common.html | 16 +- .../resources/templates/fragments/footer.html | 5 + .../templates/misc/add-page-numbers.html | 5 +- src/main/resources/templates/multi-tool.html | 2 +- .../software/SPDF/utils/PdfUtilsTest.java | 7 - 74 files changed, 735 insertions(+), 758 deletions(-) create mode 100644 src/main/java/stirling/software/SPDF/EE/EEAppConfig.java create mode 100644 src/main/java/stirling/software/SPDF/config/PdfMetadataService.java diff --git a/exampleYmlFiles/docker-compose-latest-fat-security.yml b/exampleYmlFiles/docker-compose-latest-fat-security.yml index a581fa9b866..f29a8a9fa18 100644 --- a/exampleYmlFiles/docker-compose-latest-fat-security.yml +++ b/exampleYmlFiles/docker-compose-latest-fat-security.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF-Security-Fat diff --git a/exampleYmlFiles/docker-compose-latest-security-with-sso.yml b/exampleYmlFiles/docker-compose-latest-security-with-sso.yml index 1970a079dd6..7791ae7ae03 100644 --- a/exampleYmlFiles/docker-compose-latest-security-with-sso.yml +++ b/exampleYmlFiles/docker-compose-latest-security-with-sso.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF-Security diff --git a/exampleYmlFiles/docker-compose-latest-security.yml b/exampleYmlFiles/docker-compose-latest-security.yml index b0ec6ea1d0a..c1e8e712f4c 100644 --- a/exampleYmlFiles/docker-compose-latest-security.yml +++ b/exampleYmlFiles/docker-compose-latest-security.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF-Security diff --git a/exampleYmlFiles/docker-compose-latest-ultra-lite-security.yml b/exampleYmlFiles/docker-compose-latest-ultra-lite-security.yml index 2faac865f5f..d20ed21ceb5 100644 --- a/exampleYmlFiles/docker-compose-latest-ultra-lite-security.yml +++ b/exampleYmlFiles/docker-compose-latest-ultra-lite-security.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF-Ultra-Lite-Security diff --git a/exampleYmlFiles/docker-compose-latest-ultra-lite.yml b/exampleYmlFiles/docker-compose-latest-ultra-lite.yml index da868ff2885..990ba959bdf 100644 --- a/exampleYmlFiles/docker-compose-latest-ultra-lite.yml +++ b/exampleYmlFiles/docker-compose-latest-ultra-lite.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF-Ultra-Lite diff --git a/exampleYmlFiles/docker-compose-latest.yml b/exampleYmlFiles/docker-compose-latest.yml index 6090e291e3d..cab8c3cd762 100644 --- a/exampleYmlFiles/docker-compose-latest.yml +++ b/exampleYmlFiles/docker-compose-latest.yml @@ -1,4 +1,3 @@ -version: '3.3' services: stirling-pdf: container_name: Stirling-PDF diff --git a/src/main/java/stirling/software/SPDF/EE/EEAppConfig.java b/src/main/java/stirling/software/SPDF/EE/EEAppConfig.java new file mode 100644 index 00000000000..0818aa7df7b --- /dev/null +++ b/src/main/java/stirling/software/SPDF/EE/EEAppConfig.java @@ -0,0 +1,25 @@ +package stirling.software.SPDF.EE; + +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.context.annotation.Bean; +import org.springframework.context.annotation.Configuration; +import org.springframework.context.annotation.Lazy; + +import stirling.software.SPDF.model.ApplicationProperties; + +@Configuration +@Lazy +public class EEAppConfig { + + private static final Logger logger = LoggerFactory.getLogger(EEAppConfig.class); + + @Autowired ApplicationProperties applicationProperties; + + @Bean(name = "RunningEE") + public boolean runningEnterpriseEdition() { + // TODO: Implement EE detection + return false; + } +} diff --git a/src/main/java/stirling/software/SPDF/SPdfApplication.java b/src/main/java/stirling/software/SPDF/SPdfApplication.java index bc17e0f139c..6c90ee3c8d5 100644 --- a/src/main/java/stirling/software/SPDF/SPdfApplication.java +++ b/src/main/java/stirling/software/SPDF/SPdfApplication.java @@ -31,7 +31,6 @@ public class SPdfApplication { @Autowired private Environment env; - @Autowired ApplicationProperties applicationProperties; private static String serverPortStatic; diff --git a/src/main/java/stirling/software/SPDF/config/AppConfig.java b/src/main/java/stirling/software/SPDF/config/AppConfig.java index ffa56c197ae..6fc09917614 100644 --- a/src/main/java/stirling/software/SPDF/config/AppConfig.java +++ b/src/main/java/stirling/software/SPDF/config/AppConfig.java @@ -135,4 +135,29 @@ public Predicate processOnlyFiles() { } }; } + + @Bean(name = "termsAndConditions") + public String termsAndConditions() { + return applicationProperties.getLegal().getTermsAndConditions(); + } + + @Bean(name = "privacyPolicy") + public String privacyPolicy() { + return applicationProperties.getLegal().getPrivacyPolicy(); + } + + @Bean(name = "cookiePolicy") + public String cookiePolicy() { + return applicationProperties.getLegal().getCookiePolicy(); + } + + @Bean(name = "impressum") + public String impressum() { + return applicationProperties.getLegal().getImpressum(); + } + + @Bean(name = "accessibilityStatement") + public String accessibilityStatement() { + return applicationProperties.getLegal().getAccessibilityStatement(); + } } diff --git a/src/main/java/stirling/software/SPDF/config/AppUpdateService.java b/src/main/java/stirling/software/SPDF/config/AppUpdateService.java index 7c7a9a49c71..7fc87629083 100644 --- a/src/main/java/stirling/software/SPDF/config/AppUpdateService.java +++ b/src/main/java/stirling/software/SPDF/config/AppUpdateService.java @@ -18,7 +18,7 @@ class AppUpdateService { @Bean(name = "shouldShow") @Scope("request") public boolean shouldShow() { - boolean showUpdate = applicationProperties.getSystem().getShowUpdate(); + boolean showUpdate = applicationProperties.getSystem().isShowUpdate(); boolean showAdminResult = (showAdmin != null) ? showAdmin.getShowUpdateOnlyAdmins() : true; return showUpdate && showAdminResult; } diff --git a/src/main/java/stirling/software/SPDF/config/MetricsFilter.java b/src/main/java/stirling/software/SPDF/config/MetricsFilter.java index 876613f4fc2..c0231ca7f35 100644 --- a/src/main/java/stirling/software/SPDF/config/MetricsFilter.java +++ b/src/main/java/stirling/software/SPDF/config/MetricsFilter.java @@ -13,6 +13,7 @@ import jakarta.servlet.ServletException; import jakarta.servlet.http.HttpServletRequest; import jakarta.servlet.http.HttpServletResponse; +import stirling.software.SPDF.utils.RequestUriUtils; @Component public class MetricsFilter extends OncePerRequestFilter { @@ -30,32 +31,16 @@ protected void doFilterInternal( throws ServletException, IOException { String uri = request.getRequestURI(); - // System.out.println("uri="+uri + ", method=" + request.getMethod() ); - // Ignore static resources - if (!(uri.startsWith("/js") - || uri.startsWith("/v1/api-docs") - || uri.endsWith("robots.txt") - || uri.startsWith("/images") - || uri.endsWith(".png") - || uri.endsWith(".ico") - || uri.endsWith(".css") - || uri.endsWith(".map") - || uri.endsWith(".svg") - || uri.endsWith(".js") - || uri.contains("swagger") - || uri.startsWith("/api/v1/info") - || uri.startsWith("/site.webmanifest") - || uri.startsWith("/fonts") - || uri.startsWith("/pdfjs"))) { + if (RequestUriUtils.isTrackableResource(request.getContextPath(), uri)) { Counter counter = Counter.builder("http.requests") - .tag("uri", uri) + .tag("session", request.getSession().getId()) .tag("method", request.getMethod()) + .tag("uri", uri) .register(meterRegistry); counter.increment(); - // System.out.println("Counted"); } filterChain.doFilter(request, response); diff --git a/src/main/java/stirling/software/SPDF/config/PdfMetadataService.java b/src/main/java/stirling/software/SPDF/config/PdfMetadataService.java new file mode 100644 index 00000000000..9eba472e7b7 --- /dev/null +++ b/src/main/java/stirling/software/SPDF/config/PdfMetadataService.java @@ -0,0 +1,139 @@ +package stirling.software.SPDF.config; + +import java.util.Calendar; + +import org.apache.pdfbox.pdmodel.PDDocument; +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.beans.factory.annotation.Qualifier; +import org.springframework.stereotype.Service; + +import jakarta.annotation.PostConstruct; +import stirling.software.SPDF.controller.api.pipeline.UserServiceInterface; +import stirling.software.SPDF.model.ApplicationProperties; +import stirling.software.SPDF.model.PdfMetadata; + +@Service +public class PdfMetadataService { + + private static PdfMetadataService instance; + + private final ApplicationProperties applicationProperties; + private final String appVersion; + private final UserServiceInterface userService; + + @Autowired + public PdfMetadataService( + ApplicationProperties applicationProperties, + @Qualifier("appVersion") String appVersion, + @Autowired(required = false) UserServiceInterface userService) { + this.applicationProperties = applicationProperties; + this.appVersion = appVersion; + this.userService = userService; + } + + @PostConstruct + public void init() { + instance = this; + } + + // Static methods for easy access + + public static PdfMetadata extractMetadataFromPdf(PDDocument pdf) { + return instance.extractMetadataFromPdfInstance(pdf); + } + + public static void setDefaultMetadata(PDDocument pdf) { + instance.setDefaultMetadataInstance(pdf); + } + + public static void setMetadataToPdf(PDDocument pdf, PdfMetadata pdfMetadata) { + instance.setMetadataToPdfInstance(pdf, pdfMetadata); + } + + public static void setMetadataToPdf( + PDDocument pdf, PdfMetadata pdfMetadata, boolean newlyCreated) { + instance.setMetadataToPdfInstance(pdf, pdfMetadata, newlyCreated); + } + + // Instance methods + + private PdfMetadata extractMetadataFromPdfInstance(PDDocument pdf) { + return PdfMetadata.builder() + .author(pdf.getDocumentInformation().getAuthor()) + .producer(pdf.getDocumentInformation().getProducer()) + .title(pdf.getDocumentInformation().getTitle()) + .creator(pdf.getDocumentInformation().getCreator()) + .subject(pdf.getDocumentInformation().getSubject()) + .keywords(pdf.getDocumentInformation().getKeywords()) + .creationDate(pdf.getDocumentInformation().getCreationDate()) + .modificationDate(pdf.getDocumentInformation().getModificationDate()) + .build(); + } + + private void setDefaultMetadataInstance(PDDocument pdf) { + PdfMetadata metadata = extractMetadataFromPdfInstance(pdf); + setMetadataToPdfInstance(pdf, metadata); + } + + private void setMetadataToPdfInstance(PDDocument pdf, PdfMetadata pdfMetadata) { + setMetadataToPdfInstance(pdf, pdfMetadata, true); + } + + private void setMetadataToPdfInstance( + PDDocument pdf, PdfMetadata pdfMetadata, boolean newlyCreated) { + if (newlyCreated || pdfMetadata.getCreationDate() == null) { + setNewDocumentMetadata(pdf, pdfMetadata); + } + setCommonMetadata(pdf, pdfMetadata); + } + + private void setNewDocumentMetadata(PDDocument pdf, PdfMetadata pdfMetadata) { + + String title = pdfMetadata.getTitle(); + String creator = "Stirling-PDF"; + +// if (applicationProperties +// .getEnterpriseEdition() +// .getCustomMetadata() +// .isAutoUpdateMetadata()) { + + // producer = + // + // applicationProperties.getEnterpriseEdition().getCustomMetadata().getProducer(); + // creator = + // applicationProperties.getEnterpriseEdition().getCustomMetadata().getCreator(); + // title = applicationProperties.getEnterpriseEdition().getCustomMetadata().getTitle(); + +// if ("{filename}".equals(title)) { +// title = "Filename"; // Replace with actual filename logic +// } else if ("{unchanged}".equals(title)) { +// title = pdfMetadata.getTitle(); // Keep the original title +// } +// } + + pdf.getDocumentInformation().setTitle(title); + pdf.getDocumentInformation().setCreator(creator + " " + appVersion); + pdf.getDocumentInformation().setCreationDate(Calendar.getInstance()); + } + + private void setCommonMetadata(PDDocument pdf, PdfMetadata pdfMetadata) { + String producer = "Stirling-PDF"; + pdf.getDocumentInformation().setProducer(producer + " " + appVersion); + pdf.getDocumentInformation().setSubject(pdfMetadata.getSubject()); + pdf.getDocumentInformation().setKeywords(pdfMetadata.getKeywords()); + pdf.getDocumentInformation().setModificationDate(Calendar.getInstance()); + + String author = pdfMetadata.getAuthor(); + // if (applicationProperties + // .getEnterpriseEdition() + // .getCustomMetadata() + // .isAutoUpdateMetadata()) { + // author = applicationProperties.getEnterpriseEdition().getCustomMetadata().getAuthor(); + + // if (userService != null) { + // author = author.replace("username", userService.getCurrentUsername()); + // } + // } + pdf.getDocumentInformation().setAuthor(author); + } +} diff --git a/src/main/java/stirling/software/SPDF/config/security/AppUpdateAuthService.java b/src/main/java/stirling/software/SPDF/config/security/AppUpdateAuthService.java index c4b53ad51f0..5a16aa308ab 100644 --- a/src/main/java/stirling/software/SPDF/config/security/AppUpdateAuthService.java +++ b/src/main/java/stirling/software/SPDF/config/security/AppUpdateAuthService.java @@ -20,7 +20,7 @@ class AppUpdateAuthService implements ShowAdminInterface { @Override public boolean getShowUpdateOnlyAdmins() { - boolean showUpdate = applicationProperties.getSystem().getShowUpdate(); + boolean showUpdate = applicationProperties.getSystem().isShowUpdate(); if (!showUpdate) { return showUpdate; } diff --git a/src/main/java/stirling/software/SPDF/config/security/SecurityConfiguration.java b/src/main/java/stirling/software/SPDF/config/security/SecurityConfiguration.java index ed7b7921a7c..aa266d2f144 100644 --- a/src/main/java/stirling/software/SPDF/config/security/SecurityConfiguration.java +++ b/src/main/java/stirling/software/SPDF/config/security/SecurityConfiguration.java @@ -152,8 +152,8 @@ public SecurityFilterChain filterChain(HttpSecurity http) throws Exception { .authenticated()); // Handle OAUTH2 Logins - if (applicationProperties.getSecurity().getOAUTH2() != null - && applicationProperties.getSecurity().getOAUTH2().getEnabled() + if (applicationProperties.getSecurity().getOauth2() != null + && applicationProperties.getSecurity().getOauth2().getEnabled() && !applicationProperties .getSecurity() .getLoginMethod() @@ -222,7 +222,7 @@ public ClientRegistrationRepository clientRegistrationRepository() { } private Optional googleClientRegistration() { - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (oauth == null || !oauth.getEnabled()) { return Optional.empty(); } @@ -251,7 +251,7 @@ private Optional googleClientRegistration() { } private Optional keycloakClientRegistration() { - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (oauth == null || !oauth.getEnabled()) { return Optional.empty(); } @@ -275,7 +275,7 @@ private Optional keycloakClientRegistration() { } private Optional githubClientRegistration() { - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (oauth == null || !oauth.getEnabled()) { return Optional.empty(); } @@ -304,7 +304,7 @@ private Optional githubClientRegistration() { } private Optional oidcClientRegistration() { - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (oauth == null || oauth.getIssuer() == null || oauth.getIssuer().isEmpty() @@ -352,7 +352,7 @@ GrantedAuthoritiesMapper userAuthoritiesMapper() { String useAsUsername = applicationProperties .getSecurity() - .getOAUTH2() + .getOauth2() .getUseAsUsername(); Optional userOpt = userService.findByUsernameIgnoreCase( diff --git a/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java b/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java index b71fab77034..79b285211d0 100644 --- a/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java +++ b/src/main/java/stirling/software/SPDF/config/security/UserAuthenticationFilter.java @@ -159,7 +159,10 @@ protected boolean shouldNotFilter(HttpServletRequest request) throws ServletExce }; for (String pattern : permitAllPatterns) { - if (uri.startsWith(pattern) || uri.endsWith(".svg") || uri.endsWith(".png") || uri.endsWith(".ico")) { + if (uri.startsWith(pattern) + || uri.endsWith(".svg") + || uri.endsWith(".png") + || uri.endsWith(".ico")) { return true; } } diff --git a/src/main/java/stirling/software/SPDF/config/security/UserService.java b/src/main/java/stirling/software/SPDF/config/security/UserService.java index 5adfdbc7689..ece81355501 100644 --- a/src/main/java/stirling/software/SPDF/config/security/UserService.java +++ b/src/main/java/stirling/software/SPDF/config/security/UserService.java @@ -11,6 +11,7 @@ import org.springframework.security.core.Authentication; import org.springframework.security.core.GrantedAuthority; import org.springframework.security.core.authority.SimpleGrantedAuthority; +import org.springframework.security.core.context.SecurityContextHolder; import org.springframework.security.core.session.SessionInformation; import org.springframework.security.core.userdetails.UserDetails; import org.springframework.security.core.userdetails.UsernameNotFoundException; @@ -342,4 +343,14 @@ public void invalidateUserSessions(String username) { } } } + + public String getCurrentUsername() { + Object principal = SecurityContextHolder.getContext().getAuthentication().getPrincipal(); + + if (principal instanceof UserDetails) { + return ((UserDetails) principal).getUsername(); + } else { + return principal.toString(); + } + } } diff --git a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2AuthenticationSuccessHandler.java b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2AuthenticationSuccessHandler.java index 4c7e04d9bfe..c8c3f2174d4 100644 --- a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2AuthenticationSuccessHandler.java +++ b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2AuthenticationSuccessHandler.java @@ -66,7 +66,7 @@ public void onAuthenticationSuccess( // Redirect to the original destination super.onAuthenticationSuccess(request, response, authentication); } else { - OAUTH2 oAuth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oAuth = applicationProperties.getSecurity().getOauth2(); if (loginAttemptService.isBlocked(username)) { if (session != null) { diff --git a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2LogoutSuccessHandler.java b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2LogoutSuccessHandler.java index 907ddd4bcfc..5bbff53f623 100644 --- a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2LogoutSuccessHandler.java +++ b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2LogoutSuccessHandler.java @@ -43,7 +43,7 @@ public void onLogoutSuccess( } return; } - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (authentication instanceof OAuth2AuthenticationToken) { OAuth2AuthenticationToken oauthToken = (OAuth2AuthenticationToken) authentication; diff --git a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2UserService.java b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2UserService.java index b9766480fc4..ebe734b5806 100644 --- a/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2UserService.java +++ b/src/main/java/stirling/software/SPDF/config/security/oauth2/CustomOAuth2UserService.java @@ -43,7 +43,7 @@ public CustomOAuth2UserService( @Override public OidcUser loadUser(OidcUserRequest userRequest) throws OAuth2AuthenticationException { - OAUTH2 oauth2 = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth2 = applicationProperties.getSecurity().getOauth2(); String usernameAttribute = oauth2.getUseAsUsername(); if (usernameAttribute == null || usernameAttribute.trim().isEmpty()) { Client client = oauth2.getClient(); diff --git a/src/main/java/stirling/software/SPDF/controller/api/SplitPDFController.java b/src/main/java/stirling/software/SPDF/controller/api/SplitPDFController.java index ab80cad34ad..af662433bbc 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/SplitPDFController.java +++ b/src/main/java/stirling/software/SPDF/controller/api/SplitPDFController.java @@ -27,9 +27,9 @@ import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; +import stirling.software.SPDF.config.PdfMetadataService; import stirling.software.SPDF.model.PdfMetadata; import stirling.software.SPDF.model.api.PDFWithPageNums; -import stirling.software.SPDF.utils.PdfUtils; import stirling.software.SPDF.utils.WebResponseUtils; @RestController @@ -51,7 +51,7 @@ public ResponseEntity splitPdf(@ModelAttribute PDFWithPageNums request) // open the pdf document PDDocument document = Loader.loadPDF(file.getBytes()); - PdfMetadata metadata = PdfUtils.extractMetadataFromPdf(document); + PdfMetadata metadata = PdfMetadataService.extractMetadataFromPdf(document); int totalPages = document.getNumberOfPages(); List pageNumbers = request.getPageNumbersList(document, false); System.out.println( @@ -79,7 +79,7 @@ public ResponseEntity splitPdf(@ModelAttribute PDFWithPageNums request) previousPageNumber = splitPoint + 1; // Transfer metadata to split pdf - PdfUtils.setMetadataToPdf(splitDocument, metadata); + PdfMetadataService.setMetadataToPdf(splitDocument, metadata); ByteArrayOutputStream baos = new ByteArrayOutputStream(); splitDocument.save(baos); diff --git a/src/main/java/stirling/software/SPDF/controller/api/SplitPdfByChaptersController.java b/src/main/java/stirling/software/SPDF/controller/api/SplitPdfByChaptersController.java index 776856ecefd..8d020b16b27 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/SplitPdfByChaptersController.java +++ b/src/main/java/stirling/software/SPDF/controller/api/SplitPdfByChaptersController.java @@ -31,9 +31,9 @@ import lombok.Data; import lombok.EqualsAndHashCode; import lombok.NoArgsConstructor; +import stirling.software.SPDF.config.PdfMetadataService; import stirling.software.SPDF.model.PdfMetadata; import stirling.software.SPDF.model.api.SplitPdfByChaptersRequest; -import stirling.software.SPDF.utils.PdfUtils; import stirling.software.SPDF.utils.WebResponseUtils; @RestController @@ -258,7 +258,7 @@ public List getSplitDocumentsBoas( List splitDocumentsBoas = new ArrayList<>(); PdfMetadata metadata = null; if (includeMetadata) { - metadata = PdfUtils.extractMetadataFromPdf(sourceDocument); + metadata = PdfMetadataService.extractMetadataFromPdf(sourceDocument); } for (Bookmark bookmark : bookmarks) { try (PDDocument splitDocument = new PDDocument()) { @@ -273,7 +273,7 @@ public List getSplitDocumentsBoas( } ByteArrayOutputStream baos = new ByteArrayOutputStream(); if (includeMetadata) { - PdfUtils.setMetadataToPdf(splitDocument, metadata); + PdfMetadataService.setMetadataToPdf(splitDocument, metadata); } splitDocument.save(baos); diff --git a/src/main/java/stirling/software/SPDF/controller/api/UserController.java b/src/main/java/stirling/software/SPDF/controller/api/UserController.java index bc59b91e571..719adac19a5 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/UserController.java +++ b/src/main/java/stirling/software/SPDF/controller/api/UserController.java @@ -208,8 +208,8 @@ public String updateUserSettings(HttpServletRequest request, Principal principal @PreAuthorize("hasRole('ROLE_ADMIN')") @PostMapping("/admin/saveUser") public RedirectView saveUser( - @RequestParam String username, - @RequestParam(name = "password", required = false) String password, + @RequestParam(name = "username", required = true) String username, + @RequestParam(name = "password", required = true) String password, @RequestParam(name = "role") String role, @RequestParam(name = "authType") String authType, @RequestParam(name = "forceChange", required = false, defaultValue = "false") diff --git a/src/main/java/stirling/software/SPDF/controller/api/misc/FlattenController.java b/src/main/java/stirling/software/SPDF/controller/api/misc/FlattenController.java index 888d76702b6..c180fafa7ae 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/misc/FlattenController.java +++ b/src/main/java/stirling/software/SPDF/controller/api/misc/FlattenController.java @@ -25,9 +25,9 @@ import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.tags.Tag; +import stirling.software.SPDF.config.PdfMetadataService; import stirling.software.SPDF.model.PdfMetadata; import stirling.software.SPDF.model.api.misc.FlattenRequest; -import stirling.software.SPDF.utils.PdfUtils; import stirling.software.SPDF.utils.WebResponseUtils; @RestController @@ -46,7 +46,7 @@ public ResponseEntity flatten(@ModelAttribute FlattenRequest request) th MultipartFile file = request.getFileInput(); PDDocument document = Loader.loadPDF(file.getBytes()); - PdfMetadata metadata = PdfUtils.extractMetadataFromPdf(document); + PdfMetadata metadata = PdfMetadataService.extractMetadataFromPdf(document); Boolean flattenOnlyForms = request.getFlattenOnlyForms(); if (Boolean.TRUE.equals(flattenOnlyForms)) { @@ -80,7 +80,7 @@ public ResponseEntity flatten(@ModelAttribute FlattenRequest request) th logger.error("exception", e); } } - PdfUtils.setMetadataToPdf(newDocument, metadata); + PdfMetadataService.setMetadataToPdf(newDocument, metadata); return WebResponseUtils.pdfDocToWebResponse( newDocument, Filenames.toSimpleFileName(file.getOriginalFilename())); } diff --git a/src/main/java/stirling/software/SPDF/controller/api/misc/ShowJavascript.java b/src/main/java/stirling/software/SPDF/controller/api/misc/ShowJavascript.java index 0a93bf1d4b2..a608fde9515 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/misc/ShowJavascript.java +++ b/src/main/java/stirling/software/SPDF/controller/api/misc/ShowJavascript.java @@ -9,6 +9,7 @@ import org.apache.pdfbox.pdmodel.interactive.action.PDActionJavaScript; import org.slf4j.Logger; import org.slf4j.LoggerFactory; +import org.springframework.http.MediaType; import org.springframework.http.ResponseEntity; import org.springframework.web.bind.annotation.ModelAttribute; import org.springframework.web.bind.annotation.PostMapping; @@ -75,7 +76,8 @@ public ResponseEntity extractHeader(@ModelAttribute PDFFile request) thr return WebResponseUtils.bytesToWebResponse( script.getBytes(StandardCharsets.UTF_8), - Filenames.toSimpleFileName(inputFile.getOriginalFilename()) + ".js"); + Filenames.toSimpleFileName(inputFile.getOriginalFilename()) + ".js", + MediaType.TEXT_PLAIN); } } } diff --git a/src/main/java/stirling/software/SPDF/controller/api/pipeline/UserServiceInterface.java b/src/main/java/stirling/software/SPDF/controller/api/pipeline/UserServiceInterface.java index 1a60441e1cc..c3facb35e11 100644 --- a/src/main/java/stirling/software/SPDF/controller/api/pipeline/UserServiceInterface.java +++ b/src/main/java/stirling/software/SPDF/controller/api/pipeline/UserServiceInterface.java @@ -2,4 +2,6 @@ public interface UserServiceInterface { String getApiKeyForUser(String username); + + String getCurrentUsername(); } diff --git a/src/main/java/stirling/software/SPDF/controller/web/AccountWebController.java b/src/main/java/stirling/software/SPDF/controller/web/AccountWebController.java index 2f6b504209a..9081164d285 100644 --- a/src/main/java/stirling/software/SPDF/controller/web/AccountWebController.java +++ b/src/main/java/stirling/software/SPDF/controller/web/AccountWebController.java @@ -51,7 +51,7 @@ public String login(HttpServletRequest request, Model model, Authentication auth Map providerList = new HashMap<>(); - OAUTH2 oauth = applicationProperties.getSecurity().getOAUTH2(); + OAUTH2 oauth = applicationProperties.getSecurity().getOauth2(); if (oauth != null) { if (oauth.isSettingsValid()) { providerList.put("oidc", oauth.getProvider()); @@ -82,7 +82,7 @@ public String login(HttpServletRequest request, Model model, Authentication auth model.addAttribute("loginMethod", applicationProperties.getSecurity().getLoginMethod()); model.addAttribute( - "oAuth2Enabled", applicationProperties.getSecurity().getOAUTH2().getEnabled()); + "oAuth2Enabled", applicationProperties.getSecurity().getOauth2().getEnabled()); model.addAttribute("currentPage", "login"); @@ -345,7 +345,7 @@ public String account(HttpServletRequest request, Model model, Authentication au // Retrieve username and other attributes username = userDetails.getAttribute( - applicationProperties.getSecurity().getOAUTH2().getUseAsUsername()); + applicationProperties.getSecurity().getOauth2().getUseAsUsername()); // Add oAuth2 Login attributes to the model model.addAttribute("oAuth2Login", true); } diff --git a/src/main/java/stirling/software/SPDF/controller/web/MetricsController.java b/src/main/java/stirling/software/SPDF/controller/web/MetricsController.java index 155fcdbd3cc..ff73fb2d3bd 100644 --- a/src/main/java/stirling/software/SPDF/controller/web/MetricsController.java +++ b/src/main/java/stirling/software/SPDF/controller/web/MetricsController.java @@ -2,11 +2,7 @@ import java.time.Duration; import java.time.LocalDateTime; -import java.util.Comparator; -import java.util.HashMap; -import java.util.List; -import java.util.Map; -import java.util.Optional; +import java.util.*; import java.util.stream.Collectors; import org.springframework.beans.factory.annotation.Autowired; @@ -18,19 +14,20 @@ import org.springframework.web.bind.annotation.RestController; import io.micrometer.core.instrument.Counter; -import io.micrometer.core.instrument.Meter; import io.micrometer.core.instrument.MeterRegistry; import io.swagger.v3.oas.annotations.Operation; import io.swagger.v3.oas.annotations.Parameter; import io.swagger.v3.oas.annotations.tags.Tag; import jakarta.annotation.PostConstruct; +import lombok.extern.slf4j.Slf4j; import stirling.software.SPDF.config.StartupApplicationListener; import stirling.software.SPDF.model.ApplicationProperties; @RestController @RequestMapping("/api/v1/info") @Tag(name = "Info", description = "Info APIs") +@Slf4j public class MetricsController { @Autowired ApplicationProperties applicationProperties; @@ -46,6 +43,7 @@ public void init() { this.metricsEnabled = metricsEnabled; } + @Autowired public MetricsController(MeterRegistry meterRegistry) { this.meterRegistry = meterRegistry; } @@ -66,11 +64,11 @@ public ResponseEntity getStatus() { return ResponseEntity.ok(status); } - @GetMapping("/loads") + @GetMapping("/load") @Operation( summary = "GET request count", description = - "This endpoint returns the total count of GET requests or the count of GET requests for a specific endpoint.") + "This endpoint returns the total count of GET requests for a specific endpoint or all endpoints.") public ResponseEntity getPageLoads( @RequestParam(required = false, name = "endpoint") @Parameter(description = "endpoint") Optional endpoint) { @@ -78,44 +76,33 @@ public ResponseEntity getPageLoads( return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); } try { + double count = getRequestCount("GET", endpoint); + return ResponseEntity.ok(count); + } catch (Exception e) { + return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); + } + } - double count = 0.0; - - for (Meter meter : meterRegistry.getMeters()) { - if (meter.getId().getName().equals("http.requests")) { - String method = meter.getId().getTag("method"); - if (method != null && "GET".equals(method)) { - - if (endpoint.isPresent() && !endpoint.get().isBlank()) { - if (!endpoint.get().startsWith("/")) { - endpoint = Optional.of("/" + endpoint.get()); - } - System.out.println( - "loads " - + endpoint.get() - + " vs " - + meter.getId().getTag("uri")); - if (endpoint.get().equals(meter.getId().getTag("uri"))) { - if (meter instanceof Counter) { - count += ((Counter) meter).count(); - } - } - } else { - if (meter instanceof Counter) { - count += ((Counter) meter).count(); - } - } - } - } - } - + @GetMapping("/load/unique") + @Operation( + summary = "Unique users count for GET requests", + description = + "This endpoint returns the count of unique users for GET requests for a specific endpoint or all endpoints.") + public ResponseEntity getUniquePageLoads( + @RequestParam(required = false, name = "endpoint") @Parameter(description = "endpoint") + Optional endpoint) { + if (!metricsEnabled) { + return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); + } + try { + double count = getUniqueUserCount("GET", endpoint); return ResponseEntity.ok(count); } catch (Exception e) { return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); } } - @GetMapping("/loads/all") + @GetMapping("/load/all") @Operation( summary = "GET requests count for all endpoints", description = "This endpoint returns the count of GET requests for each endpoint.") @@ -124,59 +111,27 @@ public ResponseEntity getAllEndpointLoads() { return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); } try { - Map counts = new HashMap<>(); - - for (Meter meter : meterRegistry.getMeters()) { - if (meter.getId().getName().equals("http.requests")) { - String method = meter.getId().getTag("method"); - if (method != null && "GET".equals(method)) { - String uri = meter.getId().getTag("uri"); - if (uri != null) { - double currentCount = counts.getOrDefault(uri, 0.0); - if (meter instanceof Counter) { - currentCount += ((Counter) meter).count(); - } - counts.put(uri, currentCount); - } - } - } - } - - List results = - counts.entrySet().stream() - .map(entry -> new EndpointCount(entry.getKey(), entry.getValue())) - .sorted(Comparator.comparing(EndpointCount::getCount).reversed()) - .collect(Collectors.toList()); - + List results = getEndpointCounts("GET"); return ResponseEntity.ok(results); } catch (Exception e) { return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); } } - public class EndpointCount { - private String endpoint; - private double count; - - public EndpointCount(String endpoint, double count) { - this.endpoint = endpoint; - this.count = count; - } - - public String getEndpoint() { - return endpoint; - } - - public void setEndpoint(String endpoint) { - this.endpoint = endpoint; - } - - public double getCount() { - return count; + @GetMapping("/load/all/unique") + @Operation( + summary = "Unique users count for GET requests for all endpoints", + description = + "This endpoint returns the count of unique users for GET requests for each endpoint.") + public ResponseEntity getAllUniqueEndpointLoads() { + if (!metricsEnabled) { + return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); } - - public void setCount(double count) { - this.count = count; + try { + List results = getUniqueUserCounts("GET"); + return ResponseEntity.ok(results); + } catch (Exception e) { + return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); } } @@ -184,7 +139,7 @@ public void setCount(double count) { @Operation( summary = "POST request count", description = - "This endpoint returns the total count of POST requests or the count of POST requests for a specific endpoint.") + "This endpoint returns the total count of POST requests for a specific endpoint or all endpoints.") public ResponseEntity getTotalRequests( @RequestParam(required = false, name = "endpoint") @Parameter(description = "endpoint") Optional endpoint) { @@ -192,29 +147,26 @@ public ResponseEntity getTotalRequests( return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); } try { - double count = 0.0; - - for (Meter meter : meterRegistry.getMeters()) { - if (meter.getId().getName().equals("http.requests")) { - String method = meter.getId().getTag("method"); - if (method != null && "POST".equals(method)) { - if (endpoint.isPresent() && !endpoint.get().isBlank()) { - if (!endpoint.get().startsWith("/")) { - endpoint = Optional.of("/" + endpoint.get()); - } - if (endpoint.get().equals(meter.getId().getTag("uri"))) { - if (meter instanceof Counter) { - count += ((Counter) meter).count(); - } - } - } else { - if (meter instanceof Counter) { - count += ((Counter) meter).count(); - } - } - } - } - } + double count = getRequestCount("POST", endpoint); + return ResponseEntity.ok(count); + } catch (Exception e) { + return ResponseEntity.ok(-1); + } + } + + @GetMapping("/requests/unique") + @Operation( + summary = "Unique users count for POST requests", + description = + "This endpoint returns the count of unique users for POST requests for a specific endpoint or all endpoints.") + public ResponseEntity getUniqueTotalRequests( + @RequestParam(required = false, name = "endpoint") @Parameter(description = "endpoint") + Optional endpoint) { + if (!metricsEnabled) { + return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); + } + try { + double count = getUniqueUserCount("POST", endpoint); return ResponseEntity.ok(count); } catch (Exception e) { return ResponseEntity.ok(-1); @@ -230,36 +182,145 @@ public ResponseEntity getAllPostRequests() { return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); } try { - Map counts = new HashMap<>(); - - for (Meter meter : meterRegistry.getMeters()) { - if (meter.getId().getName().equals("http.requests")) { - String method = meter.getId().getTag("method"); - if (method != null && "POST".equals(method)) { - String uri = meter.getId().getTag("uri"); - if (uri != null) { - double currentCount = counts.getOrDefault(uri, 0.0); - if (meter instanceof Counter) { - currentCount += ((Counter) meter).count(); - } - counts.put(uri, currentCount); - } - } - } - } - - List results = - counts.entrySet().stream() - .map(entry -> new EndpointCount(entry.getKey(), entry.getValue())) - .sorted(Comparator.comparing(EndpointCount::getCount).reversed()) - .collect(Collectors.toList()); + List results = getEndpointCounts("POST"); + return ResponseEntity.ok(results); + } catch (Exception e) { + return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); + } + } + @GetMapping("/requests/all/unique") + @Operation( + summary = "Unique users count for POST requests for all endpoints", + description = + "This endpoint returns the count of unique users for POST requests for each endpoint.") + public ResponseEntity getAllUniquePostRequests() { + if (!metricsEnabled) { + return ResponseEntity.status(HttpStatus.FORBIDDEN).body("This endpoint is disabled."); + } + try { + List results = getUniqueUserCounts("POST"); return ResponseEntity.ok(results); } catch (Exception e) { return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build(); } } + private double getRequestCount(String method, Optional endpoint) { + log.info( + "Getting request count for method: {}, endpoint: {}", + method, + endpoint.orElse("all")); + double count = + meterRegistry.find("http.requests").tag("method", method).counters().stream() + .filter( + counter -> + !endpoint.isPresent() + || endpoint.get() + .equals(counter.getId().getTag("uri"))) + .mapToDouble(Counter::count) + .sum(); + log.info("Request count: {}", count); + return count; + } + + private List getEndpointCounts(String method) { + log.info("Getting endpoint counts for method: {}", method); + Map counts = new HashMap<>(); + meterRegistry + .find("http.requests") + .tag("method", method) + .counters() + .forEach( + counter -> { + String uri = counter.getId().getTag("uri"); + counts.merge(uri, counter.count(), Double::sum); + }); + + List result = + counts.entrySet().stream() + .map(entry -> new EndpointCount(entry.getKey(), entry.getValue())) + .sorted(Comparator.comparing(EndpointCount::getCount).reversed()) + .collect(Collectors.toList()); + log.info("Found {} endpoints with counts", result.size()); + return result; + } + + private double getUniqueUserCount(String method, Optional endpoint) { + log.info( + "Getting unique user count for method: {}, endpoint: {}", + method, + endpoint.orElse("all")); + Set uniqueUsers = new HashSet<>(); + meterRegistry.find("http.requests").tag("method", method).counters().stream() + .filter( + counter -> + !endpoint.isPresent() + || endpoint.get().equals(counter.getId().getTag("uri"))) + .forEach( + counter -> { + String session = counter.getId().getTag("session"); + if (session != null) { + uniqueUsers.add(session); + } + }); + log.info("Unique user count: {}", uniqueUsers.size()); + return uniqueUsers.size(); + } + + private List getUniqueUserCounts(String method) { + log.info("Getting unique user counts for method: {}", method); + Map> uniqueUsers = new HashMap<>(); + + meterRegistry + .find("http.requests") + .tag("method", method) + .counters() + .forEach( + counter -> { + String uri = counter.getId().getTag("uri"); + String session = counter.getId().getTag("session"); + if (uri != null && session != null) { + uniqueUsers.computeIfAbsent(uri, k -> new HashSet<>()).add(session); + } + }); + + List result = + uniqueUsers.entrySet().stream() + .map(entry -> new EndpointCount(entry.getKey(), entry.getValue().size())) + .sorted(Comparator.comparing(EndpointCount::getCount).reversed()) + .collect(Collectors.toList()); + + log.info("Found {} endpoints with unique user counts", result.size()); + return result; + } + + public static class EndpointCount { + private String endpoint; + private double count; + + public EndpointCount(String endpoint, double count) { + this.endpoint = endpoint; + this.count = count; + } + + public String getEndpoint() { + return endpoint; + } + + public void setEndpoint(String endpoint) { + this.endpoint = endpoint; + } + + public double getCount() { + return count; + } + + public void setCount(double count) { + this.count = count; + } + } + @GetMapping("/uptime") public ResponseEntity getUptime() { if (!metricsEnabled) { diff --git a/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java b/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java index 266d4520f87..46c92633a48 100644 --- a/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java +++ b/src/main/java/stirling/software/SPDF/model/ApplicationProperties.java @@ -12,6 +12,8 @@ import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.PropertySource; +import lombok.Data; +import lombok.ToString; import stirling.software.SPDF.config.YamlPropertySourceFactory; import stirling.software.SPDF.model.provider.GithubProvider; import stirling.software.SPDF.model.provider.GoogleProvider; @@ -21,225 +23,56 @@ @Configuration @ConfigurationProperties(prefix = "") @PropertySource(value = "file:./configs/settings.yml", factory = YamlPropertySourceFactory.class) +@Data public class ApplicationProperties { - private Security security; - private System system; - private Ui ui; - private Endpoints endpoints; - private Metrics metrics; - private AutomaticallyGenerated automaticallyGenerated; - private AutoPipeline autoPipeline; - private static final Logger logger = LoggerFactory.getLogger(ApplicationProperties.class); - - public AutoPipeline getAutoPipeline() { - return autoPipeline != null ? autoPipeline : new AutoPipeline(); - } - - public void setAutoPipeline(AutoPipeline autoPipeline) { - this.autoPipeline = autoPipeline; - } - - public Security getSecurity() { - return security != null ? security : new Security(); - } - - public void setSecurity(Security security) { - this.security = security; - } - - public System getSystem() { - return system != null ? system : new System(); - } - - public void setSystem(System system) { - this.system = system; - } - - public Ui getUi() { - return ui != null ? ui : new Ui(); - } - - public void setUi(Ui ui) { - this.ui = ui; - } - - public Endpoints getEndpoints() { - return endpoints != null ? endpoints : new Endpoints(); - } - - public void setEndpoints(Endpoints endpoints) { - this.endpoints = endpoints; - } - - public Metrics getMetrics() { - return metrics != null ? metrics : new Metrics(); - } - - public void setMetrics(Metrics metrics) { - this.metrics = metrics; - } - public AutomaticallyGenerated getAutomaticallyGenerated() { - return automaticallyGenerated != null - ? automaticallyGenerated - : new AutomaticallyGenerated(); - } - - public void setAutomaticallyGenerated(AutomaticallyGenerated automaticallyGenerated) { - this.automaticallyGenerated = automaticallyGenerated; - } - - @Override - public String toString() { - return "ApplicationProperties [security=" - + security - + ", system=" - + system - + ", ui=" - + ui - + ", endpoints=" - + endpoints - + ", metrics=" - + metrics - + ", automaticallyGenerated=" - + automaticallyGenerated - + ", autoPipeline=" - + autoPipeline - + "]"; - } + private Legal legal = new Legal(); + private Security security = new Security(); + private System system = new System(); + private Ui ui = new Ui(); + private Endpoints endpoints = new Endpoints(); + private Metrics metrics = new Metrics(); + private AutomaticallyGenerated automaticallyGenerated = new AutomaticallyGenerated(); + private EnterpriseEdition enterpriseEdition = new EnterpriseEdition(); + private AutoPipeline autoPipeline = new AutoPipeline(); + private static final Logger logger = LoggerFactory.getLogger(ApplicationProperties.class); + @Data public static class AutoPipeline { private String outputFolder; + } - public String getOutputFolder() { - return outputFolder; - } - - public void setOutputFolder(String outputFolder) { - this.outputFolder = outputFolder; - } - - @Override - public String toString() { - return "AutoPipeline [outputFolder=" + outputFolder + "]"; - } + @Data + public static class Legal { + private String termsAndConditions; + private String privacyPolicy; + private String accessibilityStatement; + private String cookiePolicy; + private String impressum; } + @Data public static class Security { private Boolean enableLogin; private Boolean csrfDisabled; - private InitialLogin initialLogin; - private OAUTH2 oauth2; + private InitialLogin initialLogin = new InitialLogin(); + private OAUTH2 oauth2 = new OAUTH2(); private int loginAttemptCount; private long loginResetTimeMinutes; private String loginMethod = "all"; - public String getLoginMethod() { - return loginMethod; - } - - public void setLoginMethod(String loginMethod) { - this.loginMethod = loginMethod; - } - - public int getLoginAttemptCount() { - return loginAttemptCount; - } - - public void setLoginAttemptCount(int loginAttemptCount) { - this.loginAttemptCount = loginAttemptCount; - } - - public long getLoginResetTimeMinutes() { - return loginResetTimeMinutes; - } - - public void setLoginResetTimeMinutes(long loginResetTimeMinutes) { - this.loginResetTimeMinutes = loginResetTimeMinutes; - } - - public InitialLogin getInitialLogin() { - return initialLogin != null ? initialLogin : new InitialLogin(); - } - - public void setInitialLogin(InitialLogin initialLogin) { - this.initialLogin = initialLogin; - } - - public OAUTH2 getOAUTH2() { - return oauth2 != null ? oauth2 : new OAUTH2(); - } - - public void setOAUTH2(OAUTH2 oauth2) { - this.oauth2 = oauth2; - } - - public Boolean getEnableLogin() { - return enableLogin; - } - - public void setEnableLogin(Boolean enableLogin) { - this.enableLogin = enableLogin; - } - - public Boolean getCsrfDisabled() { - return csrfDisabled; - } - - public void setCsrfDisabled(Boolean csrfDisabled) { - this.csrfDisabled = csrfDisabled; - } - - @Override - public String toString() { - return "Security [enableLogin=" - + enableLogin - + ", oauth2=" - + oauth2 - + ", initialLogin=" - + initialLogin - + ", csrfDisabled=" - + csrfDisabled - + ", loginMethod=" - + loginMethod - + "]"; - } - + @Data public static class InitialLogin { private String username; - private String password; - - public String getUsername() { - return username; - } - - public void setUsername(String username) { - this.username = username; - } - - public String getPassword() { - return password; - } - - public void setPassword(String password) { - this.password = password; - } - - @Override - public String toString() { - return "InitialLogin [username=" - + username - + ", password=" - + (password != null && !password.isEmpty() ? "MASKED" : "NULL") - + "]"; - } + @ToString.Exclude private String password; } + @Data public static class OAUTH2 { private Boolean enabled = false; private String issuer; private String clientId; - private String clientSecret; + @ToString.Exclude private String clientSecret; private Boolean autoCreateUser = false; private Boolean blockRegistration = false; private String useAsUsername; @@ -247,74 +80,6 @@ public static class OAUTH2 { private String provider; private Client client = new Client(); - public Boolean getEnabled() { - return enabled; - } - - public void setEnabled(Boolean enabled) { - this.enabled = enabled; - } - - public String getIssuer() { - return issuer; - } - - public void setIssuer(String issuer) { - this.issuer = issuer; - } - - public String getClientId() { - return clientId; - } - - public void setClientId(String clientId) { - this.clientId = clientId; - } - - public String getClientSecret() { - return clientSecret; - } - - public void setClientSecret(String clientSecret) { - this.clientSecret = clientSecret; - } - - public Boolean getAutoCreateUser() { - return autoCreateUser; - } - - public void setAutoCreateUser(Boolean autoCreateUser) { - this.autoCreateUser = autoCreateUser; - } - - public Boolean getBlockRegistration() { - return blockRegistration; - } - - public void setBlockRegistration(Boolean blockRegistration) { - this.blockRegistration = blockRegistration; - } - - public String getUseAsUsername() { - return useAsUsername; - } - - public void setUseAsUsername(String useAsUsername) { - this.useAsUsername = useAsUsername; - } - - public String getProvider() { - return provider; - } - - public void setProvider(String provider) { - this.provider = provider; - } - - public Collection getScopes() { - return scopes; - } - public void setScopes(String scopes) { List scopesList = Arrays.stream(scopes.split(",")) @@ -323,26 +88,12 @@ public void setScopes(String scopes) { this.scopes.addAll(scopesList); } - public Client getClient() { - return client; - } - - public void setClient(Client client) { - this.client = client; - } - protected boolean isValid(String value, String name) { - if (value != null && !value.trim().isEmpty()) { - return true; - } - return false; + return value != null && !value.trim().isEmpty(); } protected boolean isValid(Collection value, String name) { - if (value != null && !value.isEmpty()) { - return true; - } - return false; + return value != null && !value.isEmpty(); } public boolean isSettingsValid() { @@ -353,31 +104,7 @@ && isValid(this.getScopes(), "scopes") && isValid(this.getUseAsUsername(), "useAsUsername"); } - @Override - public String toString() { - return "OAUTH2 [enabled=" - + enabled - + ", issuer=" - + issuer - + ", clientId=" - + clientId - + ", clientSecret=" - + (clientSecret != null && !clientSecret.isEmpty() ? "MASKED" : "NULL") - + ", autoCreateUser=" - + autoCreateUser - + ", blockRegistration=" - + blockRegistration - + ", useAsUsername=" - + useAsUsername - + ", provider=" - + provider - + ", client=" - + client - + ", scopes=" - + scopes - + "]"; - } - + @Data public static class Client { private GoogleProvider google = new GoogleProvider(); private GithubProvider github = new GithubProvider(); @@ -392,50 +119,15 @@ public Provider get(String registrationId) throws UnsupportedProviderException { case "keycloak": return getKeycloak(); default: - break; + throw new UnsupportedProviderException( + "Logout from the provider is not supported? Report it at https://github.com/Stirling-Tools/Stirling-PDF/issues"); } - throw new UnsupportedProviderException( - "Logout from the provider is not supported? Report it at https://github.com/Stirling-Tools/Stirling-PDF/issues"); - } - - public GoogleProvider getGoogle() { - return google; - } - - public void setGoogle(GoogleProvider google) { - this.google = google; - } - - public GithubProvider getGithub() { - return github; - } - - public void setGithub(GithubProvider github) { - this.github = github; - } - - public KeycloakProvider getKeycloak() { - return keycloak; - } - - public void setKeycloak(KeycloakProvider keycloak) { - this.keycloak = keycloak; - } - - @Override - public String toString() { - return "Client [google=" - + google - + ", github=" - + github - + ", keycloak=" - + keycloak - + "]"; } } } } + @Data public static class System { private String defaultLocale; private Boolean googlevisibility; @@ -443,184 +135,67 @@ public static class System { private Boolean showUpdateOnlyAdmin; private boolean customHTMLFiles; private String tessdataDir; - - public String getTessdataDir() { - return tessdataDir; - } - - public void setTessdataDir(String tessdataDir) { - this.tessdataDir = tessdataDir; - } - - public boolean isCustomHTMLFiles() { - return customHTMLFiles; - } - - public void setCustomHTMLFiles(boolean customHTMLFiles) { - this.customHTMLFiles = customHTMLFiles; - } - - public boolean getShowUpdateOnlyAdmin() { - return showUpdateOnlyAdmin; - } - - public void setShowUpdateOnlyAdmin(boolean showUpdateOnlyAdmin) { - this.showUpdateOnlyAdmin = showUpdateOnlyAdmin; - } - - public boolean getShowUpdate() { - return showUpdate; - } - - public void setShowUpdate(boolean showUpdate) { - this.showUpdate = showUpdate; - } - private Boolean enableAlphaFunctionality; - - public Boolean getEnableAlphaFunctionality() { - return enableAlphaFunctionality; - } - - public void setEnableAlphaFunctionality(Boolean enableAlphaFunctionality) { - this.enableAlphaFunctionality = enableAlphaFunctionality; - } - - public String getDefaultLocale() { - return defaultLocale; - } - - public void setDefaultLocale(String defaultLocale) { - this.defaultLocale = defaultLocale; - } - - public Boolean getGooglevisibility() { - return googlevisibility; - } - - public void setGooglevisibility(Boolean googlevisibility) { - this.googlevisibility = googlevisibility; - } - - @Override - public String toString() { - return "System [defaultLocale=" - + defaultLocale - + ", googlevisibility=" - + googlevisibility - + ", enableAlphaFunctionality=" - + enableAlphaFunctionality - + ", showUpdate=" - + showUpdate - + ", showUpdateOnlyAdmin=" - + showUpdateOnlyAdmin - + "]"; - } } + @Data public static class Ui { private String appName; private String homeDescription; private String appNameNavbar; public String getAppName() { - if (appName != null && appName.trim().length() == 0) return null; - return appName; - } - - public void setAppName(String appName) { - this.appName = appName; + return appName != null && appName.trim().length() > 0 ? appName : null; } public String getHomeDescription() { - if (homeDescription != null && homeDescription.trim().length() == 0) return null; - return homeDescription; - } - - public void setHomeDescription(String homeDescription) { - this.homeDescription = homeDescription; + return homeDescription != null && homeDescription.trim().length() > 0 + ? homeDescription + : null; } public String getAppNameNavbar() { - if (appNameNavbar != null && appNameNavbar.trim().length() == 0) return null; - return appNameNavbar; - } - - public void setAppNameNavbar(String appNameNavbar) { - this.appNameNavbar = appNameNavbar; - } - - @Override - public String toString() { - return "UserInterface [appName=" - + appName - + ", homeDescription=" - + homeDescription - + ", appNameNavbar=" - + appNameNavbar - + "]"; + return appNameNavbar != null && appNameNavbar.trim().length() > 0 + ? appNameNavbar + : null; } } + @Data public static class Endpoints { private List toRemove; private List groupsToRemove; - - public List getToRemove() { - return toRemove; - } - - public void setToRemove(List toRemove) { - this.toRemove = toRemove; - } - - public List getGroupsToRemove() { - return groupsToRemove; - } - - public void setGroupsToRemove(List groupsToRemove) { - this.groupsToRemove = groupsToRemove; - } - - @Override - public String toString() { - return "Endpoints [toRemove=" + toRemove + ", groupsToRemove=" + groupsToRemove + "]"; - } } + @Data public static class Metrics { private Boolean enabled; - - public Boolean getEnabled() { - return enabled; - } - - public void setEnabled(Boolean enabled) { - this.enabled = enabled; - } - - @Override - public String toString() { - return "Metrics [enabled=" + enabled + "]"; - } } + @Data public static class AutomaticallyGenerated { - private String key; + @ToString.Exclude private String key; + } - public String getKey() { - return key; - } + @Data + public static class EnterpriseEdition { + @ToString.Exclude private String key; + private CustomMetadata customMetadata = new CustomMetadata(); - public void setKey(String key) { - this.key = key; - } + @Data + public static class CustomMetadata { + private boolean autoUpdateMetadata; + private String author; + private String creator; + private String producer; - @Override - public String toString() { - return "AutomaticallyGenerated [key=" - + (key != null && !key.isEmpty() ? "MASKED" : "NULL") - + "]"; + public String getCreator() { + return creator == null || creator.trim().isEmpty() ? "Stirling-PDF" : creator; + } + + public String getProducer() { + return producer == null || producer.trim().isEmpty() ? "Stirling-PDF" : producer; + } } } } diff --git a/src/main/java/stirling/software/SPDF/repository/UserRepository.java b/src/main/java/stirling/software/SPDF/repository/UserRepository.java index 4f231e0faec..0f5387f79ba 100644 --- a/src/main/java/stirling/software/SPDF/repository/UserRepository.java +++ b/src/main/java/stirling/software/SPDF/repository/UserRepository.java @@ -16,7 +16,6 @@ public interface UserRepository extends JpaRepository { @Query("FROM User u LEFT JOIN FETCH u.settings where upper(u.username) = upper(:username)") Optional findByUsernameIgnoreCaseWithSettings(@Param("username") String username); - Optional findByUsername(String username); Optional findByApiKey(String apiKey); diff --git a/src/main/java/stirling/software/SPDF/utils/PdfUtils.java b/src/main/java/stirling/software/SPDF/utils/PdfUtils.java index c1589902958..160e01da407 100644 --- a/src/main/java/stirling/software/SPDF/utils/PdfUtils.java +++ b/src/main/java/stirling/software/SPDF/utils/PdfUtils.java @@ -6,7 +6,6 @@ import java.io.ByteArrayOutputStream; import java.io.IOException; import java.util.ArrayList; -import java.util.Calendar; import java.util.HashMap; import java.util.List; import java.util.zip.ZipEntry; @@ -37,8 +36,6 @@ import io.github.pixee.security.Filenames; -import stirling.software.SPDF.model.PdfMetadata; - public class PdfUtils { private static final Logger logger = LoggerFactory.getLogger(PdfUtils.class); @@ -506,30 +503,6 @@ public static byte[] overlayImage( return baos.toByteArray(); } - public static PdfMetadata extractMetadataFromPdf(PDDocument pdf) { - return PdfMetadata.builder() - .author(pdf.getDocumentInformation().getAuthor()) - .producer(pdf.getDocumentInformation().getProducer()) - .title(pdf.getDocumentInformation().getTitle()) - .creator(pdf.getDocumentInformation().getCreator()) - .subject(pdf.getDocumentInformation().getSubject()) - .keywords(pdf.getDocumentInformation().getKeywords()) - .creationDate(pdf.getDocumentInformation().getCreationDate()) - .modificationDate(pdf.getDocumentInformation().getModificationDate()) - .build(); - } - - public static void setMetadataToPdf(PDDocument pdf, PdfMetadata pdfMetadata) { - pdf.getDocumentInformation().setAuthor(pdfMetadata.getAuthor()); - pdf.getDocumentInformation().setProducer(pdfMetadata.getProducer()); - pdf.getDocumentInformation().setTitle(pdfMetadata.getTitle()); - pdf.getDocumentInformation().setCreator(pdfMetadata.getCreator()); - pdf.getDocumentInformation().setSubject(pdfMetadata.getSubject()); - pdf.getDocumentInformation().setKeywords(pdfMetadata.getKeywords()); - pdf.getDocumentInformation().setCreationDate(pdfMetadata.getCreationDate()); - pdf.getDocumentInformation().setModificationDate(Calendar.getInstance()); - } - /** Key for storing the dimensions of a rendered image in a map. */ private record PdfRenderSettingsKey(float mediaBoxWidth, float mediaBoxHeight, int rotation) {} diff --git a/src/main/java/stirling/software/SPDF/utils/RequestUriUtils.java b/src/main/java/stirling/software/SPDF/utils/RequestUriUtils.java index a9c404e1c28..7aeab8e177f 100644 --- a/src/main/java/stirling/software/SPDF/utils/RequestUriUtils.java +++ b/src/main/java/stirling/software/SPDF/utils/RequestUriUtils.java @@ -12,6 +12,7 @@ public static boolean isStaticResource(String contextPath, String requestURI) { return requestURI.startsWith(contextPath + "/css/") || requestURI.startsWith(contextPath + "/fonts/") || requestURI.startsWith(contextPath + "/js/") + || requestURI.endsWith(contextPath + "robots.txt") || requestURI.startsWith(contextPath + "/images/") || requestURI.startsWith(contextPath + "/public/") || requestURI.startsWith(contextPath + "/pdfjs/") @@ -22,4 +23,26 @@ public static boolean isStaticResource(String contextPath, String requestURI) { || requestURI.endsWith(".webmanifest") || requestURI.startsWith(contextPath + "/api/v1/info/status"); } + + public static boolean isTrackableResource(String requestURI) { + return isTrackableResource("", requestURI); + } + + public static boolean isTrackableResource(String contextPath, String requestURI) { + return !(requestURI.startsWith("/js") + || requestURI.startsWith("/v1/api-docs") + || requestURI.endsWith("robots.txt") + || requestURI.startsWith("/images") + || requestURI.endsWith(".png") + || requestURI.endsWith(".ico") + || requestURI.endsWith(".css") + || requestURI.endsWith(".map") + || requestURI.endsWith(".svg") + || requestURI.endsWith(".js") + || requestURI.contains("swagger") + || requestURI.startsWith("/api/v1/info") + || requestURI.startsWith("/site.webmanifest") + || requestURI.startsWith("/fonts") + || requestURI.startsWith("/pdfjs")); + } } diff --git a/src/main/resources/messages_ar_AR.properties b/src/main/resources/messages_ar_AR.properties index 4b19d9413bc..edf93176be0 100644 --- a/src/main/resources/messages_ar_AR.properties +++ b/src/main/resources/messages_ar_AR.properties @@ -77,7 +77,11 @@ color=لون sponsor=راعٍ info=معلومات - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_bg_BG.properties b/src/main/resources/messages_bg_BG.properties index 442f8789d46..859fb0a2c47 100644 --- a/src/main/resources/messages_bg_BG.properties +++ b/src/main/resources/messages_bg_BG.properties @@ -77,7 +77,11 @@ color=Цвят sponsor=Спонсор info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ca_CA.properties b/src/main/resources/messages_ca_CA.properties index 4673d2d46be..afa0eb1d1ca 100644 --- a/src/main/resources/messages_ca_CA.properties +++ b/src/main/resources/messages_ca_CA.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_cs_CZ.properties b/src/main/resources/messages_cs_CZ.properties index a5396d3905a..ef0cdb26479 100644 --- a/src/main/resources/messages_cs_CZ.properties +++ b/src/main/resources/messages_cs_CZ.properties @@ -77,7 +77,11 @@ color=Barva sponsor=Sponzor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_da_DK.properties b/src/main/resources/messages_da_DK.properties index 198c9d821bc..1260d104a99 100644 --- a/src/main/resources/messages_da_DK.properties +++ b/src/main/resources/messages_da_DK.properties @@ -77,7 +77,11 @@ color=Farve sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_de_DE.properties b/src/main/resources/messages_de_DE.properties index a903444515c..ceaf679cad0 100644 --- a/src/main/resources/messages_de_DE.properties +++ b/src/main/resources/messages_de_DE.properties @@ -77,7 +77,11 @@ color=Farbe sponsor=Sponsor info=Informationen - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_el_GR.properties b/src/main/resources/messages_el_GR.properties index ba1c4e57bd1..6e4c1df4b95 100644 --- a/src/main/resources/messages_el_GR.properties +++ b/src/main/resources/messages_el_GR.properties @@ -77,7 +77,11 @@ color=Χρώμα sponsor=Yποστηρικτής info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_en_GB.properties b/src/main/resources/messages_en_GB.properties index 46a28f6e478..206fae045b4 100644 --- a/src/main/resources/messages_en_GB.properties +++ b/src/main/resources/messages_en_GB.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_en_US.properties b/src/main/resources/messages_en_US.properties index 58ae0f84551..3fcfa3d5b6d 100644 --- a/src/main/resources/messages_en_US.properties +++ b/src/main/resources/messages_en_US.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_es_ES.properties b/src/main/resources/messages_es_ES.properties index cd28db08036..31dcacd51f4 100644 --- a/src/main/resources/messages_es_ES.properties +++ b/src/main/resources/messages_es_ES.properties @@ -77,7 +77,11 @@ color=Color sponsor=Patrocinador info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_eu_ES.properties b/src/main/resources/messages_eu_ES.properties index 5a297eac7b6..1ebbf542ed0 100644 --- a/src/main/resources/messages_eu_ES.properties +++ b/src/main/resources/messages_eu_ES.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_fr_FR.properties b/src/main/resources/messages_fr_FR.properties index f51e31d2b75..b563103ea4c 100644 --- a/src/main/resources/messages_fr_FR.properties +++ b/src/main/resources/messages_fr_FR.properties @@ -77,7 +77,11 @@ color=Couleur sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ga_IE.properties b/src/main/resources/messages_ga_IE.properties index 65b90cfaccf..8223da8afb5 100644 --- a/src/main/resources/messages_ga_IE.properties +++ b/src/main/resources/messages_ga_IE.properties @@ -77,7 +77,11 @@ color=Dath sponsor=Urraitheoir info=Eolas - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_hi_IN.properties b/src/main/resources/messages_hi_IN.properties index 24e68fbd8ab..4c7137a1a1b 100644 --- a/src/main/resources/messages_hi_IN.properties +++ b/src/main/resources/messages_hi_IN.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_hr_HR.properties b/src/main/resources/messages_hr_HR.properties index d669f1f7749..980bda1a9ee 100644 --- a/src/main/resources/messages_hr_HR.properties +++ b/src/main/resources/messages_hr_HR.properties @@ -77,7 +77,11 @@ color=Boja sponsor=Sponzor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_hu_HU.properties b/src/main/resources/messages_hu_HU.properties index 417c39097c5..11f1d8bace9 100644 --- a/src/main/resources/messages_hu_HU.properties +++ b/src/main/resources/messages_hu_HU.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_id_ID.properties b/src/main/resources/messages_id_ID.properties index 109b57c0d7c..dba627cd9a0 100644 --- a/src/main/resources/messages_id_ID.properties +++ b/src/main/resources/messages_id_ID.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_it_IT.properties b/src/main/resources/messages_it_IT.properties index b680c4cb6d0..58505fcd165 100644 --- a/src/main/resources/messages_it_IT.properties +++ b/src/main/resources/messages_it_IT.properties @@ -77,7 +77,11 @@ color=Colore sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ja_JP.properties b/src/main/resources/messages_ja_JP.properties index b7e64251179..c0a441b8dfd 100644 --- a/src/main/resources/messages_ja_JP.properties +++ b/src/main/resources/messages_ja_JP.properties @@ -77,7 +77,11 @@ color=色 sponsor=スポンサー info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ko_KR.properties b/src/main/resources/messages_ko_KR.properties index 3110922ad6d..3c47f3c5e36 100644 --- a/src/main/resources/messages_ko_KR.properties +++ b/src/main/resources/messages_ko_KR.properties @@ -77,7 +77,11 @@ color=색상 sponsor=스폰서 info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_nl_NL.properties b/src/main/resources/messages_nl_NL.properties index b0ed1ca8cd7..643e9ebef69 100644 --- a/src/main/resources/messages_nl_NL.properties +++ b/src/main/resources/messages_nl_NL.properties @@ -77,7 +77,11 @@ color=Kleur sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_no_NB.properties b/src/main/resources/messages_no_NB.properties index a9794fbb6f0..73309cb2285 100644 --- a/src/main/resources/messages_no_NB.properties +++ b/src/main/resources/messages_no_NB.properties @@ -77,7 +77,11 @@ color=Farge sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_pl_PL.properties b/src/main/resources/messages_pl_PL.properties index 7c4556abb93..0800397aa55 100755 --- a/src/main/resources/messages_pl_PL.properties +++ b/src/main/resources/messages_pl_PL.properties @@ -77,7 +77,11 @@ color=kolor sponsor=sponsor info=informacje - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_pt_BR.properties b/src/main/resources/messages_pt_BR.properties index 4f3804c0cfe..f6e93ba2562 100644 --- a/src/main/resources/messages_pt_BR.properties +++ b/src/main/resources/messages_pt_BR.properties @@ -77,7 +77,11 @@ color=Cor sponsor=Patrocine info=Informações - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_pt_PT.properties b/src/main/resources/messages_pt_PT.properties index 9d13fd99d54..bce2fb4524e 100644 --- a/src/main/resources/messages_pt_PT.properties +++ b/src/main/resources/messages_pt_PT.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ro_RO.properties b/src/main/resources/messages_ro_RO.properties index e4d96c2551a..7c800454954 100644 --- a/src/main/resources/messages_ro_RO.properties +++ b/src/main/resources/messages_ro_RO.properties @@ -77,7 +77,11 @@ color=Culoare sponsor=Sponsor info=Informații - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_ru_RU.properties b/src/main/resources/messages_ru_RU.properties index 4fd23c09ff9..4c02d9349a9 100644 --- a/src/main/resources/messages_ru_RU.properties +++ b/src/main/resources/messages_ru_RU.properties @@ -77,7 +77,11 @@ color=Цвет sponsor=Спонсор info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_sk_SK.properties b/src/main/resources/messages_sk_SK.properties index 3452d732316..968f29a85b1 100644 --- a/src/main/resources/messages_sk_SK.properties +++ b/src/main/resources/messages_sk_SK.properties @@ -77,7 +77,11 @@ color=Farba sponsor=Sponzorovať info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_sr_LATN_RS.properties b/src/main/resources/messages_sr_LATN_RS.properties index 53226f250fd..6744321fa53 100644 --- a/src/main/resources/messages_sr_LATN_RS.properties +++ b/src/main/resources/messages_sr_LATN_RS.properties @@ -77,7 +77,11 @@ color=Color sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_sv_SE.properties b/src/main/resources/messages_sv_SE.properties index 3d159d3a5bf..9a6232b4183 100644 --- a/src/main/resources/messages_sv_SE.properties +++ b/src/main/resources/messages_sv_SE.properties @@ -77,7 +77,11 @@ color=Färg sponsor=Sponsor info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_th_TH.properties b/src/main/resources/messages_th_TH.properties index ae82a70fa17..c8e6232e9d1 100644 --- a/src/main/resources/messages_th_TH.properties +++ b/src/main/resources/messages_th_TH.properties @@ -77,7 +77,11 @@ color=สี sponsor=ผู้สนับสนุน info=ข้อมูล - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_tr_TR.properties b/src/main/resources/messages_tr_TR.properties index c5d72ccf8e2..0dd75373fcc 100644 --- a/src/main/resources/messages_tr_TR.properties +++ b/src/main/resources/messages_tr_TR.properties @@ -77,7 +77,11 @@ color=Renk sponsor=Bağış info=Bilgi - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_uk_UA.properties b/src/main/resources/messages_uk_UA.properties index 41ca0038597..e91a09fd575 100644 --- a/src/main/resources/messages_uk_UA.properties +++ b/src/main/resources/messages_uk_UA.properties @@ -77,7 +77,11 @@ color=Колір sponsor=Спонсор info=Інформація - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_vi_VN.properties b/src/main/resources/messages_vi_VN.properties index 28ef3fa182e..6d417a96929 100644 --- a/src/main/resources/messages_vi_VN.properties +++ b/src/main/resources/messages_vi_VN.properties @@ -77,7 +77,11 @@ color=Màu sắc sponsor=Nhà tài trợ info=Thông tin - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_zh_CN.properties b/src/main/resources/messages_zh_CN.properties index e0f312cfd9d..a1c2059ad5a 100644 --- a/src/main/resources/messages_zh_CN.properties +++ b/src/main/resources/messages_zh_CN.properties @@ -77,7 +77,11 @@ color=颜色 sponsor=赞助 info=信息 - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/messages_zh_TW.properties b/src/main/resources/messages_zh_TW.properties index 096a5625372..6b523d1340d 100644 --- a/src/main/resources/messages_zh_TW.properties +++ b/src/main/resources/messages_zh_TW.properties @@ -77,7 +77,11 @@ color=顏色 sponsor=贊助 info=Info - +legal.privacy=Privacy Policy +legal.terms=Terms and Conditions +legal.accessibility=Accessibility +legal.cookie=Cookie Policy +legal.impressum=Impressum ############### # Pipeline # diff --git a/src/main/resources/settings.yml.template b/src/main/resources/settings.yml.template index 2e06ea9f853..2a5400b6018 100644 --- a/src/main/resources/settings.yml.template +++ b/src/main/resources/settings.yml.template @@ -48,6 +48,22 @@ security: scopes: openid, profile, email # Specify the scopes for which the application will request permissions provider: google # Set this to your OAuth provider's name, e.g., 'google' or 'keycloak' +# Enterprise edition settings unused for now please ignore! +EnterpriseEdition: + key: 00000000-0000-0000-0000-000000000000 + CustomMetadata: + autoUpdateMetadata: true # set to 'true' to automatically update metadata with below values + author: username # Supports text such as 'John Doe' or types such as username + creator: Stirling-PDF # Supports text such as 'Company-PDF' + producer: Stirling-PDF # Supports text such as 'Company-PDF' + +legal: + termsAndConditions: '' # URL to the terms and conditions of your application (e.g. https://example.com/terms) Empty string to disable or filename to load from local file in static folder + privacyPolicy: '' # URL to the privacy policy of your application (e.g. https://example.com/privacy) Empty string to disable or filename to load from local file in static folder + accessibilityStatement: '' # URL to the accessibility statement of your application (e.g. https://example.com/accessibility) Empty string to disable or filename to load from local file in static folder + cookiePolicy: '' # URL to the cookie policy of your application (e.g. https://example.com/cookie) Empty string to disable or filename to load from local file in static folder + impressum: '' # URL to the impressum of your application (e.g. https://example.com/impressum) Empty string to disable or filename to load from local file in static folder + system: defaultLocale: en-US # Set the default language (e.g. 'de-DE', 'fr-FR', etc) googlevisibility: false # 'true' to allow Google visibility (via robots.txt), 'false' to disallow diff --git a/src/main/resources/static/js/multitool/PdfContainer.js b/src/main/resources/static/js/multitool/PdfContainer.js index 0654cac9305..55635c69e8c 100644 --- a/src/main/resources/static/js/multitool/PdfContainer.js +++ b/src/main/resources/static/js/multitool/PdfContainer.js @@ -446,7 +446,7 @@ function detectImageType(uint8Array) { // Check for TIFF signature (little-endian and big-endian) if ((uint8Array[0] === 73 && uint8Array[1] === 73 && uint8Array[2] === 42 && uint8Array[3] === 0) || - (uint8Array[0] === 77 && uint8Array[1] === 77 && uint8Array[2] === 0 && uint8Array[3] === 42)) { + (uint8Array[0] === 77 && uint8Array[1] === 77 && uint8Array[2] === 0 && uint8Array[3] === 42)) { return 'TIFF'; } diff --git a/src/main/resources/templates/fragments/common.html b/src/main/resources/templates/fragments/common.html index 59a3b5f4b91..96d52157ee1 100644 --- a/src/main/resources/templates/fragments/common.html +++ b/src/main/resources/templates/fragments/common.html @@ -24,7 +24,7 @@ - + @@ -35,10 +35,10 @@ - + - + @@ -58,17 +58,21 @@ - + + + + + - + - + diff --git a/src/main/resources/templates/fragments/footer.html b/src/main/resources/templates/fragments/footer.html index 7d0c1a5484f..40f64419a3f 100644 --- a/src/main/resources/templates/fragments/footer.html +++ b/src/main/resources/templates/fragments/footer.html @@ -5,6 +5,11 @@ diff --git a/src/main/resources/templates/misc/add-page-numbers.html b/src/main/resources/templates/misc/add-page-numbers.html index eb9a87de7be..186253b6945 100644 --- a/src/main/resources/templates/misc/add-page-numbers.html +++ b/src/main/resources/templates/misc/add-page-numbers.html @@ -94,10 +94,9 @@
diff --git a/src/main/resources/templates/multi-tool.html b/src/main/resources/templates/multi-tool.html index 9f165014925..61fcc28e755 100644 --- a/src/main/resources/templates/multi-tool.html +++ b/src/main/resources/templates/multi-tool.html @@ -110,4 +110,4 @@ - \ No newline at end of file + diff --git a/src/test/java/stirling/software/SPDF/utils/PdfUtilsTest.java b/src/test/java/stirling/software/SPDF/utils/PdfUtilsTest.java index 03a52663c01..a650e891800 100644 --- a/src/test/java/stirling/software/SPDF/utils/PdfUtilsTest.java +++ b/src/test/java/stirling/software/SPDF/utils/PdfUtilsTest.java @@ -49,12 +49,5 @@ void testHasImagesOnPage() throws IOException { assertTrue(PdfUtils.hasImagesOnPage(page)); } - @Test - void testExtractMetadataFromPdf() throws IOException { - PDDocument document = Mockito.mock(PDDocument.class); - Mockito.when(document.getDocumentInformation()).thenReturn(Mockito.mock(org.apache.pdfbox.pdmodel.PDDocumentInformation.class)); - PdfMetadata metadata = PdfUtils.extractMetadataFromPdf(document); - assertNotNull(metadata); - } } From 8788a7ee343fb6daa5592f44a8cb54e0fc509831 Mon Sep 17 00:00:00 2001 From: albanobattistella <34811668+albanobattistella@users.noreply.github.com> Date: Fri, 13 Sep 2024 19:17:19 +0200 Subject: [PATCH 3/5] Update messages_it_IT.properties (#1891) --- src/main/resources/messages_it_IT.properties | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/main/resources/messages_it_IT.properties b/src/main/resources/messages_it_IT.properties index 58505fcd165..9798fb348c4 100644 --- a/src/main/resources/messages_it_IT.properties +++ b/src/main/resources/messages_it_IT.properties @@ -77,10 +77,10 @@ color=Colore sponsor=Sponsor info=Info -legal.privacy=Privacy Policy -legal.terms=Terms and Conditions -legal.accessibility=Accessibility -legal.cookie=Cookie Policy +legal.privacy=Informativa sulla privacy +legal.terms=Termini e Condizioni +legal.accessibility=Accessibilità +legal.cookie=Informativa sui cookie legal.impressum=Impressum ############### From b2862a3fc404a54cb3bbb8ee87f5f93b3dd8e617 Mon Sep 17 00:00:00 2001 From: Charan19001A0231 Date: Sat, 14 Sep 2024 15:16:16 +0530 Subject: [PATCH 4/5] Update add-watermark.html (#1893) shifted add watermark submit button to left --- src/main/resources/templates/security/add-watermark.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/main/resources/templates/security/add-watermark.html b/src/main/resources/templates/security/add-watermark.html index 013316b1943..d6614e85c93 100644 --- a/src/main/resources/templates/security/add-watermark.html +++ b/src/main/resources/templates/security/add-watermark.html @@ -108,7 +108,7 @@
-
+
@@ -144,4 +144,4 @@
- \ No newline at end of file + From bb1c859e0d8cf8dad043822923766df54d41ac46 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Sat, 14 Sep 2024 10:46:25 +0100 Subject: [PATCH 5/5] :memo: Update README: Translation Progress Table (#1890) :memo: Sync README > Made via sync_files.yml Co-authored-by: github-actions[bot] --- README.md | 38 +++++++++++++++++++------------------- 1 file changed, 19 insertions(+), 19 deletions(-) diff --git a/README.md b/README.md index f56fa5cd4e0..7773e738082 100644 --- a/README.md +++ b/README.md @@ -174,35 +174,35 @@ Stirling PDF currently supports 38! | ------------------------------------------- | -------------------------------------- | | Arabic (العربية) (ar_AR) | ![99%](https://geps.dev/progress/99) | | Basque (Euskara) (eu_ES) | ![60%](https://geps.dev/progress/60) | -| Bulgarian (Български) (bg_BG) | ![92%](https://geps.dev/progress/92) | +| Bulgarian (Български) (bg_BG) | ![91%](https://geps.dev/progress/91) | | Catalan (Català) (ca_CA) | ![47%](https://geps.dev/progress/47) | -| Croatian (Hrvatski) (hr_HR) | ![92%](https://geps.dev/progress/92) | -| Czech (Česky) (cs_CZ) | ![88%](https://geps.dev/progress/88) | -| Danish (Dansk) (da_DK) | ![97%](https://geps.dev/progress/97) | +| Croatian (Hrvatski) (hr_HR) | ![91%](https://geps.dev/progress/91) | +| Czech (Česky) (cs_CZ) | ![87%](https://geps.dev/progress/87) | +| Danish (Dansk) (da_DK) | ![96%](https://geps.dev/progress/96) | | Dutch (Nederlands) (nl_NL) | ![93%](https://geps.dev/progress/93) | | English (English) (en_GB) | ![100%](https://geps.dev/progress/100) | | English (US) (en_US) | ![100%](https://geps.dev/progress/100) | -| French (Français) (fr_FR) | ![91%](https://geps.dev/progress/91) | -| German (Deutsch) (de_DE) | ![99%](https://geps.dev/progress/99) | -| Greek (Ελληνικά) (el_GR) | ![80%](https://geps.dev/progress/80) | +| French (Français) (fr_FR) | ![90%](https://geps.dev/progress/90) | +| German (Deutsch) (de_DE) | ![98%](https://geps.dev/progress/98) | +| Greek (Ελληνικά) (el_GR) | ![79%](https://geps.dev/progress/79) | | Hindi (हिंदी) (hi_IN) | ![76%](https://geps.dev/progress/76) | -| Hungarian (Magyar) (hu_HU) | ![74%](https://geps.dev/progress/74) | +| Hungarian (Magyar) (hu_HU) | ![73%](https://geps.dev/progress/73) | | Indonesia (Bahasa Indonesia) (id_ID) | ![74%](https://geps.dev/progress/74) | -| Irish (Gaeilge) (ga_IE) | ![96%](https://geps.dev/progress/96) | +| Irish (Gaeilge) (ga_IE) | ![95%](https://geps.dev/progress/95) | | Italian (Italiano) (it_IT) | ![99%](https://geps.dev/progress/99) | -| Japanese (日本語) (ja_JP) | ![90%](https://geps.dev/progress/90) | -| Korean (한국어) (ko_KR) | ![82%](https://geps.dev/progress/82) | +| Japanese (日本語) (ja_JP) | ![89%](https://geps.dev/progress/89) | +| Korean (한국어) (ko_KR) | ![81%](https://geps.dev/progress/81) | | Norwegian (Norsk) (no_NB) | ![95%](https://geps.dev/progress/95) | -| Polish (Polski) (pl_PL) | ![90%](https://geps.dev/progress/90) | +| Polish (Polski) (pl_PL) | ![89%](https://geps.dev/progress/89) | | Portuguese (Português) (pt_PT) | ![76%](https://geps.dev/progress/76) | -| Portuguese Brazilian (Português) (pt_BR) | ![99%](https://geps.dev/progress/99) | -| Romanian (Română) (ro_RO) | ![98%](https://geps.dev/progress/98) | -| Russian (Русский) (ru_RU) | ![82%](https://geps.dev/progress/82) | +| Portuguese Brazilian (Português) (pt_BR) | ![98%](https://geps.dev/progress/98) | +| Romanian (Română) (ro_RO) | ![97%](https://geps.dev/progress/97) | +| Russian (Русский) (ru_RU) | ![81%](https://geps.dev/progress/81) | | Serbian Latin alphabet (Srpski) (sr_LATN_RS) | ![76%](https://geps.dev/progress/76) | -| Simplified Chinese (简体中文) (zh_CN) | ![99%](https://geps.dev/progress/99) | -| Slovakian (Slovensky) (sk_SK) | ![90%](https://geps.dev/progress/90) | -| Spanish (Español) (es_ES) | ![99%](https://geps.dev/progress/99) | -| Swedish (Svenska) (sv_SE) | ![98%](https://geps.dev/progress/98) | +| Simplified Chinese (简体中文) (zh_CN) | ![98%](https://geps.dev/progress/98) | +| Slovakian (Slovensky) (sk_SK) | ![89%](https://geps.dev/progress/89) | +| Spanish (Español) (es_ES) | ![98%](https://geps.dev/progress/98) | +| Swedish (Svenska) (sv_SE) | ![97%](https://geps.dev/progress/97) | | Thai (ไทย) (th_TH) | ![96%](https://geps.dev/progress/96) | | Traditional Chinese (繁體中文) (zh_TW) | ![95%](https://geps.dev/progress/95) | | Turkish (Türkçe) (tr_TR) | ![96%](https://geps.dev/progress/96) |