obsidian-dev-utils
Version:
This is the collection of useful functions that you can use for your Obsidian plugin development
129 lines (126 loc) • 14.9 kB
JavaScript
/*
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
if you want to view the source, please visit the github repository of this plugin
*/
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
import {
FunctionHandlingMode,
toJson
} from "../../ObjectUtils.mjs";
import {
makeValidVariableName,
replaceAll
} from "../../String.mjs";
import { readFile } from "../NodeModules.mjs";
function preprocessPlugin(isEsm) {
const replacements = isEsm ? {} : {
[replaceAll("import(dot)meta(dot)url", "(dot)", ".")]: () => {
if (typeof __filename === "string") {
const localRequire = require;
const url = localRequire("node:url");
return url.pathToFileURL(__filename).href;
}
if (typeof window !== "undefined") {
return window.location.href;
}
return "";
}
};
return {
name: "preprocess",
setup(build) {
build.initialOptions.define ??= {};
for (const key of Object.keys(replacements)) {
build.initialOptions.define[key] = `__${makeValidVariableName(key)}`;
}
build.initialOptions.banner ??= {};
build.initialOptions.banner["js"] ??= "";
build.initialOptions.banner["js"] += `
(${String(isEsm ? initEsm : initCjs)})();
`;
build.onLoad({ filter: /\.(?:js|ts|cjs|mjs|cts|mts)$/ }, async (args) => {
let contents = await readFile(args.path, "utf-8");
for (const [key, value] of Object.entries(replacements)) {
const variable = `__${makeValidVariableName(key)}`;
if (!contents.includes(key)) {
continue;
}
const valueStr = typeof value === "function" ? `(${String(value)})()` : toJson(value, { functionHandlingMode: FunctionHandlingMode.Full });
if (contents.includes(`var ${variable}`)) {
continue;
}
contents = `var ${variable} = globalThis['${key}'] ?? ${valueStr};
${contents}`;
}
contents = replaceAll(contents, /`\r?\n\/\/# sourceMappingURL/g, "`\n//#${''} sourceMappingURL");
return {
contents,
loader: "ts"
};
});
}
};
}
function initCjs() {
const globalThisRecord = globalThis;
globalThisRecord["__name"] ??= name;
const originalRequire = require;
if (originalRequire && !originalRequire.__isPatched) {
require = Object.assign(
(id) => requirePatched(id),
originalRequire,
{
__isPatched: true
}
);
}
const newFuncs = {
__extractDefault: () => extractDefault,
process: () => {
const browserProcess = {
browser: true,
cwd: () => "/",
env: {},
platform: "android"
};
return browserProcess;
}
};
for (const key of Object.keys(newFuncs)) {
globalThisRecord[key] ??= newFuncs[key]?.();
}
function name(obj) {
return obj;
}
function extractDefault(module) {
return module && module.__esModule && "default" in module ? module.default : module;
}
function requirePatched(id) {
const module = originalRequire?.(id);
if (module) {
return extractDefault(module);
}
if (id === "process" || id === "node:process") {
console.error(`Module not found: ${id}. Fake process object is returned instead.`);
return globalThis.process;
}
console.error(`Module not found: ${id}. Empty object is returned instead.`);
return {};
}
}
function initEsm() {
if (globalThis.process) {
return;
}
const browserProcess = {
browser: true,
cwd: () => "/",
env: {},
platform: "android"
};
globalThis.process = browserProcess;
}
export {
preprocessPlugin
};
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/ScriptUtils/esbuild/preprocessPlugin.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module defines a custom esbuild plugin that preprocesses JavaScript and TypeScript files.\n * The preprocessing includes replacing `import(dot)meta(dot)url` with a Node.js-compatible alternative,\n * ensuring compatibility with Obsidian's plugin system, and adding a basic `process` object for environments\n * where `process` is not available (like mobile or web environments).\n *\n * @remarks\n * We cannot use `.` instead of `(dot)` in the above description because the file itself is preprocessed with the same rule.\n */\n\nimport type { Plugin } from 'esbuild';\n\nimport type { GenericObject } from '../../ObjectUtils.ts';\n\nimport {\n  FunctionHandlingMode,\n  toJson\n} from '../../ObjectUtils.ts';\nimport {\n  makeValidVariableName,\n  replaceAll\n} from '../../String.ts';\nimport { readFile } from '../NodeModules.ts';\n\ninterface BrowserProcess extends Partial<NodeJS.Process> {\n  browser: boolean;\n}\n\ninterface EsmModule {\n  __esModule: boolean;\n  default: unknown;\n}\n\ninterface RequirePatched extends NodeJS.Require {\n  __isPatched: boolean;\n}\n\n/**\n * Creates an esbuild plugin that preprocesses JavaScript and TypeScript files.\n *\n * This plugin performs the following tasks:\n * - Replaces instances of `import(dot)meta(dot)url` with a Node.js-compatible `__filename` alternative.\n * - Modifies the `sourceMappingURL` comment to ensure compatibility with Obsidian's plugin system.\n * - Adds a basic `process` object to the global scope if `process` is referenced but not defined.\n *\n * @param isEsm - Whether the build is for an ESM format.\n * @returns An esbuild `Plugin` object that handles the preprocessing.\n */\nexport function preprocessPlugin(isEsm?: boolean): Plugin {\n  const replacements = isEsm\n    ? {}\n    : {\n      [replaceAll('import(dot)meta(dot)url', '(dot)', '.')]: (): string => {\n        if (typeof __filename === 'string') {\n          const localRequire = require;\n          const url = localRequire('node:url') as typeof import('node:url');\n          return url.pathToFileURL(__filename).href;\n        }\n\n        if (typeof window !== 'undefined') {\n          return window.location.href;\n        }\n\n        // Fallback to an empty string if the environment is unknown\n        return '';\n      }\n    };\n\n  return {\n    name: 'preprocess',\n    setup(build): void {\n      build.initialOptions.define ??= {};\n\n      for (const key of Object.keys(replacements)) {\n        build.initialOptions.define[key] = `__${makeValidVariableName(key)}`;\n      }\n\n      build.initialOptions.banner ??= {};\n      build.initialOptions.banner['js'] ??= '';\n      build.initialOptions.banner['js'] += `\\n(${String(isEsm ? initEsm : initCjs)})();\\n`;\n\n      build.onLoad({ filter: /\\.(?:js|ts|cjs|mjs|cts|mts)$/ }, async (args) => {\n        let contents = await readFile(args.path, 'utf-8');\n\n        for (const [key, value] of Object.entries(replacements)) {\n          const variable = `__${makeValidVariableName(key)}`;\n          if (!contents.includes(key)) {\n            continue;\n          }\n          const valueStr = typeof value === 'function' ? `(${String(value)})()` : toJson(value, { functionHandlingMode: FunctionHandlingMode.Full });\n          if (contents.includes(`var ${variable}`)) {\n            continue;\n          }\n          contents = `var ${variable} = globalThis['${key}'] ?? ${valueStr};\\n${contents}`;\n        }\n\n        // HACK: The ${''} part is used to ensure Obsidian loads the plugin properly,\n        // Otherwise, it stops loading after the first line of the sourceMappingURL comment.\n        // eslint-disable-next-line no-template-curly-in-string\n        contents = replaceAll(contents, /`\\r?\\n\\/\\/# sourceMappingURL/g, '`\\n//#${\\'\\'} sourceMappingURL');\n\n        return {\n          contents,\n          loader: 'ts'\n        };\n      });\n    }\n  };\n}\n\nfunction initCjs(): void {\n  const globalThisRecord = globalThis as unknown as GenericObject;\n  globalThisRecord['__name'] ??= name;\n  const originalRequire = require as (NodeJS.Require & Partial<RequirePatched> | undefined);\n  if (originalRequire && !originalRequire.__isPatched) {\n    require = Object.assign(\n      (id: string) => requirePatched(id),\n      originalRequire,\n      {\n        __isPatched: true\n      }\n    ) as RequirePatched;\n  }\n\n  const newFuncs: Record<string, () => unknown> = {\n    __extractDefault: () => extractDefault,\n    process: () => {\n      const browserProcess: BrowserProcess = {\n        browser: true,\n        cwd: () => '/',\n        env: {},\n        platform: 'android'\n      };\n      return browserProcess;\n    }\n  };\n\n  for (const key of Object.keys(newFuncs)) {\n    globalThisRecord[key] ??= newFuncs[key]?.();\n  }\n\n  function name(obj: unknown): unknown {\n    return obj;\n  }\n\n  function extractDefault(module: Partial<EsmModule> | undefined): unknown {\n    return module && module.__esModule && 'default' in module ? module.default : module;\n  }\n\n  function requirePatched(id: string): unknown {\n    const module = originalRequire?.(id) as (Partial<EsmModule> | undefined);\n    if (module) {\n      return extractDefault(module);\n    }\n\n    if (id === 'process' || id === 'node:process') {\n      console.error(`Module not found: ${id}. Fake process object is returned instead.`);\n      return globalThis.process;\n    }\n\n    console.error(`Module not found: ${id}. Empty object is returned instead.`);\n    return {};\n  }\n}\n\nfunction initEsm(): void {\n  if ((globalThis.process as NodeJS.Process | undefined)) {\n    return;\n  }\n\n  const browserProcess: BrowserProcess = {\n    browser: true,\n    cwd: () => '/',\n    env: {},\n    platform: 'android'\n  };\n  globalThis.process = browserProcess as NodeJS.Process;\n}\n"],
  "mappings": "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",
  "names": []
}
