From 38ae04484011a18f418c758eb0a2b2a80abca537 Mon Sep 17 00:00:00 2001
From: Maciej PrzepiĆ³ra <maciej@przepiora.me>
Date: Thu, 17 Jan 2019 11:46:49 +0100
Subject: Fix #11369 (#11403)

Clear contents of iframe clone that is in a Window so that when it's reattached to DOM we don't get 404
---
 .../vaadin/client/ui/window/WindowConnector.java   |  17 +++++
 ...gWindowWithBrowserFrameShouldntGenerate404.java |  70 +++++++++++++++++++++
 uitest/src/main/resources/sample.pdf               | Bin 0 -> 18123 bytes
 ...dowWithBrowserFrameShouldntGenerate404Test.java |  39 ++++++++++++
 4 files changed, 126 insertions(+)
 create mode 100644 uitest/src/main/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404.java
 create mode 100644 uitest/src/main/resources/sample.pdf
 create mode 100644 uitest/src/test/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404Test.java

diff --git a/client/src/main/java/com/vaadin/client/ui/window/WindowConnector.java b/client/src/main/java/com/vaadin/client/ui/window/WindowConnector.java
index aa244c7888..feb34e502e 100644
--- a/client/src/main/java/com/vaadin/client/ui/window/WindowConnector.java
+++ b/client/src/main/java/com/vaadin/client/ui/window/WindowConnector.java
@@ -326,6 +326,23 @@ public class WindowConnector extends AbstractSingleComponentContainerConnector
                 }
                 return newEl;
             }
+
+            if ("iframe".equalsIgnoreCase(old.getTagName())
+                    && old.hasAttribute("src")
+                    && old.getAttribute("src").contains("APP/connector")) {
+                // an iframe reloads when reattached to the DOM, but
+                // unfortunately, when newEl is reattached, server-side
+                // resource (e.g. generated PDF) is already gone, so this will
+                // end up with 404, which might be undesirable.
+                // Instead of the resource that is not available anymore,
+                // let's just use empty iframe.
+                // See
+                // https://github.com/vaadin/framework/issues/11369
+                // for details of the issue this works around
+                Element newEl = old.cloneNode(false).cast();
+                newEl.setAttribute("src", "about:blank");
+                return newEl;
+            }
         }
         Node res = node.cloneNode(false);
         if (node.hasChildNodes()) {
diff --git a/uitest/src/main/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404.java b/uitest/src/main/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404.java
new file mode 100644
index 0000000000..74c7938974
--- /dev/null
+++ b/uitest/src/main/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404.java
@@ -0,0 +1,70 @@
+package com.vaadin.tests.components.window;
+
+import com.vaadin.server.StreamResource;
+import com.vaadin.server.VaadinRequest;
+import com.vaadin.tests.components.AbstractTestUI;
+import com.vaadin.ui.*;
+import org.apache.commons.io.IOUtils;
+
+import java.io.ByteArrayInputStream;
+import java.io.IOException;
+import java.io.InputStream;
+
+public class ClosingWindowWithBrowserFrameShouldntGenerate404
+        extends AbstractTestUI {
+
+    private VerticalLayout layout;
+
+    @Override
+    protected void setup(VaadinRequest request) {
+        layout = new VerticalLayout();
+
+        Button button = new Button("Click Me", e -> showPdfInAWindow());
+        layout.addComponent(button);
+        addComponent(layout);
+    }
+
+    private void showPdfInAWindow() {
+        try {
+            final String fileName = "sample.pdf";
+            final byte[] bytes = IOUtils
+                    .toByteArray(getClass().getResource("/" + fileName));
+            StreamResource.StreamSource source = new StreamResource.StreamSource() {
+                @Override
+                public InputStream getStream() {
+                    return new ByteArrayInputStream(bytes);
+                }
+            };
+
+            StreamResource resource = new StreamResource(source, fileName);
+
+            resource.setMIMEType("application/pdf");
+            resource.getStream().setParameter("Content-Disposition",
+                    "attachment; filename=" + fileName);
+            resource.setCacheTime(-1);
+
+            // Use browser frame
+            BrowserFrame frame = new BrowserFrame();
+            frame.setSizeFull();
+            frame.setSource(resource);
+            frame.setHeight("650px");
+            Window pdfWindow = new Window("Sample PDF");
+            pdfWindow.center();
+            pdfWindow.setModal(true);
+            pdfWindow.setResizable(false);
+            pdfWindow.setHeight("700px");
+            pdfWindow.setWidth("900px");
+            pdfWindow.setContent(frame);
+
+            pdfWindow.addCloseListener(e -> {
+                layout.addComponent(new Label("PDF was sent"));
+
+            });
+
+            UI.getCurrent().addWindow(pdfWindow);
+
+        } catch (IOException e) {
+            e.printStackTrace();
+        }
+    }
+}
diff --git a/uitest/src/main/resources/sample.pdf b/uitest/src/main/resources/sample.pdf
new file mode 100644
index 0000000000..758b45a2be
Binary files /dev/null and b/uitest/src/main/resources/sample.pdf differ
diff --git a/uitest/src/test/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404Test.java b/uitest/src/test/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404Test.java
new file mode 100644
index 0000000000..a0a32cb77b
--- /dev/null
+++ b/uitest/src/test/java/com/vaadin/tests/components/window/ClosingWindowWithBrowserFrameShouldntGenerate404Test.java
@@ -0,0 +1,39 @@
+package com.vaadin.tests.components.window;
+
+import com.vaadin.testbench.elements.ButtonElement;
+import com.vaadin.testbench.elements.LabelElement;
+import com.vaadin.testbench.elements.WindowElement;
+import com.vaadin.tests.tb3.MultiBrowserTest;
+import org.junit.Assert;
+import org.junit.Test;
+import org.openqa.selenium.logging.LogEntry;
+
+import java.util.List;
+
+public class ClosingWindowWithBrowserFrameShouldntGenerate404Test
+        extends MultiBrowserTest {
+
+    private static boolean contains404(LogEntry logEntry) {
+        return logEntry.getMessage().contains("404");
+    }
+
+    @Test
+    public void openWindowWithFrame_closeWindow_no404() {
+        openTestURL();
+        $(ButtonElement.class).first().click();
+
+        $(WindowElement.class).first().close();
+
+        $(LabelElement.class).exists();
+
+        List<LogEntry> logs = getDriver().manage().logs().get("browser")
+                .getAll();
+
+        Assert.assertTrue(theresNoLogWith404In(logs));
+    }
+
+    private boolean theresNoLogWith404In(List<LogEntry> logs) {
+        return !logs.stream().anyMatch(
+                ClosingWindowWithBrowserFrameShouldntGenerate404Test::contains404);
+    }
+}
-- 
cgit v1.2.3