aboutsummaryrefslogtreecommitdiffstats
path: root/packages/main/src/infrastructure/electron/impl/hardenSession.ts
diff options
context:
space:
mode:
Diffstat (limited to 'packages/main/src/infrastructure/electron/impl/hardenSession.ts')
-rw-r--r--packages/main/src/infrastructure/electron/impl/hardenSession.ts87
1 files changed, 87 insertions, 0 deletions
diff --git a/packages/main/src/infrastructure/electron/impl/hardenSession.ts b/packages/main/src/infrastructure/electron/impl/hardenSession.ts
new file mode 100644
index 0000000..71d8148
--- /dev/null
+++ b/packages/main/src/infrastructure/electron/impl/hardenSession.ts
@@ -0,0 +1,87 @@
1/*
2 * Copyright (C) 2022 Kristóf Marussy <kristof@marussy.com>
3 *
4 * This file is part of Sophie.
5 *
6 * Sophie is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU Affero General Public License as
8 * published by the Free Software Foundation, version 3.
9 *
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU Affero General Public License for more details.
14 *
15 * You should have received a copy of the GNU Affero General Public License
16 * along with this program. If not, see <https://www.gnu.org/licenses/>.
17 *
18 * SPDX-License-Identifier: AGPL-3.0-only
19 */
20
21import { URL } from 'node:url';
22
23import type { Session } from 'electron';
24
25import { getLogger } from '../../../utils/log';
26import type Resources from '../../resources/Resources';
27
28import { DEVMODE_ALLOWED_URL_PREFIXES } from './devTools';
29
30const log = getLogger('hardenSession');
31
32/**
33 * Hardens a session to prevent loading resources outside the renderer resources and
34 * to reject all permission requests.
35 *
36 * In dev mode, installation of extensions and opening the devtools will be allowed.
37 *
38 * @param resources The resource handle associated with the paths and URL of the application.
39 * @param devMode Whether the application is in development mode.
40 * @param session The session to harden.
41 */
42export default function hardenSession(
43 resources: Resources,
44 devMode: boolean,
45 session: Session,
46): void {
47 session.setPermissionRequestHandler((_webContents, _permission, callback) => {
48 callback(false);
49 });
50
51 const rendererBaseURL = resources.getRendererURL('/');
52 log.debug('Renderer base URL:', rendererBaseURL);
53
54 const webSocketBaseURL = rendererBaseURL.replace(/^http(s)?:/, 'ws$1:');
55 log.debug('WebSocket base URL:', webSocketBaseURL);
56
57 function shouldCancelRequest(url: string, method: string): boolean {
58 if (method !== 'GET') {
59 return true;
60 }
61 let normalizedURL: string;
62 try {
63 normalizedURL = new URL(url).toString();
64 } catch {
65 return true;
66 }
67 if (
68 devMode &&
69 DEVMODE_ALLOWED_URL_PREFIXES.some((prefix) =>
70 normalizedURL.startsWith(prefix),
71 )
72 ) {
73 return false;
74 }
75 const isHttp = normalizedURL.startsWith(rendererBaseURL);
76 const isWs = normalizedURL.startsWith(webSocketBaseURL);
77 return !isHttp && !isWs;
78 }
79
80 session.webRequest.onBeforeRequest(({ url, method }, callback) => {
81 const cancel = shouldCancelRequest(url, method);
82 if (cancel) {
83 log.error('Prevented loading', method, url);
84 }
85 callback({ cancel });
86 });
87}