From b1e5e4f4ae4ce40d398d6adf1b6a3d53cf3f7c62 Mon Sep 17 00:00:00 2001
From: Marius Dumitru Florea
Date: Tue, 10 Dec 2024 20:00:55 +0200
Subject: [PATCH] XWIKI-22694: The output of Document#getExternalURL() in the
PDF cover doesn't match the URL used to access XWiki * Set the Forwarded HTTP
header when opening the print preview page in the headless Chrome instance in
order for the external URL to be computed properly.
---
.../browser/AbstractBrowserPDFPrinter.java | 41 ++++++++++++++++++-
.../xwiki/export/pdf/browser/BrowserTab.java | 16 ++++++++
.../pdf/browser/BrowserPDFPrinterTest.java | 29 +++++++++++++
.../export/pdf/internal/chrome/ChromeTab.java | 16 ++++++++
.../xwiki/export/pdf/test/ui/PDFExportIT.java | 22 ++++++----
5 files changed, 115 insertions(+), 9 deletions(-)
diff --git a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/AbstractBrowserPDFPrinter.java b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/AbstractBrowserPDFPrinter.java
index b5a9d092865..3343c0c4175 100644
--- a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/AbstractBrowserPDFPrinter.java
+++ b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/AbstractBrowserPDFPrinter.java
@@ -26,8 +26,10 @@
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
+import java.util.Enumeration;
import java.util.LinkedList;
import java.util.List;
+import java.util.Map;
import java.util.Optional;
import java.util.stream.Stream;
@@ -54,6 +56,10 @@
*/
public abstract class AbstractBrowserPDFPrinter implements PDFPrinter
{
+ private static final String HTTP_HEADER_FORWARDED = "Forwarded";
+
+ private static final String HTTP_HEADER_FORWARDED_FOR = "X-Forwarded-For";
+
@Inject
protected Logger logger;
@@ -75,6 +81,7 @@ public InputStream print(URL printPreviewURL) throws IOException
CookieFilterContext cookieFilterContext = findCookieFilterContext(printPreviewURL, browserTab);
Cookie[] cookies = getCookies(cookieFilterContext);
try {
+ browserTab.setExtraHTTPHeaders(getExtraHTTPHeaders(cookieFilterContext));
if (!browserTab.navigate(cookieFilterContext.getTargetURL(), cookies, true,
this.configuration.getPageReadyTimeout())) {
throw new IOException("Failed to load the print preview URL: " + cookieFilterContext.getTargetURL());
@@ -180,7 +187,7 @@ private Optional getCookieFilterContext(URL targetURL, bool
BrowserTab browserTab)
{
Optional browserIPAddress = isFilterRequired ? getBrowserIPAddress(targetURL, browserTab)
- : Optional.of(StringUtils.defaultString(getRequest().getHeader("X-Forwarded-For")));
+ : Optional.of(StringUtils.defaultString(getRequest().getHeader(HTTP_HEADER_FORWARDED_FOR)));
return browserIPAddress.map(ip -> new CookieFilterContext()
{
@Override
@@ -215,6 +222,38 @@ private Optional getBrowserIPAddress(URL targetURL, BrowserTab browserTa
return Optional.empty();
}
+ private Map> getExtraHTTPHeaders(CookieFilterContext cookieFilterContext)
+ {
+ HttpServletRequest request = getRequest();
+
+ List forwarded = new LinkedList<>();
+ Enumeration forwardedValues = request.getHeaders(HTTP_HEADER_FORWARDED);
+ if (forwardedValues != null) {
+ forwardedValues.asIterator().forEachRemaining(forwarded::add);
+ }
+
+ String forwardedFor = request.getHeader(HTTP_HEADER_FORWARDED_FOR);
+ if (StringUtils.isBlank(forwardedFor)) {
+ forwardedFor = request.getRemoteAddr();
+ }
+
+ String host = request.getHeader("X-Forwarded-Host");
+ if (StringUtils.isBlank(host)) {
+ host = request.getHeader("Host");
+ }
+
+ String protocol = request.getHeader("X-Forwarded-Proto");
+ if (StringUtils.isBlank(protocol)) {
+ protocol = request.getScheme();
+ }
+
+ String lastForwarded = String.format("by=%s;for=%s;host=%s;proto=%s", cookieFilterContext.getBrowserIPAddress(),
+ forwardedFor, host, protocol);
+ forwarded.add(lastForwarded);
+
+ return Map.of(HTTP_HEADER_FORWARDED, forwarded);
+ }
+
@Override
public boolean isAvailable()
{
diff --git a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/BrowserTab.java b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/BrowserTab.java
index 434a0cd54f5..f482c1ea194 100644
--- a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/BrowserTab.java
+++ b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/main/java/org/xwiki/export/pdf/browser/BrowserTab.java
@@ -22,6 +22,8 @@
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
+import java.util.List;
+import java.util.Map;
import javax.servlet.http.Cookie;
@@ -33,6 +35,20 @@
*/
public interface BrowserTab extends AutoCloseable
{
+ /**
+ * Sets the extra HTTP headers to use when requesting web pages in this browser tab.
+ *
+ * @param headers the extra HTTP headers to use when requesting web pages in this browser tab
+ * @since 15.10.16
+ * @since 16.4.6
+ * @since 16.10.2
+ * @since 17.0.0RC1
+ */
+ default void setExtraHTTPHeaders(Map> headers)
+ {
+ // Do nothing by default.
+ }
+
/**
* Navigates to the specified web page, optionally waiting for it to be ready (fully loaded).
*
diff --git a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/test/java/org/xwiki/export/pdf/browser/BrowserPDFPrinterTest.java b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/test/java/org/xwiki/export/pdf/browser/BrowserPDFPrinterTest.java
index 342560b4aeb..9023af677b0 100644
--- a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/test/java/org/xwiki/export/pdf/browser/BrowserPDFPrinterTest.java
+++ b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-api/src/test/java/org/xwiki/export/pdf/browser/BrowserPDFPrinterTest.java
@@ -37,6 +37,7 @@
import java.net.URL;
import java.util.Arrays;
import java.util.List;
+import java.util.Map;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletRequest;
@@ -127,6 +128,10 @@ void print() throws Exception
Cookie[] cookies = new Cookie[] {mock(Cookie.class)};
when(this.request.getCookies()).thenReturn(cookies);
+ when(this.request.getRemoteAddr()).thenReturn("192.168.0.118");
+ when(this.request.getHeader("Host")).thenReturn("external:9293");
+ when(this.request.getScheme()).thenReturn("https");
+
when(this.browserManager.createIncognitoTab()).thenReturn(this.browserTab);
when(this.browserTab.navigate(new URL("http://xwiki-host:9293/xwiki/rest/client?media=json"))).thenReturn(true);
when(this.browserTab.getSource()).thenReturn("{\"ip\":\"172.12.0.3\"}");
@@ -152,6 +157,9 @@ public InputStream answer(InvocationOnMock invocation) throws Throwable
assertEquals("172.12.0.3", this.cookieFilterContextCaptor.getValue().getBrowserIPAddress());
assertEquals(browserPrintPreviewURL, this.cookieFilterContextCaptor.getValue().getTargetURL());
+ verify(this.browserTab).setExtraHTTPHeaders(
+ Map.of("Forwarded", List.of("by=172.12.0.3;for=192.168.0.118;host=external:9293;proto=https")));
+
// Only the configured XWiki URI should be used to get the browser IP address.
verify(this.browserTab, never()).navigate(new URL("http://external:9293/xwiki/rest/client?media=json"));
verify(this.browserTab).close();
@@ -213,6 +221,27 @@ void printWithXWikiSchemeAndPortSpecified() throws Exception
verify(this.browserTab).navigate(new URL("ftp://xwiki-host:8080/xwiki/rest/client?media=json"));
}
+ @Test
+ void printWithReverseProxy() throws Exception
+ {
+ URL printPreviewURL = new URL("http://external:9293/xwiki/bin/export/Some/Page?x=y#z");
+ URL browserPrintPreviewURL = new URL("http://xwiki-host:9293/xwiki/bin/export/Some/Page?x=y#z");
+
+ when(this.cookieFilter.isFilterRequired()).thenReturn(false);
+
+ when(this.request.getHeader("X-Forwarded-For")).thenReturn("192.168.0.117");
+ when(this.request.getHeader("Host")).thenReturn("external:9293");
+ when(this.request.getHeader("X-Forwarded-Proto")).thenReturn("ftp");
+
+ when(this.browserManager.createIncognitoTab()).thenReturn(this.browserTab);
+ when(this.browserTab.navigate(browserPrintPreviewURL, null, true, 30)).thenReturn(true);
+
+ this.printer.print(printPreviewURL);
+
+ verify(this.browserTab).setExtraHTTPHeaders(
+ Map.of("Forwarded", List.of("by=192.168.0.117;for=192.168.0.117;host=external:9293;proto=ftp")));
+ }
+
@Test
void isAvailable()
{
diff --git a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-default/src/main/java/org/xwiki/export/pdf/internal/chrome/ChromeTab.java b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-default/src/main/java/org/xwiki/export/pdf/internal/chrome/ChromeTab.java
index 671fdcbfb94..02fe18da12d 100644
--- a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-default/src/main/java/org/xwiki/export/pdf/internal/chrome/ChromeTab.java
+++ b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-default/src/main/java/org/xwiki/export/pdf/internal/chrome/ChromeTab.java
@@ -25,6 +25,7 @@
import java.nio.charset.StandardCharsets;
import java.util.Collections;
import java.util.List;
+import java.util.Map;
import java.util.Objects;
import java.util.stream.Collectors;
import java.util.stream.Stream;
@@ -32,6 +33,7 @@
import javax.servlet.http.Cookie;
import org.apache.commons.io.IOUtils;
+import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.xwiki.export.pdf.PDFExportConfiguration;
@@ -256,4 +258,18 @@ private void setCookies(Cookie[] servletCookies, URL targetURL)
network.clearBrowserCookies();
network.setCookies(browserCookies);
}
+
+ @Override
+ public void setExtraHTTPHeaders(Map> headers)
+ {
+ LOGGER.debug("Setting extra HTTP headers [{}].", headers);
+ Network network = this.tabDevToolsService.getNetwork();
+ network.enable();
+ // The documentation of setExtraHTTPHeaders is not clear about the type of value we can pass for a header (key).
+ // We tried passing a List and a String[] but in both cases we got an exception: "Invalid header value,
+ // string expected". So we concatenate the values of a header with a comma.
+ Map extraHeaders = headers.entrySet().stream()
+ .collect(Collectors.toMap(Map.Entry::getKey, entry -> StringUtils.join(entry.getValue(), ",")));
+ network.setExtraHTTPHeaders(extraHeaders);
+ }
}
diff --git a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-test/xwiki-platform-export-pdf-test-docker/src/test/it/org/xwiki/export/pdf/test/ui/PDFExportIT.java b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-test/xwiki-platform-export-pdf-test-docker/src/test/it/org/xwiki/export/pdf/test/ui/PDFExportIT.java
index 754883c0e6f..2f7efe4a264 100644
--- a/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-test/xwiki-platform-export-pdf-test-docker/src/test/it/org/xwiki/export/pdf/test/ui/PDFExportIT.java
+++ b/xwiki-platform-core/xwiki-platform-export/xwiki-platform-export-pdf/xwiki-platform-export-pdf-test/xwiki-platform-export-pdf-test-docker/src/test/it/org/xwiki/export/pdf/test/ui/PDFExportIT.java
@@ -19,6 +19,10 @@
*/
package org.xwiki.export.pdf.test.ui;
+import static org.junit.jupiter.api.Assertions.assertEquals;
+import static org.junit.jupiter.api.Assertions.assertFalse;
+import static org.junit.jupiter.api.Assertions.assertTrue;
+
import java.net.URL;
import java.util.Arrays;
import java.util.Collections;
@@ -54,12 +58,9 @@
import org.xwiki.test.docker.junit5.UITest;
import org.xwiki.test.ui.TestUtils;
import org.xwiki.test.ui.po.LiveTableElement;
+import org.xwiki.test.ui.po.SuggestInputElement;
import org.xwiki.test.ui.po.ViewPage;
-import static org.junit.jupiter.api.Assertions.assertEquals;
-import static org.junit.jupiter.api.Assertions.assertFalse;
-import static org.junit.jupiter.api.Assertions.assertTrue;
-
/**
* Tests for PDF export.
*
@@ -296,7 +297,8 @@ void exportWithCustomPDFTemplate(TestUtils setup, TestReference testReference, T
setup.createPage(testReference, "", "").createPage().createPageFromTemplate("My cool template", null, null,
"XWiki.PDFExport.TemplateProvider");
PDFTemplateEditPage templateEditPage = new PDFTemplateEditPage();
- templateEditPage.setCover(templateEditPage.getCover().replace("", "Book: "));
+ templateEditPage.setCover(templateEditPage.getCover().replace("", "Book: ").replace("
",
+ "\n$escapetool.xml($tdoc.externalURL)
"));
templateEditPage
.setTableOfContents(templateEditPage.getTableOfContents().replace("core.pdf.tableOfContents", "Chapters"));
templateEditPage.setHeader(templateEditPage.getHeader().replaceFirst("