mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-02 12:47:41 +02:00
269 lines
8.0 KiB
TypeScript
269 lines
8.0 KiB
TypeScript
import { net, protocol } from 'electron';
|
|
import { dirname, resolve, normalize } from 'path';
|
|
import { fileURLToPath, pathToFileURL } from 'url';
|
|
import { contentProtocolName } from './constants';
|
|
import resolvePathWithinDir from '@joplin/lib/utils/resolvePathWithinDir';
|
|
import { LoggerWrapper } from '@joplin/utils/Logger';
|
|
import * as fs from 'fs-extra';
|
|
import { createReadStream } from 'fs';
|
|
import { fromFilename } from '@joplin/lib/mime-utils';
|
|
import { createSecureRandom } from '@joplin/lib/uuid';
|
|
|
|
export interface AccessController {
|
|
remove(): void;
|
|
}
|
|
|
|
export interface CustomProtocolHandler {
|
|
// note-viewer/ URLs
|
|
allowReadAccessToDirectory(path: string): void;
|
|
allowReadAccessToFile(path: string): AccessController;
|
|
|
|
// file-media/ URLs
|
|
setMediaAccessEnabled(enabled: boolean): void;
|
|
getMediaAccessKey(): string;
|
|
}
|
|
|
|
|
|
// In some cases, the NodeJS built-in adapter (Readable.toWeb) closes its controller twice,
|
|
// leading to an error dialog. See:
|
|
// - https://github.com/nodejs/node/blob/e578c0b1e8d3dd817e692a0c5df1b97580bc7c7f/lib/internal/webstreams/adapters.js#L454
|
|
// - https://github.com/nodejs/node/issues/54205
|
|
// We work around this by creating a more-error-tolerant custom adapter.
|
|
const nodeStreamToWeb = (resultStream: fs.ReadStream) => {
|
|
resultStream.pause();
|
|
|
|
let closed = false;
|
|
|
|
return new ReadableStream({
|
|
start: (controller) => {
|
|
resultStream.on('data', (chunk) => {
|
|
if (closed) {
|
|
return;
|
|
}
|
|
|
|
if (Buffer.isBuffer(chunk)) {
|
|
controller.enqueue(new Uint8Array(chunk));
|
|
} else {
|
|
controller.enqueue(chunk);
|
|
}
|
|
|
|
if (controller.desiredSize <= 0) {
|
|
resultStream.pause();
|
|
}
|
|
});
|
|
|
|
resultStream.on('error', (error) => {
|
|
controller.error(error);
|
|
});
|
|
|
|
resultStream.on('end', () => {
|
|
if (!closed) {
|
|
closed = true;
|
|
controller.close();
|
|
}
|
|
});
|
|
},
|
|
pull: (_controller) => {
|
|
if (closed) {
|
|
return;
|
|
}
|
|
|
|
resultStream.resume();
|
|
},
|
|
cancel: () => {
|
|
if (!closed) {
|
|
closed = true;
|
|
resultStream.close();
|
|
}
|
|
},
|
|
}, { highWaterMark: resultStream.readableHighWaterMark });
|
|
};
|
|
|
|
// Allows seeking videos.
|
|
// See https://github.com/electron/electron/issues/38749 for why this is necessary.
|
|
const handleRangeRequest = async (request: Request, targetPath: string) => {
|
|
const makeUnsupportedRangeResponse = () => {
|
|
return new Response('unsupported range', {
|
|
status: 416, // Range Not Satisfiable
|
|
});
|
|
};
|
|
|
|
const rangeHeader = request.headers.get('Range');
|
|
if (!rangeHeader.startsWith('bytes=')) {
|
|
return makeUnsupportedRangeResponse();
|
|
}
|
|
|
|
const stat = await fs.stat(targetPath);
|
|
// Ranges are requested using one of the following formats
|
|
// bytes=1234-5679
|
|
// bytes=-5678
|
|
// bytes=1234-
|
|
// See https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Range
|
|
const startByte = Number(rangeHeader.match(/(\d+)-/)?.[1] || '0');
|
|
const endByte = Number(rangeHeader.match(/-(\d+)/)?.[1] || `${stat.size - 1}`);
|
|
|
|
if (endByte > stat.size || startByte < 0) {
|
|
return makeUnsupportedRangeResponse();
|
|
}
|
|
|
|
// Note: end is inclusive.
|
|
const resultStream = createReadStream(targetPath, { start: startByte, end: endByte });
|
|
|
|
// See the HTTP range requests guide: https://developer.mozilla.org/en-US/docs/Web/HTTP/Range_requests
|
|
const headers = new Headers([
|
|
['Accept-Ranges', 'bytes'],
|
|
['Content-Type', fromFilename(targetPath)],
|
|
['Content-Length', `${endByte + 1 - startByte}`],
|
|
['Content-Range', `bytes ${startByte}-${endByte}/${stat.size}`],
|
|
]);
|
|
|
|
|
|
return new Response(
|
|
nodeStreamToWeb(resultStream),
|
|
{ headers, status: 206 },
|
|
);
|
|
};
|
|
|
|
// Creating a custom protocol allows us to isolate iframes by giving them
|
|
// different domain names from the main Joplin app.
|
|
//
|
|
// For example, an iframe with url joplin-content://note-viewer/path/to/iframe.html will run
|
|
// in a different process from a parent frame with url file://path/to/iframe.html.
|
|
//
|
|
// See note_viewer_isolation.md for why this is important.
|
|
//
|
|
// TODO: Use Logger.create (doesn't work for now because Logger is only initialized
|
|
// in the main process.)
|
|
const handleCustomProtocols = (logger: LoggerWrapper): CustomProtocolHandler => {
|
|
logger = {
|
|
...logger,
|
|
debug: () => {},
|
|
};
|
|
|
|
// Allow-listed files/directories for joplin-content://note-viewer/
|
|
const readableDirectories: string[] = [];
|
|
const readableFiles = new Map<string, number>();
|
|
// Access for joplin-content://file-media/
|
|
let mediaAccessKey: string|false = false;
|
|
|
|
// See also the protocol.handle example: https://www.electronjs.org/docs/latest/api/protocol#protocolhandlescheme-handler
|
|
protocol.handle(contentProtocolName, async request => {
|
|
const url = new URL(request.url);
|
|
const host = url.host;
|
|
|
|
let pathname = normalize(fileURLToPath(`file://${url.pathname}`));
|
|
|
|
// See https://security.stackexchange.com/a/123723
|
|
if (pathname.startsWith('..')) {
|
|
throw new Error(`Invalid URL (not absolute), ${request.url}`);
|
|
}
|
|
|
|
pathname = resolve(appBundleDirectory, pathname);
|
|
|
|
let canRead = false;
|
|
let mediaOnly = true;
|
|
if (host === 'note-viewer') {
|
|
if (readableFiles.has(pathname)) {
|
|
canRead = true;
|
|
} else {
|
|
for (const readableDirectory of readableDirectories) {
|
|
if (resolvePathWithinDir(readableDirectory, pathname)) {
|
|
canRead = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
mediaOnly = false;
|
|
} else if (host === 'file-media') {
|
|
if (!mediaAccessKey) {
|
|
throw new Error('Media access denied. This must be enabled with .setMediaAccessEnabled');
|
|
}
|
|
|
|
canRead = true;
|
|
mediaOnly = true;
|
|
|
|
const accessKey = url.searchParams.get('access-key');
|
|
if (accessKey !== mediaAccessKey) {
|
|
throw new Error(`Invalid or missing media access key (was ${accessKey}). An allow-listed ?access-key= parameter must be provided.`);
|
|
}
|
|
} else {
|
|
throw new Error(`Invalid URL ${request.url}`);
|
|
}
|
|
|
|
if (!canRead) {
|
|
throw new Error(`Read access not granted for URL ${request.url}`);
|
|
}
|
|
|
|
const asFileUrl = pathToFileURL(pathname).toString();
|
|
logger.debug('protocol handler: Fetch file URL', asFileUrl);
|
|
|
|
const rangeHeader = request.headers.get('Range');
|
|
let response;
|
|
if (!rangeHeader) {
|
|
response = await net.fetch(asFileUrl);
|
|
} else {
|
|
response = await handleRangeRequest(request, pathname);
|
|
}
|
|
|
|
if (mediaOnly) {
|
|
// Tells the browser to avoid MIME confusion attacks. See
|
|
// https://blog.mozilla.org/security/2016/08/26/mitigating-mime-confusion-attacks-in-firefox/
|
|
response.headers.set('X-Content-Type-Options', 'nosniff');
|
|
|
|
// This is an extra check to prevent loading text/html and arbitrary non-media content from the URL.
|
|
const contentType = response.headers.get('Content-Type');
|
|
if (!contentType || !contentType.match(/^(image|video|audio)\//)) {
|
|
throw new Error(`Attempted to access non-media file from ${request.url}, which is media-only. Content type was ${contentType}.`);
|
|
}
|
|
}
|
|
|
|
return response;
|
|
});
|
|
|
|
const appBundleDirectory = dirname(dirname(__dirname));
|
|
return {
|
|
allowReadAccessToDirectory: (path: string) => {
|
|
path = resolve(appBundleDirectory, path);
|
|
logger.debug('protocol handler: Allow read access to directory', path);
|
|
|
|
readableDirectories.push(path);
|
|
},
|
|
allowReadAccessToFile: (path: string) => {
|
|
path = resolve(appBundleDirectory, path);
|
|
logger.debug('protocol handler: Allow read access to file', path);
|
|
|
|
if (readableFiles.has(path)) {
|
|
readableFiles.set(path, readableFiles.get(path) + 1);
|
|
} else {
|
|
readableFiles.set(path, 1);
|
|
}
|
|
|
|
return {
|
|
remove: () => {
|
|
if ((readableFiles.get(path) ?? 0) <= 1) {
|
|
logger.debug('protocol handler: Remove read access to file', path);
|
|
readableFiles.delete(path);
|
|
} else {
|
|
readableFiles.set(path, readableFiles.get(path) - 1);
|
|
}
|
|
},
|
|
};
|
|
},
|
|
setMediaAccessEnabled: (enabled: boolean) => {
|
|
if (enabled) {
|
|
mediaAccessKey ||= createSecureRandom();
|
|
} else {
|
|
mediaAccessKey = false;
|
|
}
|
|
},
|
|
// Allows access to all local media files, provided a matching ?access-key=<key> is added
|
|
// to the request URL.
|
|
getMediaAccessKey: () => {
|
|
return mediaAccessKey || null;
|
|
},
|
|
};
|
|
};
|
|
|
|
export default handleCustomProtocols;
|