chore: notion v4.0.0 compat

use https://www.notion.so/api/__notion-enhancer for internal asset proxy
This commit is contained in:
dragonwocky 2024-11-15 14:36:54 +11:00
parent d5e0810915
commit 7dde3b8b3c
Signed by: dragonwocky
GPG Key ID: 7998D08F7D7BD7A8
6 changed files with 47 additions and 42 deletions

View File

@ -44,8 +44,7 @@ const patches = {
)); ));
// https://github.com/notion-enhancer/notion-enhancer/issues/160: // https://github.com/notion-enhancer/notion-enhancer/issues/160:
// enable the notion:// protocol, windows-style tab layouts, and // run the app in windows mode on linux (instead of macos mode)
// quitting the app when the last window is closed on linux
const isWindows = const isWindows =
/(?:"win32"===process\.platform(?:(?=,isFullscreen)|(?=&&\w\.BrowserWindow)|(?=&&\(\w\.app\.requestSingleInstanceLock)))/g, /(?:"win32"===process\.platform(?:(?=,isFullscreen)|(?=&&\w\.BrowserWindow)|(?=&&\(\w\.app\.requestSingleInstanceLock)))/g,
isWindowsOrLinux = '["win32","linux"].includes(process.platform)'; isWindowsOrLinux = '["win32","linux"].includes(process.platform)';
@ -55,20 +54,6 @@ const patches = {
// so the notion-enhancer can be require()-d into it // so the notion-enhancer can be require()-d into it
replace(/sandbox:!0/g, `sandbox:!1,nodeIntegration:!0,session:require('electron').session.fromPartition("persist:notion")`); replace(/sandbox:!0/g, `sandbox:!1,nodeIntegration:!0,session:require('electron').session.fromPartition("persist:notion")`);
// bypass webRequest filter to load enhancer menu
replace(/(\w)\.top!==\w\?(\w)\(\{cancel:!0\}\)/, "$1.top!==$1?$2({})");
// serve enhancer sources over the notion:// protocol
const protocolHandler = /try\{const \w=await \w\.assetCache\.handleRequest\(\w\);/,
protocolInterceptor = `{const n="notion://www.notion.so/__notion-enhancer/";if(e.url.startsWith(n))return require("electron").net.fetch(\`file://\${require("path").join(__dirname,"..","..","node_modules","notion-enhancer",e.url.slice(n.length))}\`)}`;
prepend(protocolHandler, protocolInterceptor);
// disable csp entirely. risky? yes. necessary? also yes.
// potentially could extend default policy instead someday.
const headerHandler = /\.onHeadersReceived\(\(\((\w),\w\)=>{/,
cspDisabler = `{const responseHeaders=$1.responseHeaders;if(responseHeaders){delete responseHeaders["content-security-policy"];return t({responseHeaders})}}`
append(headerHandler, cspDisabler);
// expose the app's config + cache + preferences to the global namespace // expose the app's config + cache + preferences to the global namespace
// e.g. to enable development mode or check if keep in background is enabled // e.g. to enable development mode or check if keep in background is enabled
prepend(/\w\.exports=JSON\.parse\('\{"env":"production"/, "globalThis.__notionConfig="); prepend(/\w\.exports=JSON\.parse\('\{"env":"production"/, "globalThis.__notionConfig=");

View File

@ -21,11 +21,12 @@ const platform = IS_ELECTRON
version = IS_ELECTRON version = IS_ELECTRON
? require("notion-enhancer/package.json").version ? require("notion-enhancer/package.json").version
: chrome.runtime.getManifest().version, : chrome.runtime.getManifest().version,
// forms a url to a notion-enhancer asset or source file // packages a url to access notion-enhancer assets and sources,
// that can be accessed reliably over http // proxies via api in desktop app to bypass service worker cache
enhancerUrl = (target) => enhancerUrl = (target = "") =>
IS_ELECTRON IS_ELECTRON
? `notion://www.notion.so/__notion-enhancer/${target.replace(/^\//, "")}` ? "https://www.notion.so/api/__notion-enhancer/" +
target.replace(/^\//, "")
: chrome.runtime.getURL(target), : chrome.runtime.getURL(target),
// require a file from the root of notion's app/ folder, // require a file from the root of notion's app/ folder,
// only available in an electron main process // only available in an electron main process
@ -96,10 +97,6 @@ const readFile = (file) => {
{ resolve } = require("path"); { resolve } = require("path");
return fsp.readFile(resolve(`${__dirname}/../${file}`), "utf-8"); return fsp.readFile(resolve(`${__dirname}/../${file}`), "utf-8");
} }
// prefer using versions of files cached by the app
// or routed through the notion-enhancer's url interception
const notionProtocol = "notion://www.notion.so/";
file = file.replace(/^https:\/\/www\.notion\.so\//, notionProtocol);
} else file = file.startsWith("http") ? file : enhancerUrl(file); } else file = file.startsWith("http") ? file : enhancerUrl(file);
return fetch(file).then((res) => res.text()); return fetch(file).then((res) => res.text());
}, },
@ -112,8 +109,6 @@ const readFile = (file) => {
const { resolve } = require("path"); const { resolve } = require("path");
return require(resolve(`${__dirname}/../${file}`)); return require(resolve(`${__dirname}/../${file}`));
} }
const notionProtocol = "notion://www.notion.so/";
file = file.replace(/^https:\/\/www\.notion\.so\//, notionProtocol);
} else file = file.startsWith("http") ? file : enhancerUrl(file); } else file = file.startsWith("http") ? file : enhancerUrl(file);
return fetch(file).then((res) => res.json()); return fetch(file).then((res) => res.json());
}; };

View File

@ -43,10 +43,11 @@ const shouldLoadThemeOverrides = async (api, db) => {
}; };
const insertMenu = async (api, db) => { const insertMenu = async (api, db) => {
const notionSidebar = `.notion-sidebar-container .notion-sidebar > :nth-child(3) > div > :nth-child(2)`, const notionSettings = `.notion-sidebar-container .notion-sidebar [role="button"]:has(.newSidebarSettings)`,
{ html, addMutationListener, removeMutationListener } = api, { html, addMutationListener, removeMutationListener } = api,
{ addKeyListener, platform, enhancerUrl, onMessage } = api, { addKeyListener, platform, enhancerUrl, onMessage } = api,
menuButtonIconStyle = await db.get("menuButtonIconStyle"), menuButtonIconStyle = await db.get("menuButtonIconStyle"),
menuButtonLabel = await db.get("menuButtonLabel"),
openMenuHotkey = await db.get("openMenuHotkey"), openMenuHotkey = await db.get("openMenuHotkey"),
menuPing = { menuPing = {
channel: "notion-enhancer", channel: "notion-enhancer",
@ -84,18 +85,18 @@ const insertMenu = async (api, db) => {
icon="notion-enhancer${menuButtonIconStyle === "Monochrome" icon="notion-enhancer${menuButtonIconStyle === "Monochrome"
? "?mask" ? "?mask"
: " text-[16px]"}" : " text-[16px]"}"
>notion-enhancer >${menuButtonLabel}
<//>`; <//>`;
const appendToDom = () => { const appendToDom = () => {
if (!document.body.contains($modal)) document.body.append($modal); if (!document.body.contains($modal)) document.body.append($modal);
else if (!document.body.contains($button)) { else if (!document.body.contains($button)) {
document.querySelector(notionSidebar)?.append($button); document.querySelector(notionSettings)?.after($button);
} else removeMutationListener(appendToDom); } else removeMutationListener(appendToDom);
}; };
html`<${Tooltip}> html`<${Tooltip}>
<b>Configure the notion-enhancer and its mods</b> <b>Configure the notion-enhancer and its mods</b>
<//>`.attach($button, "right"); <//>`.attach($button, "right");
addMutationListener(notionSidebar, appendToDom); addMutationListener(notionSettings, appendToDom);
addMutationListener(".notion-app-inner", updateMenuTheme, { subtree: false }); addMutationListener(".notion-app-inner", updateMenuTheme, { subtree: false });
appendToDom(); appendToDom();

View File

@ -15,10 +15,10 @@ function MenuButton(
class: `notion-enhancer--menu-button flex select-none class: `notion-enhancer--menu-button flex select-none
cursor-pointer rounded-[6px] text-[14px] font-medium cursor-pointer rounded-[6px] text-[14px] font-medium
transition hover:bg-[color:var(--theme--bg-hover)] transition hover:bg-[color:var(--theme--bg-hover)]
w-full h-[30px] px-[8px] py-[4px] items-center`, w-full h-[30px] px-[10px] py-[4px] items-center`,
}); });
return html`<div ...${props}> return html`<div ...${props}>
<div class="flex items-center justify-center size-[22px] mr-[8px]"> <div class="flex items-center justify-center text-[18px] mr-[10px]">
<i class="i-${icon}"></i> <i class="i-${icon}"></i>
</div> </div>
<div>${children}</div> <div>${children}</div>

View File

@ -44,10 +44,16 @@
"description": "Loads the styling required for a theme to customise Notion's interface. Disabling this may increase client performance, but will also disable all themes.", "description": "Loads the styling required for a theme to customise Notion's interface. Disabling this may increase client performance, but will also disable all themes.",
"values": ["Auto", "Enabled", "Disabled"] "values": ["Auto", "Enabled", "Disabled"]
}, },
{
"type": "text",
"key": "menuButtonLabel",
"description": "Sets the text to label the notion-enhancer button added to Notion's sidebar with.",
"value": "notion-enhancer"
},
{ {
"type": "select", "type": "select",
"key": "menuButtonIconStyle", "key": "menuButtonIconStyle",
"description": "Sets whether the notion-enhancer icon added to Notion's sidebar should be coloured or monochrome. The latter style will match the theme's icon colour for users who would like the icon to be less noticeable.", "description": "Sets whether the icon beside the notion-enhancer button added to Notion's sidebar should be coloured or monochrome. The latter style will match the theme's icon colour for users who would like the icon to be less noticeable.",
"values": ["Colour", "Monochrome"] "values": ["Colour", "Monochrome"]
}, },
{ {

View File

@ -6,6 +6,8 @@
"use strict"; "use strict";
const { escape } = require("querystring");
const IS_ELECTRON = typeof module !== "undefined"; const IS_ELECTRON = typeof module !== "undefined";
let __db, __statements, __transactions; let __db, __statements, __transactions;
@ -113,15 +115,31 @@ const initDatabase = async () => {
}; };
if (IS_ELECTRON) { if (IS_ELECTRON) {
const { ipcMain } = require("electron"), const { reloadApp, enhancerUrl } = globalThis.__enhancerApi,
{ reloadApp } = globalThis.__enhancerApi; { ipcMain, session, app, net } = require("electron");
ipcMain.handle("notion-enhancer", ({}, message) => { app.on("ready", () => {
if (message?.action !== "query-database") return; // proxies notion-enhancer sources over www.notion.so via https
const { namespace, query, args } = message.data; const { protocol } = session.fromPartition("persist:notion");
return queryDatabase(namespace, query, args); protocol.handle("https", (req) => {
}); if (req.url.startsWith(enhancerUrl())) {
ipcMain.on("notion-enhancer", ({}, message) => { let url = req.url.slice(enhancerUrl().length);
if (message === "reload-app") reloadApp(); url = `file://${require("path").join(__dirname, url)}`;
return net.fetch(url);
} else return net.fetch(req);
});
// webRequest.onHeadersReceived(({ details: { responseHeaders }, callback }) => {
// delete responseHeaders["content-security-policy"];
// return callback({ responseHeaders });
// });
ipcMain.handle("notion-enhancer", ({}, message) => {
if (message?.action !== "query-database") return;
const { namespace, query, args } = message.data;
return queryDatabase(namespace, query, args);
});
ipcMain.on("notion-enhancer", ({}, message) => {
if (message === "reload-app") reloadApp();
});
}); });
} else { } else {
const notionUrl = "https://www.notion.so/", const notionUrl = "https://www.notion.so/",