Skip to content

Commit 695f27e

Browse files
committed
feat: add PDF URL validation and content type check
Signed-off-by: Yukai Huang <[email protected]>
1 parent 9448c91 commit 695f27e

File tree

2 files changed

+48
-6
lines changed

2 files changed

+48
-6
lines changed

public/js/extra.js

+36-6
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ import './lib/renderer/lightbox'
2828
import { renderCSVPreview } from './lib/renderer/csvpreview'
2929

3030
import { escapeAttrValue } from './render'
31-
import { sanitizeUrl } from './utils'
31+
import { sanitizeUrl, isPdfUrl } from './utils'
3232

3333
import markdownit from 'markdown-it'
3434
import markdownitContainer from 'markdown-it-container'
@@ -634,11 +634,41 @@ export function finishView (view) {
634634
const cleanUrl = sanitizeUrl(url)
635635
const inner = $('<div></div>')
636636
$(this).append(inner)
637-
setTimeout(() => {
638-
PDFObject.embed(cleanUrl, inner, {
639-
height: '400px'
640-
})
641-
}, 1)
637+
638+
// First check URL format
639+
const isPDFByExtension = /\.pdf(\?.*)?$/i.test(cleanUrl) || cleanUrl.includes('pdf')
640+
641+
if (isPDFByExtension) {
642+
// Show loading message while we check content type
643+
const loadingMessage = $('<div class="alert alert-info">Verifying PDF file...</div>');
644+
inner.html(loadingMessage);
645+
646+
// Perform additional validation with HEAD request
647+
isPdfUrl(cleanUrl).then(isPDFByContentType => {
648+
if (isPDFByContentType) {
649+
// Valid PDF by content type, embed it
650+
PDFObject.embed(cleanUrl, inner, {
651+
height: '400px'
652+
});
653+
} else {
654+
// URL format looks like PDF but content type doesn't match
655+
inner.html('<div class="alert alert-warning">The URL looks like a PDF but the server didn\'t confirm it has a PDF content type.</div>');
656+
console.warn('URL has PDF extension but content type is not application/pdf:', cleanUrl);
657+
658+
// Try to embed anyway as a fallback
659+
setTimeout(() => {
660+
PDFObject.embed(cleanUrl, inner, {
661+
height: '400px',
662+
fallbackLink: 'This doesn\'t appear to be a valid PDF. <a href="[url]">Click here to try downloading it directly</a>.'
663+
});
664+
}, 1);
665+
}
666+
});
667+
} else {
668+
// Not a valid PDF URL by extension
669+
inner.html('<div class="alert alert-danger">Invalid PDF URL. The URL must point to a PDF file.</div>');
670+
console.warn('Invalid PDF URL format:', cleanUrl);
671+
}
642672
})
643673
// syntax highlighting
644674
view.find('code.raw').removeClass('raw')

public/js/utils.js

+12
Original file line numberDiff line numberDiff line change
@@ -46,3 +46,15 @@ export function sanitizeUrl (rawUrl) {
4646
return 'about:blank'
4747
}
4848
}
49+
50+
// Check if URL is a PDF based on Content-Type header
51+
export async function isPdfUrl(url) {
52+
try {
53+
const response = await fetch(url, { method: 'HEAD' });
54+
const contentType = response.headers.get('Content-Type');
55+
return contentType === 'application/pdf';
56+
} catch (error) {
57+
console.warn('Error checking PDF content type:', error);
58+
return false;
59+
}
60+
}

0 commit comments

Comments
 (0)