Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Enable Chromium to display PDFs in the viewer iframe #924

Merged
merged 1 commit into from
Apr 5, 2023

Conversation

veloman-yunkan
Copy link
Collaborator

Fixes #916

The fix doesn't work for Chromium versions under 91. Frankly I haven't verified that it works for Chromium 91+ because I only have 90.

@@ -412,6 +412,12 @@ ContentResponse::ContentResponse(const std::string& root, bool verbose, const st
m_mimeType(mimetype)
{
add_header(MHD_HTTP_HEADER_CONTENT_TYPE, m_mimeType);
if ( !startsWith(m_mimeType, "application/pdf") ) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@veloman-yunkan Have you tested that this actually catches PDF files? E.g., what if the mimeType isn't set? You might want to check also the file name with a case-insensitive regex like /\.pdf$/i.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Have you tested that this actually catches PDF files?

On the test case that I am using - it does.

@kelson42 kelson42 requested a review from mgautierfr March 29, 2023 15:30
@@ -412,6 +412,12 @@ ContentResponse::ContentResponse(const std::string& root, bool verbose, const st
m_mimeType(mimetype)
{
add_header(MHD_HTTP_HEADER_CONTENT_TYPE, m_mimeType);
if ( !startsWith(m_mimeType, "application/pdf") ) {
add_header("Content-Security-Policy",
"default-src 'self' data: blob: about: chrome-extension: 'unsafe-inline' 'unsafe-eval'; "
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

chrome-extension: would only be needed if you were planning to package Kiwix Serve inside an extension or inside an Electron App (the latter, at least, would be possible, but probably using a node-based server). Seems unlikely, so this could be removed.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for the explanation. I thought the meaning of chrome-extension: was to allow running the PDF viewer (which is an browser extension under Chrome) in the iframe.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@veloman-yunkan I've just tested that in a local copy of Kiwix JS by removing chrome-extension: here and in the meta http-equiv CSP, and it doesn't affect anything even in Chromium 90 (PDFs still load in iframe). In any case, the CSP header here won't be served if the requested file is a PDF. I mean, there is no harm in keeping these extension schemata, but they are not necessary for Kiwix Serve, only for Kiwix JS.

@@ -2,6 +2,10 @@
<html>
<head>
<meta charset="UTF-8">
<meta http-equiv="Content-Security-Policy"
content="default-src 'self' data: 'unsafe-inline' 'unsafe-eval';
frame-src 'self' moz-extension: chrome-extension:;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ditto for both moz-extension: and chrome-extension:. Unlikely that you'd be serving extension URIs. The reason we have these in Kiwix JS is that we package the app inside both types of extension.

@Jaifroid
Copy link
Member

@veloman-yunkan Just as a general comment on this PR, it will probably need to deal with opening external links, which will be blocked from opening in the iframe (or should be, though I no longer trust the parallels with the case of Kiwix JS). The way I suggested dealing with this was to add an on-click event listener on the iframe's content window (code provided in the relevant issue). Of course it would be worth testing first what happens to external links in this implementation. Another way to deal with it would be to alter the HTML of all external links in the server backend, adding a target="_blank" attribute. However, that could conflict with an existing target attribute, so doing it dynamically is cleaner.

@kelson42
Copy link
Collaborator

kelson42 commented Mar 29, 2023

@Jaifroid Sounds something maybe better to discuss in kiwix/kiwix-tools#591 which is anyway one of the last tickets still open in kiwix-tools 3.5.0 milestone?

@Jaifroid
Copy link
Member

@Jaifroid Sounds something maybe better to discuss in kiwix/kiwix-tools#591 which is anyway one of the last tickets still open in kiwix-tools 3.5.0 milestone?

OK, but I fear this PR will end up blocking any external links from loading if there is no attempt to handle the situation. They should never load in the iframe, but it would be terrible UX if clicking on one just produces the result below.

image

@veloman-yunkan
Copy link
Collaborator Author

@veloman-yunkan Just as a general comment on this PR, it will probably need to deal with opening external links, which will be blocked from opening in the iframe (or should be, though I no longer trust the parallels with the case of Kiwix JS). The way I suggested dealing with this was to add an on-click event listener on the iframe's content window (code provided in the relevant issue). Of course it would be worth testing first what happens to external links in this implementation. Another way to deal with it would be to alter the HTML of all external links in the server backend, adding a target="_blank" attribute. However, that could conflict with an existing target attribute, so doing it dynamically is cleaner.

@Jaifroid I primarily published this PR as a means of sharing my changes with you. At the moment my goal is to reproduce your solution in kiwix-serve whereupon I will productize it.

@Jaifroid
Copy link
Member

OK, to understand why this PR isn't working in the same way as Kiwix JS at least vis-à-vis Chromium <= 90 I would need to be able to run it locally, and I can't manage that on my Windows machine unfortunately. I wonder if we might deploy a version of this to the dev server (dev.library.kiwix.org), then I'd be happy to try to debug.

This fix requires Chromium version above 90.
@mgautierfr mgautierfr force-pushed the pdf-friendly-kiwix-serve branch from 795c7f3 to a6659cb Compare April 5, 2023 13:08
@mgautierfr
Copy link
Member

LGTM. At least I haven't found blockers.
Let's merge it and see/test it with nightly and dev.library.kiwix.org

@mgautierfr mgautierfr merged commit 453f02c into main Apr 5, 2023
@mgautierfr mgautierfr deleted the pdf-friendly-kiwix-serve branch April 5, 2023 13:41
veloman-yunkan added a commit that referenced this pull request Apr 10, 2023
The previous "fix" (merged PR #924) was buggy. It not only didn't work
in Chromium v90, but in more recent versions too.

I verified that this fix works in Firefox (v111) and Chromium (v90):

- Attempts by the ZIM content to break out of the viewer iframe are
blocked.
- PDFs are displayed in the viewer.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

PDF content blocked by Chrome
4 participants