obsidian-dev-utils 22.0.0 → 22.1.1-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +9 -0
- package/dist/lib/cjs/Async.cjs +2 -2
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/Object.cjs +1 -1
- package/dist/lib/cjs/Object.d.cts +6 -0
- package/dist/lib/cjs/codemirror/StateFieldSpec.cjs +1 -1
- package/dist/lib/cjs/codemirror/StateFieldSpec.d.cts +4 -4
- package/dist/lib/cjs/obsidian/Loop.cjs +2 -1
- package/dist/lib/cjs/obsidian/Modals/Prompt.cjs +1 -1
- package/dist/lib/cjs/obsidian/Modals/Prompt.d.cts +1 -1
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.cjs +9 -35
- package/dist/lib/cjs/obsidian/Plugin/PluginBase.d.cts +5 -23
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.cjs +161 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsManagerBase.d.cts +41 -0
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.cjs +34 -48
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsTabBase.d.cts +12 -31
- package/dist/lib/cjs/obsidian/Plugin/index.cjs +4 -7
- package/dist/lib/cjs/obsidian/Plugin/index.d.cts +1 -2
- package/dist/lib/cjs/obsidian/Vault.cjs +2 -2
- package/dist/lib/esm/Async.mjs +2 -2
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/Object.d.mts +6 -0
- package/dist/lib/esm/Object.mjs +1 -1
- package/dist/lib/esm/codemirror/StateFieldSpec.d.mts +4 -4
- package/dist/lib/esm/obsidian/Loop.mjs +2 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.d.mts +1 -1
- package/dist/lib/esm/obsidian/Modals/Prompt.mjs +1 -1
- package/dist/lib/esm/obsidian/Plugin/PluginBase.d.mts +5 -23
- package/dist/lib/esm/obsidian/Plugin/PluginBase.mjs +9 -35
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.d.mts +41 -0
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsManagerBase.mjs +137 -0
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.d.mts +12 -31
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsTabBase.mjs +35 -52
- package/dist/lib/esm/obsidian/Plugin/index.d.mts +1 -2
- package/dist/lib/esm/obsidian/Plugin/index.mjs +3 -5
- package/dist/lib/esm/obsidian/Vault.mjs +2 -2
- package/obsidian/Plugin/{PluginSettingsBase → PluginSettingsManagerBase}/package.json +3 -3
- package/package.json +7 -7
- package/dist/lib/cjs/obsidian/Plugin/EmptySettings.cjs +0 -38
- package/dist/lib/cjs/obsidian/Plugin/EmptySettings.d.cts +0 -14
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsBase.cjs +0 -92
- package/dist/lib/cjs/obsidian/Plugin/PluginSettingsBase.d.cts +0 -31
- package/dist/lib/esm/obsidian/Plugin/EmptySettings.d.mts +0 -14
- package/dist/lib/esm/obsidian/Plugin/EmptySettings.mjs +0 -14
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsBase.d.mts +0 -31
- package/dist/lib/esm/obsidian/Plugin/PluginSettingsBase.mjs +0 -68
- package/obsidian/Plugin/EmptySettings/package.json +0 -6
@@ -3,49 +3,34 @@
|
|
3
3
|
* This module defines a base class for creating plugin setting tabs in Obsidian.
|
4
4
|
* It provides a utility method to bind value components to plugin settings and handle changes.
|
5
5
|
*/
|
6
|
-
import type { BaseComponent } from 'obsidian';
|
7
6
|
import type { ConditionalKeys, Promisable } from 'type-fest';
|
8
7
|
import { PluginSettingTab } from 'obsidian';
|
8
|
+
import type { StringKeys } from '../../Object.cjs';
|
9
9
|
import type { ValueComponentWithChangeTracking } from '../Components/ValueComponentWithChangeTracking.cjs';
|
10
10
|
import type { PluginBase } from './PluginBase.cjs';
|
11
|
-
import type { PluginSettingsBase } from './PluginSettingsBase.cjs';
|
12
11
|
/**
|
13
12
|
* Options for binding a value component to a plugin setting.
|
14
13
|
*/
|
15
|
-
export interface BindOptions
|
14
|
+
export interface BindOptions {
|
16
15
|
/**
|
17
16
|
* A callback function that is called when the value of the component changes.
|
18
17
|
*/
|
19
|
-
onChanged
|
20
|
-
/**
|
21
|
-
* The plugin settings object to bind the component to. Default is the plugin's current settings.
|
22
|
-
*/
|
23
|
-
pluginSettings?: PluginSettings;
|
24
|
-
/**
|
25
|
-
* If true, saves the plugin settings automatically after the component value changes. Default is `true`.
|
26
|
-
*/
|
27
|
-
shouldAutoSave?: boolean;
|
18
|
+
onChanged?(): Promisable<void>;
|
28
19
|
/**
|
29
20
|
* If true, shows the validation message when the component value is invalid. Default is `true`.
|
30
21
|
*/
|
31
22
|
shouldShowValidationMessage?: boolean;
|
32
|
-
/**
|
33
|
-
* Validates the UI value before setting it on the plugin settings.
|
34
|
-
* @param uiValue - The value of the UI component.
|
35
|
-
* @returns An error message if the value is invalid, or `(empty string)` or `void` if it is valid.
|
36
|
-
*/
|
37
|
-
valueValidator?: (uiValue: UIValue) => Promisable<string | void>;
|
38
23
|
}
|
39
24
|
/**
|
40
25
|
* Extended options for binding a value component to a plugin setting.
|
41
26
|
*/
|
42
|
-
export interface BindOptionsExtended<PluginSettings, UIValue, Property extends
|
27
|
+
export interface BindOptionsExtended<PluginSettings extends object, UIValue, Property extends StringKeys<PluginSettings>> extends BindOptions {
|
43
28
|
/**
|
44
29
|
* Converts the UI component's value back to the plugin settings value.
|
45
30
|
* @param uiValue - The value of the UI component.
|
46
31
|
* @returns The value to set on the plugin settings.
|
47
32
|
*/
|
48
|
-
componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[Property];
|
33
|
+
componentToPluginSettingsValueConverter: (uiValue: UIValue) => PluginSettings[Property] | ValidationMessageHolder;
|
49
34
|
/**
|
50
35
|
* Converts the plugin settings value to the value used by the UI component.
|
51
36
|
* @param pluginSettingsValue - The value of the property in the plugin settings.
|
@@ -53,7 +38,10 @@ export interface BindOptionsExtended<PluginSettings, UIValue, Property extends k
|
|
53
38
|
*/
|
54
39
|
pluginSettingsToComponentValueConverter: (pluginSettingsValue: PluginSettings[Property]) => UIValue;
|
55
40
|
}
|
56
|
-
type ExtractPluginSettings<T extends PluginBase<any>> =
|
41
|
+
type ExtractPluginSettings<T extends PluginBase<any>> = Writable<T['settings']>;
|
42
|
+
interface ValidationMessageHolder {
|
43
|
+
validationMessage: string;
|
44
|
+
}
|
57
45
|
/**
|
58
46
|
* Base class for creating plugin settings tabs in Obsidian.
|
59
47
|
* Provides a method for binding value components to plugin settings and handling changes.
|
@@ -62,7 +50,6 @@ type ExtractPluginSettings<T extends PluginBase<any>> = PluginSettingsBase & T['
|
|
62
50
|
*/
|
63
51
|
export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<any>> extends PluginSettingTab {
|
64
52
|
plugin: TPlugin;
|
65
|
-
private validatorsMap;
|
66
53
|
constructor(plugin: TPlugin);
|
67
54
|
/**
|
68
55
|
* Binds a value component to a plugin setting.
|
@@ -74,7 +61,7 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
|
|
74
61
|
* @param options - The options for binding the value component.
|
75
62
|
* @returns The value component.
|
76
63
|
*/
|
77
|
-
bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions
|
64
|
+
bind<UIValue, TValueComponent>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: ConditionalKeys<ExtractPluginSettings<TPlugin>, UIValue>, options?: BindOptions): TValueComponent;
|
78
65
|
/**
|
79
66
|
* Binds a value component to a plugin setting.
|
80
67
|
*
|
@@ -86,13 +73,7 @@ export declare abstract class PluginSettingsTabBase<TPlugin extends PluginBase<a
|
|
86
73
|
* @param options - The options for binding the value component.
|
87
74
|
* @returns The value component.
|
88
75
|
*/
|
89
|
-
bind<UIValue, TValueComponent, Property extends
|
90
|
-
|
91
|
-
* Revalidates the value component.
|
92
|
-
*
|
93
|
-
* @param baseComponent - The base component to revalidate.
|
94
|
-
* @returns A promise that resolves to a boolean indicating whether the value component is valid.
|
95
|
-
*/
|
96
|
-
protected revalidate(baseComponent: BaseComponent): Promise<boolean>;
|
76
|
+
bind<UIValue, TValueComponent, Property extends StringKeys<ExtractPluginSettings<TPlugin>>>(valueComponent: TValueComponent & ValueComponentWithChangeTracking<UIValue>, property: Property, options: BindOptionsExtended<ExtractPluginSettings<TPlugin>, UIValue, Property>): TValueComponent;
|
77
|
+
hide(): void;
|
97
78
|
}
|
98
79
|
export {};
|
@@ -35,33 +35,30 @@ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__ge
|
|
35
35
|
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
36
36
|
var Plugin_exports = {};
|
37
37
|
__export(Plugin_exports, {
|
38
|
-
EmptySettings: () => EmptySettings,
|
39
38
|
ObsidianPluginRepoPaths: () => ObsidianPluginRepoPaths,
|
40
39
|
Plugin: () => Plugin,
|
41
40
|
PluginBase: () => PluginBase,
|
42
41
|
PluginContext: () => PluginContext,
|
43
42
|
PluginId: () => PluginId,
|
44
|
-
|
43
|
+
PluginSettingsManagerBase: () => PluginSettingsManagerBase,
|
45
44
|
PluginSettingsTabBase: () => PluginSettingsTabBase
|
46
45
|
});
|
47
46
|
module.exports = __toCommonJS(Plugin_exports);
|
48
|
-
var EmptySettings = __toESM(__extractDefault(require('./EmptySettings.cjs')), 1);
|
49
47
|
var ObsidianPluginRepoPaths = __toESM(__extractDefault(require('./ObsidianPluginRepoPaths.cjs')), 1);
|
50
48
|
var Plugin = __toESM(__extractDefault(require('./Plugin.cjs')), 1);
|
51
49
|
var PluginBase = __toESM(__extractDefault(require('./PluginBase.cjs')), 1);
|
52
50
|
var PluginContext = __toESM(__extractDefault(require('./PluginContext.cjs')), 1);
|
53
51
|
var PluginId = __toESM(__extractDefault(require('./PluginId.cjs')), 1);
|
54
|
-
var
|
52
|
+
var PluginSettingsManagerBase = __toESM(__extractDefault(require('./PluginSettingsManagerBase.cjs')), 1);
|
55
53
|
var PluginSettingsTabBase = __toESM(__extractDefault(require('./PluginSettingsTabBase.cjs')), 1);
|
56
54
|
// Annotate the CommonJS export names for ESM import in node:
|
57
55
|
0 && (module.exports = {
|
58
|
-
EmptySettings,
|
59
56
|
ObsidianPluginRepoPaths,
|
60
57
|
Plugin,
|
61
58
|
PluginBase,
|
62
59
|
PluginContext,
|
63
60
|
PluginId,
|
64
|
-
|
61
|
+
PluginSettingsManagerBase,
|
65
62
|
PluginSettingsTabBase
|
66
63
|
});
|
67
|
-
//# sourceMappingURL=data:application/json;base64,
|
64
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1BsdWdpbi9pbmRleC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyogVEhJUyBJUyBBIEdFTkVSQVRFRC9CVU5ETEVEIEZJTEUgQlkgQlVJTEQgU0NSSVBUICovXG5cbmV4cG9ydCAqIGFzIE9ic2lkaWFuUGx1Z2luUmVwb1BhdGhzIGZyb20gJy4vT2JzaWRpYW5QbHVnaW5SZXBvUGF0aHMudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luIGZyb20gJy4vUGx1Z2luLnRzJztcbmV4cG9ydCAqIGFzIFBsdWdpbkJhc2UgZnJvbSAnLi9QbHVnaW5CYXNlLnRzJztcbmV4cG9ydCAqIGFzIFBsdWdpbkNvbnRleHQgZnJvbSAnLi9QbHVnaW5Db250ZXh0LnRzJztcbmV4cG9ydCAqIGFzIFBsdWdpbklkIGZyb20gJy4vUGx1Z2luSWQudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luU2V0dGluZ3NNYW5hZ2VyQmFzZSBmcm9tICcuL1BsdWdpblNldHRpbmdzTWFuYWdlckJhc2UudHMnO1xuZXhwb3J0ICogYXMgUGx1Z2luU2V0dGluZ3NUYWJCYXNlIGZyb20gJy4vUGx1Z2luU2V0dGluZ3NUYWJCYXNlLnRzJztcbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUVBLDhCQUF5QztBQUN6QyxhQUF3QjtBQUN4QixpQkFBNEI7QUFDNUIsb0JBQStCO0FBQy9CLGVBQTBCO0FBQzFCLGdDQUEyQztBQUMzQyw0QkFBdUM7IiwKICAibmFtZXMiOiBbXQp9Cg==
|
@@ -1,8 +1,7 @@
|
|
1
|
-
export * as EmptySettings from './EmptySettings.cjs';
|
2
1
|
export * as ObsidianPluginRepoPaths from './ObsidianPluginRepoPaths.cjs';
|
3
2
|
export * as Plugin from './Plugin.cjs';
|
4
3
|
export * as PluginBase from './PluginBase.cjs';
|
5
4
|
export * as PluginContext from './PluginContext.cjs';
|
6
5
|
export * as PluginId from './PluginId.cjs';
|
7
|
-
export * as
|
6
|
+
export * as PluginSettingsManagerBase from './PluginSettingsManagerBase.cjs';
|
8
7
|
export * as PluginSettingsTabBase from './PluginSettingsTabBase.cjs';
|
@@ -128,7 +128,7 @@ function getSafeRenamePath(app, oldPathOrFile, newPath) {
|
|
128
128
|
let folderPath = (0, import_Path.dirname)(newPath);
|
129
129
|
let nonExistingPath = (0, import_Path.basename)(newPath);
|
130
130
|
let folder;
|
131
|
-
|
131
|
+
while (true) {
|
132
132
|
folder = (0, import_FileSystem.getFolderOrNull)(app, folderPath, true);
|
133
133
|
if (folder) {
|
134
134
|
break;
|
@@ -264,4 +264,4 @@ async function invokeFileActionSafe(app, pathOrFile, fileAction) {
|
|
264
264
|
readSafe,
|
265
265
|
renameSafe
|
266
266
|
});
|
267
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Vault\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  ListedFiles,\n  TFile,\n  TFolder\n} from 'obsidian';\n\nimport { parentFolderPath } from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { retryWithTimeout } from '../Async.ts';\nimport { noopAsync } from '../Function.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join\n} from '../Path.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isNote\n} from './FileSystem.ts';\n\n/**\n * Options for the `process` function.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * If `true`, the function will throw an error if the file is missing or deleted.\n   */\n  shouldFailOnMissingFile?: boolean;\n}\n\n/**\n * Copies a file safely in the vault.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to copy.\n * @param newPath - The new path to copy the file to.\n * @returns A promise that resolves to the new path of the copied file.\n */\nexport async function copySafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const file = getFile(app, oldPathOrFile);\n\n  const newFolderPath = parentFolderPath(newPath);\n  await createFolderSafe(app, newFolderPath);\n\n  const newAvailablePath = getAvailablePath(app, newPath);\n\n  try {\n    await app.vault.copy(file, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.createFolder(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n    return true;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = getFileOrNull(app, path);\n  if (file) {\n    return noopAsync;\n  }\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  try {\n    await app.vault.create(path, '');\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  return async () => {\n    file = getFile(app, path);\n    if (!file.deleted) {\n      await app.fileManager.trashFile(file);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = getFolderOrNull(app, path);\n  if (folder) {\n    return noopAsync;\n  }\n\n  const dirPath = parentFolderPath(path);\n  await createTempFolder(app, dirPath);\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  await createFolderSafe(app, path);\n\n  return async () => {\n    folder = getFolder(app, path);\n    if (!folder.deleted) {\n      await app.fileManager.trashFile(folder);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Gets an available path for a file in the vault.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get an available path for.\n * @returns The available path for the file.\n */\nexport function getAvailablePath(app: App, path: string): string {\n  const ext = extname(path);\n  return app.vault.getAvailablePath(join(dirname(path), basename(path, ext)), ext.slice(1));\n}\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Retrieves an array of all note files from the app's vault and sorts them alphabetically by their file path.\n * @param app - The Obsidian app instance.\n * @returns An array of all note files in the vault sorted by file path.\n */\nexport function getNoteFilesSorted(app: App): TFile[] {\n  return app.vault.getAllLoadedFiles().filter((file) => isFile(file) && isNote(app, file)).sort((a, b) => a.path.localeCompare(b.path)) as TFile[];\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns The safe rename path for the file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrFile: PathOrFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrFile);\n\n  if (app.vault.adapter.insensitive) {\n    let folderPath = dirname(newPath);\n    let nonExistingPath = basename(newPath);\n    let folder: null | TFolder;\n    for (;;) {\n      folder = getFolderOrNull(app, folderPath, true);\n      if (folder) {\n        break;\n      }\n      nonExistingPath = join(basename(folderPath), nonExistingPath);\n      folderPath = dirname(folderPath);\n    }\n    newPath = join(folder.getParentPrefix(), nonExistingPath);\n  }\n\n  if (oldPath.toLowerCase() === newPath.toLowerCase()) {\n    return newPath;\n  }\n\n  return getAvailablePath(app, newPath);\n}\n\n/**\n * Checks if a folder is empty.\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A promise that resolves to a boolean indicating whether the folder is empty.\n */\nexport async function isEmptyFolder(app: App, pathOrFolder: PathOrFolder): Promise<boolean> {\n  const listedFiles = await listSafe(app, getPath(app, pathOrFolder));\n  return listedFiles.files.length === 0 && listedFiles.folders.length === 0;\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to list.\n * @returns A promise that resolves to a `ListedFiles` object containing the listed files and folders.\n */\nexport async function listSafe(app: App, pathOrFolder: PathOrFolder): Promise<ListedFiles> {\n  const path = getPath(app, pathOrFolder);\n  const EMPTY = { files: [], folders: [] };\n\n  if ((await app.vault.adapter.stat(path))?.type !== 'folder') {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param options - Optional options for processing/retrying the operation.\n *\n * @returns A promise that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function process(\n  app: App,\n  pathOrFile: PathOrFile,\n  newContentProvider: ValueProvider<null | string, [string]>,\n  options: ProcessOptions = {}\n): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS = {\n    shouldFailOnMissingFile: true\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n\n  await retryWithTimeout(async () => {\n    const oldContent = await readSafe(app, pathOrFile);\n\n    if (oldContent === null) {\n      return handleMissingFile();\n    }\n\n    const newContent = await resolveValue(newContentProvider, oldContent);\n    if (newContent === null) {\n      return false;\n    }\n\n    let isSuccess = true;\n    const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n      await app.vault.process(file, (content) => {\n        if (content !== oldContent) {\n          console.warn('Content has changed since it was read. Retrying...', {\n            actualContent: content,\n            expectedContent: oldContent,\n            path: file.path\n          });\n          isSuccess = false;\n          return content;\n        }\n\n        return newContent;\n      });\n    });\n\n    if (!doesFileExist) {\n      return handleMissingFile();\n    }\n\n    return isSuccess;\n\n    function handleMissingFile(): boolean {\n      if (fullOptions.shouldFailOnMissingFile) {\n        const path = getPath(app, pathOrFile);\n        throw new Error(`File '${path}' not found`);\n      }\n      return true;\n    }\n  }, fullOptions);\n}\n\n/**\n * Reads the content of a file safely from the vault.\n *\n * It covers the case when the file was removed during the reading.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to read.\n * @returns A promise that resolves to the content of the file or `null` if the file is missing or deleted.\n */\nexport async function readSafe(app: App, pathOrFile: PathOrFile): Promise<null | string> {\n  let content: null | string = null;\n  await invokeFileActionSafe(app, pathOrFile, async (file) => {\n    content = await app.vault.read(file);\n  });\n  return content;\n}\n\n/**\n * Renames a file safely in the vault.\n * If the new path already exists, the file will be renamed to an available path.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A promise that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const oldFile = getFile(app, oldPathOrFile, false, true);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrFile, newPath);\n\n  if (oldFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldFile.path !== newPath) {\n      await app.vault.rename(oldFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.vault.rename(oldFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\nasync function invokeFileActionSafe(app: App, pathOrFile: PathOrFile, fileAction: (file: TFile) => Promise<void>): Promise<boolean> {\n  const path = getPath(app, pathOrFile);\n  let file = getFileOrNull(app, path);\n  if (!file || file.deleted) {\n    return false;\n  }\n  try {\n    await fileAction(file);\n    return true;\n  } catch (e) {\n    file = getFileOrNull(app, path);\n    if (!file || file.deleted) {\n      return false;\n    }\n    throw e;\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAYA,6BAAiC;AASjC,mBAAiC;AACjC,sBAA0B;AAC1B,kBAKO;AACP,2BAA6B;AAC7B,wBAQO;AAoBP,eAAsB,SAAS,KAAU,eAA2B,SAAkC;AACpG,QAAM,WAAO,2BAAQ,KAAK,aAAa;AAEvC,QAAM,oBAAgB,yCAAiB,OAAO;AAC9C,QAAM,iBAAiB,KAAK,aAAa;AAEzC,QAAM,mBAAmB,iBAAiB,KAAK,OAAO;AAEtD,MAAI;AACF,UAAM,IAAI,MAAM,KAAK,MAAM,gBAAgB;AAAA,EAC7C,SAAS,GAAG;AACV,QAAI,CAAC,MAAM,IAAI,MAAM,OAAO,gBAAgB,GAAG;AAC7C,YAAM;AAAA,IACR;AAAA,EACF;AAEA,SAAO;AACT;AAUA,eAAsB,iBAAiB,KAAU,MAAgC;AAC/E,MAAI,MAAM,IAAI,MAAM,QAAQ,OAAO,IAAI,GAAG;AACxC,WAAO;AAAA,EACT;AAEA,MAAI;AACF,UAAM,IAAI,MAAM,aAAa,IAAI;AACjC,WAAO;AAAA,EACT,SAAS,GAAG;AACV,QAAI,CAAC,MAAM,IAAI,MAAM,OAAO,IAAI,GAAG;AACjC,YAAM;AAAA,IACR;AACA,WAAO;AAAA,EACT;AACF;AAQA,eAAsB,eAAe,KAAU,MAA4C;AACzF,MAAI,WAAO,iCAAc,KAAK,IAAI;AAClC,MAAI,MAAM;AACR,WAAO;AAAA,EACT;AAEA,QAAM,gBAAgB,MAAM,iBAAiB,SAAK,yCAAiB,IAAI,CAAC;AAExE,MAAI;AACF,UAAM,IAAI,MAAM,OAAO,MAAM,EAAE;AAAA,EACjC,SAAS,GAAG;AACV,QAAI,CAAC,MAAM,IAAI,MAAM,OAAO,IAAI,GAAG;AACjC,YAAM;AAAA,IACR;AAAA,EACF;AAEA,SAAO,YAAY;AACjB,eAAO,2BAAQ,KAAK,IAAI;AACxB,QAAI,CAAC,KAAK,SAAS;AACjB,YAAM,IAAI,YAAY,UAAU,IAAI;AAAA,IACtC;AACA,UAAM,cAAc;AAAA,EACtB;AACF;AAQA,eAAsB,iBAAiB,KAAU,MAA4C;AAC3F,MAAI,aAAS,mCAAgB,KAAK,IAAI;AACtC,MAAI,QAAQ;AACV,WAAO;AAAA,EACT;AAEA,QAAM,cAAU,yCAAiB,IAAI;AACrC,QAAM,iBAAiB,KAAK,OAAO;AAEnC,QAAM,gBAAgB,MAAM,iBAAiB,SAAK,yCAAiB,IAAI,CAAC;AAExE,QAAM,iBAAiB,KAAK,IAAI;AAEhC,SAAO,YAAY;AACjB,iBAAS,6BAAU,KAAK,IAAI;AAC5B,QAAI,CAAC,OAAO,SAAS;AACnB,YAAM,IAAI,YAAY,UAAU,MAAM;AAAA,IACxC;AACA,UAAM,cAAc;AAAA,EACtB;AACF;AASO,SAAS,iBAAiB,KAAU,MAAsB;AAC/D,QAAM,UAAM,qBAAQ,IAAI;AACxB,SAAO,IAAI,MAAM,qBAAiB,sBAAK,qBAAQ,IAAI,OAAG,sBAAS,MAAM,GAAG,CAAC,GAAG,IAAI,MAAM,CAAC,CAAC;AAC1F;AAQO,SAAS,uBAAuB,KAAmB;AACxD,SAAO,IAAI,MAAM,iBAAiB,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,KAAK,cAAc,EAAE,IAAI,CAAC;AACjF;AAOO,SAAS,mBAAmB,KAAmB;AACpD,SAAO,IAAI,MAAM,kBAAkB,EAAE,OAAO,CAAC,aAAS,0BAAO,IAAI,SAAK,0BAAO,KAAK,IAAI,CAAC,EAAE,KAAK,CAAC,GAAG,MAAM,EAAE,KAAK,cAAc,EAAE,IAAI,CAAC;AACtI;AAUO,SAAS,kBAAkB,KAAU,eAA2B,SAAyB;AAC9F,QAAM,cAAU,2BAAQ,KAAK,aAAa;AAE1C,MAAI,IAAI,MAAM,QAAQ,aAAa;AACjC,QAAI,iBAAa,qBAAQ,OAAO;AAChC,QAAI,sBAAkB,sBAAS,OAAO;AACtC,QAAI;AACJ,eAAS;AACP,mBAAS,mCAAgB,KAAK,YAAY,IAAI;AAC9C,UAAI,QAAQ;AACV;AAAA,MACF;AACA,4BAAkB,sBAAK,sBAAS,UAAU,GAAG,eAAe;AAC5D,uBAAa,qBAAQ,UAAU;AAAA,IACjC;AACA,kBAAU,kBAAK,OAAO,gBAAgB,GAAG,eAAe;AAAA,EAC1D;AAEA,MAAI,QAAQ,YAAY,MAAM,QAAQ,YAAY,GAAG;AACnD,WAAO;AAAA,EACT;AAEA,SAAO,iBAAiB,KAAK,OAAO;AACtC;AAQA,eAAsB,cAAc,KAAU,cAA8C;AAC1F,QAAM,cAAc,MAAM,SAAS,SAAK,2BAAQ,KAAK,YAAY,CAAC;AAClE,SAAO,YAAY,MAAM,WAAW,KAAK,YAAY,QAAQ,WAAW;AAC1E;AASA,eAAsB,SAAS,KAAU,cAAkD;AACzF,QAAM,WAAO,2BAAQ,KAAK,YAAY;AACtC,QAAM,QAAQ,EAAE,OAAO,CAAC,GAAG,SAAS,CAAC,EAAE;AAEvC,OAAK,MAAM,IAAI,MAAM,QAAQ,KAAK,IAAI,IAAI,SAAS,UAAU;AAC3D,WAAO;AAAA,EACT;AAEA,MAAI;AACF,WAAO,MAAM,IAAI,MAAM,QAAQ,KAAK,IAAI;AAAA,EAC1C,SAAS,GAAG;AACV,QAAI,MAAM,IAAI,MAAM,OAAO,IAAI,GAAG;AAChC,YAAM;AAAA,IACR;AACA,WAAO;AAAA,EACT;AACF;AAgBA,eAAsB,QACpB,KACA,YACA,oBACA,UAA0B,CAAC,GACZ;AACf,QAAM,wBAAwB;AAAA,IAC5B,yBAAyB;AAAA,EAC3B;AACA,QAAM,cAAc,EAAE,GAAG,uBAAuB,GAAG,QAAQ;AAE3D,YAAM,+BAAiB,YAAY;AACjC,UAAM,aAAa,MAAM,SAAS,KAAK,UAAU;AAEjD,QAAI,eAAe,MAAM;AACvB,aAAO,kBAAkB;AAAA,IAC3B;AAEA,UAAM,aAAa,UAAM,mCAAa,oBAAoB,UAAU;AACpE,QAAI,eAAe,MAAM;AACvB,aAAO;AAAA,IACT;AAEA,QAAI,YAAY;AAChB,UAAM,gBAAgB,MAAM,qBAAqB,KAAK,YAAY,OAAO,SAAS;AAChF,YAAM,IAAI,MAAM,QAAQ,MAAM,CAAC,YAAY;AACzC,YAAI,YAAY,YAAY;AAC1B,kBAAQ,KAAK,sDAAsD;AAAA,YACjE,eAAe;AAAA,YACf,iBAAiB;AAAA,YACjB,MAAM,KAAK;AAAA,UACb,CAAC;AACD,sBAAY;AACZ,iBAAO;AAAA,QACT;AAEA,eAAO;AAAA,MACT,CAAC;AAAA,IACH,CAAC;AAED,QAAI,CAAC,eAAe;AAClB,aAAO,kBAAkB;AAAA,IAC3B;AAEA,WAAO;AAEP,aAAS,oBAA6B;AACpC,UAAI,YAAY,yBAAyB;AACvC,cAAM,WAAO,2BAAQ,KAAK,UAAU;AACpC,cAAM,IAAI,MAAM,SAAS,IAAI,aAAa;AAAA,MAC5C;AACA,aAAO;AAAA,IACT;AAAA,EACF,GAAG,WAAW;AAChB;AAWA,eAAsB,SAAS,KAAU,YAAgD;AACvF,MAAI,UAAyB;AAC7B,QAAM,qBAAqB,KAAK,YAAY,OAAO,SAAS;AAC1D,cAAU,MAAM,IAAI,MAAM,KAAK,IAAI;AAAA,EACrC,CAAC;AACD,SAAO;AACT;AAWA,eAAsB,WAAW,KAAU,eAA2B,SAAkC;AACtG,QAAM,cAAU,2BAAQ,KAAK,eAAe,OAAO,IAAI;AAEvD,QAAM,mBAAmB,kBAAkB,KAAK,eAAe,OAAO;AAEtE,MAAI,QAAQ,KAAK,YAAY,MAAM,iBAAiB,YAAY,GAAG;AACjE,QAAI,QAAQ,SAAS,SAAS;AAC5B,YAAM,IAAI,MAAM,OAAO,SAAS,gBAAgB;AAAA,IAClD;AACA,WAAO;AAAA,EACT;AAEA,QAAM,oBAAgB,yCAAiB,gBAAgB;AACvD,QAAM,iBAAiB,KAAK,aAAa;AAEzC,MAAI;AACF,UAAM,IAAI,MAAM,OAAO,SAAS,gBAAgB;AAAA,EAClD,SAAS,GAAG;AACV,QAAI,CAAC,MAAM,IAAI,MAAM,OAAO,gBAAgB,KAAK,MAAM,IAAI,MAAM,OAAO,QAAQ,IAAI,GAAG;AACrF,YAAM;AAAA,IACR;AAAA,EACF;AAEA,SAAO;AACT;AAEA,eAAe,qBAAqB,KAAU,YAAwB,YAA8D;AAClI,QAAM,WAAO,2BAAQ,KAAK,UAAU;AACpC,MAAI,WAAO,iCAAc,KAAK,IAAI;AAClC,MAAI,CAAC,QAAQ,KAAK,SAAS;AACzB,WAAO;AAAA,EACT;AACA,MAAI;AACF,UAAM,WAAW,IAAI;AACrB,WAAO;AAAA,EACT,SAAS,GAAG;AACV,eAAO,iCAAc,KAAK,IAAI;AAC9B,QAAI,CAAC,QAAQ,KAAK,SAAS;AACzB,aAAO;AAAA,IACT;AACA,UAAM;AAAA,EACR;AACF;",
  "names": []
}

|
267
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Vault\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  ListedFiles,\n  TFile,\n  TFolder\n} from 'obsidian';\n\nimport { parentFolderPath } from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { retryWithTimeout } from '../Async.ts';\nimport { noopAsync } from '../Function.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join\n} from '../Path.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isNote\n} from './FileSystem.ts';\n\n/**\n * Options for the `process` function.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * If `true`, the function will throw an error if the file is missing or deleted.\n   */\n  shouldFailOnMissingFile?: boolean;\n}\n\n/**\n * Copies a file safely in the vault.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to copy.\n * @param newPath - The new path to copy the file to.\n * @returns A promise that resolves to the new path of the copied file.\n */\nexport async function copySafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const file = getFile(app, oldPathOrFile);\n\n  const newFolderPath = parentFolderPath(newPath);\n  await createFolderSafe(app, newFolderPath);\n\n  const newAvailablePath = getAvailablePath(app, newPath);\n\n  try {\n    await app.vault.copy(file, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.createFolder(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n    return true;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = getFileOrNull(app, path);\n  if (file) {\n    return noopAsync;\n  }\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  try {\n    await app.vault.create(path, '');\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  return async () => {\n    file = getFile(app, path);\n    if (!file.deleted) {\n      await app.fileManager.trashFile(file);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = getFolderOrNull(app, path);\n  if (folder) {\n    return noopAsync;\n  }\n\n  const dirPath = parentFolderPath(path);\n  await createTempFolder(app, dirPath);\n\n  const folderCleanup = await createTempFolder(app, parentFolderPath(path));\n\n  await createFolderSafe(app, path);\n\n  return async () => {\n    folder = getFolder(app, path);\n    if (!folder.deleted) {\n      await app.fileManager.trashFile(folder);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Gets an available path for a file in the vault.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get an available path for.\n * @returns The available path for the file.\n */\nexport function getAvailablePath(app: App, path: string): string {\n  const ext = extname(path);\n  return app.vault.getAvailablePath(join(dirname(path), basename(path, ext)), ext.slice(1));\n}\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Retrieves an array of all note files from the app's vault and sorts them alphabetically by their file path.\n * @param app - The Obsidian app instance.\n * @returns An array of all note files in the vault sorted by file path.\n */\nexport function getNoteFilesSorted(app: App): TFile[] {\n  return app.vault.getAllLoadedFiles().filter((file) => isFile(file) && isNote(app, file)).sort((a, b) => a.path.localeCompare(b.path)) as TFile[];\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns The safe rename path for the file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrFile: PathOrFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrFile);\n\n  if (app.vault.adapter.insensitive) {\n    let folderPath = dirname(newPath);\n    let nonExistingPath = basename(newPath);\n    let folder: null | TFolder;\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    while (true) {\n      folder = getFolderOrNull(app, folderPath, true);\n      if (folder) {\n        break;\n      }\n      nonExistingPath = join(basename(folderPath), nonExistingPath);\n      folderPath = dirname(folderPath);\n    }\n    newPath = join(folder.getParentPrefix(), nonExistingPath);\n  }\n\n  if (oldPath.toLowerCase() === newPath.toLowerCase()) {\n    return newPath;\n  }\n\n  return getAvailablePath(app, newPath);\n}\n\n/**\n * Checks if a folder is empty.\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A promise that resolves to a boolean indicating whether the folder is empty.\n */\nexport async function isEmptyFolder(app: App, pathOrFolder: PathOrFolder): Promise<boolean> {\n  const listedFiles = await listSafe(app, getPath(app, pathOrFolder));\n  return listedFiles.files.length === 0 && listedFiles.folders.length === 0;\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to list.\n * @returns A promise that resolves to a `ListedFiles` object containing the listed files and folders.\n */\nexport async function listSafe(app: App, pathOrFolder: PathOrFolder): Promise<ListedFiles> {\n  const path = getPath(app, pathOrFolder);\n  const EMPTY = { files: [], folders: [] };\n\n  if ((await app.vault.adapter.stat(path))?.type !== 'folder') {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param options - Optional options for processing/retrying the operation.\n *\n * @returns A promise that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function process(\n  app: App,\n  pathOrFile: PathOrFile,\n  newContentProvider: ValueProvider<null | string, [string]>,\n  options: ProcessOptions = {}\n): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS = {\n    shouldFailOnMissingFile: true\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n\n  await retryWithTimeout(async () => {\n    const oldContent = await readSafe(app, pathOrFile);\n\n    if (oldContent === null) {\n      return handleMissingFile();\n    }\n\n    const newContent = await resolveValue(newContentProvider, oldContent);\n    if (newContent === null) {\n      return false;\n    }\n\n    let isSuccess = true;\n    const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n      await app.vault.process(file, (content) => {\n        if (content !== oldContent) {\n          console.warn('Content has changed since it was read. Retrying...', {\n            actualContent: content,\n            expectedContent: oldContent,\n            path: file.path\n          });\n          isSuccess = false;\n          return content;\n        }\n\n        return newContent;\n      });\n    });\n\n    if (!doesFileExist) {\n      return handleMissingFile();\n    }\n\n    return isSuccess;\n\n    function handleMissingFile(): boolean {\n      if (fullOptions.shouldFailOnMissingFile) {\n        const path = getPath(app, pathOrFile);\n        throw new Error(`File '${path}' not found`);\n      }\n      return true;\n    }\n  }, fullOptions);\n}\n\n/**\n * Reads the content of a file safely from the vault.\n *\n * It covers the case when the file was removed during the reading.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to read.\n * @returns A promise that resolves to the content of the file or `null` if the file is missing or deleted.\n */\nexport async function readSafe(app: App, pathOrFile: PathOrFile): Promise<null | string> {\n  let content: null | string = null;\n  await invokeFileActionSafe(app, pathOrFile, async (file) => {\n    content = await app.vault.read(file);\n  });\n  return content;\n}\n\n/**\n * Renames a file safely in the vault.\n * If the new path already exists, the file will be renamed to an available path.\n *\n * @param app - The application instance.\n * @param oldPathOrFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A promise that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrFile: PathOrFile, newPath: string): Promise<string> {\n  const oldFile = getFile(app, oldPathOrFile, false, true);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrFile, newPath);\n\n  if (oldFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldFile.path !== newPath) {\n      await app.vault.rename(oldFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.vault.rename(oldFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\nasync function invokeFileActionSafe(app: App, pathOrFile: PathOrFile, fileAction: (file: TFile) => Promise<void>): Promise<boolean> {\n  const path = getPath(app, pathOrFile);\n  let file = getFileOrNull(app, path);\n  if (!file || file.deleted) {\n    return false;\n  }\n  try {\n    await fileAction(file);\n    return true;\n  } catch (e) {\n    file = getFileOrNull(app, path);\n    if (!file || file.deleted) {\n      return false;\n    }\n    throw e;\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/dist/lib/esm/Async.mjs
CHANGED
@@ -65,7 +65,7 @@ async function retryWithTimeout(fn, retryOptions, stackTrace) {
|
|
65
65
|
const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };
|
66
66
|
await runWithTimeout(fullOptions.timeoutInMilliseconds, async () => {
|
67
67
|
let attempt = 0;
|
68
|
-
|
68
|
+
while (true) {
|
69
69
|
fullOptions.abortSignal?.throwIfAborted();
|
70
70
|
attempt++;
|
71
71
|
let isSuccess;
|
@@ -180,4 +180,4 @@ export {
|
|
180
180
|
timeout,
|
181
181
|
toArray
|
182
182
|
};
|
183
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Async\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { getLibDebugger } from './Debug.ts';\nimport {\n  ASYNC_ERROR_WRAPPER_MESSAGE,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError\n} from './Error.ts';\n\n/**\n * A type representing a function that resolves a Promise.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for configuring the retry behavior.\n */\nexport interface RetryOptions {\n  /**\n   * The abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * The delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * The maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * A marker interface to indicate that an error should terminate retry logic.\n */\nexport interface TerminateRetry {\n  /**\n   * A marker property to indicate that an error should terminate retry logic.\n   */\n  __terminateRetry: true;\n}\n\n/**\n * Adds an error handler to a Promise that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @returns A Promise that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>): Promise<void> {\n  const asyncErrorWrapper = new Error(ASYNC_ERROR_WRAPPER_MESSAGE);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    asyncErrorWrapper.cause = asyncError;\n    emitAsyncErrorEvent(asyncErrorWrapper);\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A Promise that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const predicateResults = await asyncMap(arr, predicate);\n  return arr.filter((_, index) => predicateResults[index] ?? false);\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A Promise that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A Promise that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await Promise.all(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>): (...args: Args) => void {\n  return (...args: Args): void => {\n    invokeAsyncSafely(() => asyncFunc(...args));\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a Promise.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return (...args: Args): Promise<Result> => Promise.resolve().then(() => syncFn(...args));\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A Promise that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A Promise that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  try {\n    return await promise;\n  } catch {\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a Promise and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>): void {\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn);\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n * @returns An error that should terminate retry logic.\n */\nexport function marksAsTerminateRetry<TError extends Error>(error: TError): TerminateRetry & TError {\n  return Object.assign(error, { __terminateRetry: true }) as TerminateRetry & TError;\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A Promise that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: () => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await runWithTimeout(fullOptions.timeoutInMilliseconds, async () => {\n    let attempt = 0;\n    for (;;) {\n      fullOptions.abortSignal?.throwIfAborted();\n      attempt++;\n      let isSuccess: boolean;\n      try {\n        isSuccess = await fn();\n      } catch (error) {\n        if (!fullOptions.shouldRetryOnError || (error as Partial<TerminateRetry>).__terminateRetry) {\n          throw error;\n        }\n        printError(error);\n        isSuccess = false;\n      }\n      if (isSuccess) {\n        if (attempt > 1) {\n          retryWithTimeoutDebugger(`Retry completed successfully after ${attempt.toString()} attempts`);\n          retryWithTimeoutDebugger.printStackTrace(stackTrace);\n        }\n        return;\n      }\n\n      retryWithTimeoutDebugger(\n        `Retry attempt ${attempt.toString()} completed unsuccessfully. Trying again in ${fullOptions.retryDelayInMilliseconds.toString()} milliseconds`,\n        {\n          fn\n        }\n      );\n      retryWithTimeoutDebugger.printStackTrace(stackTrace);\n      await sleep(fullOptions.retryDelayInMilliseconds);\n    }\n  });\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @returns A Promise that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(timeoutInMilliseconds: number, fn: () => Promisable<R>): Promise<R> {\n  let isTimedOut = true;\n  let result: R = null as R;\n  const startTime = performance.now();\n  await Promise.race([run(), innerTimeout()]);\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n  if (isTimedOut) {\n    throw new Error('Timed out');\n  }\n  return result;\n\n  async function run(): Promise<void> {\n    result = await fn();\n    isTimedOut = false;\n    const duration = performance.now() - startTime;\n    getLibDebugger('Async:runWithTimeout')(`Execution time: ${duration.toString()} milliseconds`, { fn });\n  }\n\n  async function innerTimeout(): Promise<void> {\n    if (!isTimedOut) {\n      return;\n    }\n    await sleep(timeoutInMilliseconds);\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    if (!isTimedOut) {\n      return;\n    }\n    const duration = performance.now() - startTime;\n    console.warn(`Timed out in ${duration.toString()} milliseconds`, { fn });\n    const _debugger = getLibDebugger('Async:runWithTimeout:timeout');\n    if (_debugger.enabled) {\n      _debugger(\n        `The execution is not terminated because debugger ${_debugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging for more information`\n      );\n      await innerTimeout();\n    }\n  }\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @returns A Promise that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, milliseconds);\n  });\n}\n\n/**\n * Returns a Promise that rejects when the abort signal is aborted.\n *\n * @param abortSignal - The abort signal to listen to.\n * @returns A Promise that rejects when the abort signal is aborted.\n */\nexport function throwOnAbort(abortSignal: AbortSignal): Promise<void> {\n  return new Promise((_resolve, reject) => {\n    if (handleAbort()) {\n      return;\n    }\n    abortSignal.addEventListener('abort', handleAbort, { once: true });\n\n    function handleAbort(): boolean {\n      try {\n        abortSignal.throwIfAborted();\n        return false;\n      } catch (e) {\n        reject(e as Error);\n        return true;\n      }\n    }\n  });\n}\n\n/**\n * Returns a Promise that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @returns A Promise that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number): Promise<never> {\n  await sleep(timeoutInMilliseconds);\n  throw new Error(`Timed out in ${timeoutInMilliseconds.toString()} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A Promise that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": "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",
  "names": []
}

|
183
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Async.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Async\n * Contains utility functions for asynchronous operations.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport { getLibDebugger } from './Debug.ts';\nimport {\n  ASYNC_ERROR_WRAPPER_MESSAGE,\n  emitAsyncErrorEvent,\n  getStackTrace,\n  printError\n} from './Error.ts';\n\n/**\n * A type representing a function that resolves a Promise.\n *\n * @typeParam T - The type of the value.\n */\nexport type PromiseResolve<T> = undefined extends T ? (value?: PromiseLike<T> | T) => void\n  : (value: PromiseLike<T> | T) => void;\n\n/**\n * Options for configuring the retry behavior.\n */\nexport interface RetryOptions {\n  /**\n   * The abort signal to cancel the retry operation.\n   */\n  abortSignal?: AbortSignal;\n\n  /**\n   * The delay in milliseconds between retry attempts.\n   */\n  retryDelayInMilliseconds?: number;\n\n  /**\n   * Whether to retry the function on error.\n   */\n  shouldRetryOnError?: boolean;\n\n  /**\n   * The maximum time in milliseconds to wait before giving up on retrying.\n   */\n  timeoutInMilliseconds?: number;\n}\n\n/**\n * A marker interface to indicate that an error should terminate retry logic.\n */\nexport interface TerminateRetry {\n  /**\n   * A marker property to indicate that an error should terminate retry logic.\n   */\n  __terminateRetry: true;\n}\n\n/**\n * Adds an error handler to a Promise that catches any errors and emits an async error event.\n *\n * @param asyncFn - The asynchronous function to add an error handler to.\n * @returns A Promise that resolves when the asynchronous function completes or emits async error event.\n */\nexport async function addErrorHandler(asyncFn: () => Promise<unknown>): Promise<void> {\n  const asyncErrorWrapper = new Error(ASYNC_ERROR_WRAPPER_MESSAGE);\n  try {\n    await asyncFn();\n  } catch (asyncError) {\n    asyncErrorWrapper.cause = asyncError;\n    emitAsyncErrorEvent(asyncErrorWrapper);\n  }\n}\n\n/**\n * Filters an array asynchronously, keeping only the elements that satisfy the provided predicate function.\n *\n * @typeParam T - The type of elements in the input array.\n * @param arr - The array to filter.\n * @param predicate - The predicate function to test each element.\n * @returns A Promise that resolves with an array of elements that satisfy the predicate function.\n */\nexport async function asyncFilter<T>(arr: T[], predicate: (value: T, index: number, array: T[]) => Promisable<boolean>): Promise<T[]> {\n  const predicateResults = await asyncMap(arr, predicate);\n  return arr.filter((_, index) => predicateResults[index] ?? false);\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element, and then flattens the results into a single array.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over and flatten.\n * @param callback - The callback function to apply to each element.\n * @returns A Promise that resolves with a flattened array of the results of the callback function.\n */\nexport async function asyncFlatMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U[]>): Promise<U[]> {\n  return (await asyncMap(arr, callback)).flat();\n}\n\n/**\n * Maps over an array asynchronously, applying the provided callback function to each element.\n *\n * @typeParam T - The type of elements in the input array.\n * @typeParam U - The type of elements in the output array.\n * @param arr - The array to map over.\n * @param callback - The callback function to apply to each element.\n * @returns A Promise that resolves with an array of the results of the callback function.\n */\nexport async function asyncMap<T, U>(arr: T[], callback: (value: T, index: number, array: T[]) => Promisable<U>): Promise<U[]> {\n  return await Promise.all(arr.map(callback));\n}\n\n/**\n * Converts an asynchronous function to a synchronous one by automatically handling the Promise rejection.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @param asyncFunc - The asynchronous function to convert.\n * @returns A function that wraps the asynchronous function in a synchronous interface.\n */\nexport function convertAsyncToSync<Args extends unknown[]>(asyncFunc: (...args: Args) => Promise<unknown>): (...args: Args) => void {\n  return (...args: Args): void => {\n    invokeAsyncSafely(() => asyncFunc(...args));\n  };\n}\n\n/**\n * Converts a synchronous function to an asynchronous one by wrapping it in a Promise.\n *\n * @typeParam Args - The types of the arguments the function accepts.\n * @typeParam Result - The type of the function's return value.\n * @param syncFn - The synchronous function to convert.\n * @returns A function that wraps the synchronous function in an asynchronous interface.\n */\nexport function convertSyncToAsync<Args extends unknown[], Result>(syncFn: (...args: Args) => Result): (...args: Args) => Promise<Result> {\n  return (...args: Args): Promise<Result> => Promise.resolve().then(() => syncFn(...args));\n}\n\n/**\n * Ignores an error that is thrown by an asynchronous function.\n *\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - Always `undefined`.\n * @returns A Promise that resolves when the asynchronous function completes or fails.\n */\nexport async function ignoreError(promise: Promise<unknown>, fallbackValue?: undefined): Promise<void>;\n\n/**\n * Invokes an asynchronous function and returns a fallback value if an error is thrown.\n *\n * @typeParam T - The type of the value returned by the asynchronous function.\n * @param promise - The promise to ignore the error of.\n * @param fallbackValue - The value to return if an error is thrown.\n * @returns A Promise that resolves with the value returned by the asynchronous function or the fallback value if an error is thrown.\n */\nexport async function ignoreError<T>(promise: Promise<T>, fallbackValue: T): Promise<T> {\n  try {\n    return await promise;\n  } catch {\n    return fallbackValue;\n  }\n}\n\n/**\n * Invokes a Promise and safely handles any errors by catching them and emitting an async error event.\n *\n * @param asyncFn - The asynchronous function to invoke safely.\n */\nexport function invokeAsyncSafely(asyncFn: () => Promise<unknown>): void {\n  // eslint-disable-next-line no-void\n  void addErrorHandler(asyncFn);\n}\n\n/**\n * Marks an error to terminate retry logic.\n *\n * @param error - The error to mark to terminate retry logic.\n * @returns An error that should terminate retry logic.\n */\nexport function marksAsTerminateRetry<TError extends Error>(error: TError): TerminateRetry & TError {\n  return Object.assign(error, { __terminateRetry: true }) as TerminateRetry & TError;\n}\n\n/**\n * Retries the provided function until it returns true or the timeout is reached.\n *\n * @param fn - The function to retry.\n * @param retryOptions - Optional parameters to configure the retry behavior.\n * @param stackTrace - Optional stack trace.\n * @returns A Promise that resolves when the function returns true or rejects when the timeout is reached.\n */\nexport async function retryWithTimeout(fn: () => Promisable<boolean>, retryOptions?: RetryOptions, stackTrace?: string): Promise<void> {\n  const retryWithTimeoutDebugger = getLibDebugger('Async:retryWithTimeout');\n  stackTrace ??= getStackTrace(1);\n  const DEFAULT_RETRY_OPTIONS = {\n    // eslint-disable-next-line no-magic-numbers\n    retryDelayInMilliseconds: 100,\n    shouldRetryOnError: false,\n    // eslint-disable-next-line no-magic-numbers\n    timeoutInMilliseconds: 5000\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await runWithTimeout(fullOptions.timeoutInMilliseconds, async () => {\n    let attempt = 0;\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    while (true) {\n      fullOptions.abortSignal?.throwIfAborted();\n      attempt++;\n      let isSuccess: boolean;\n      try {\n        isSuccess = await fn();\n      } catch (error) {\n        if (!fullOptions.shouldRetryOnError || (error as Partial<TerminateRetry>).__terminateRetry) {\n          throw error;\n        }\n        printError(error);\n        isSuccess = false;\n      }\n      if (isSuccess) {\n        if (attempt > 1) {\n          retryWithTimeoutDebugger(`Retry completed successfully after ${attempt.toString()} attempts`);\n          retryWithTimeoutDebugger.printStackTrace(stackTrace);\n        }\n        return;\n      }\n\n      retryWithTimeoutDebugger(\n        `Retry attempt ${attempt.toString()} completed unsuccessfully. Trying again in ${fullOptions.retryDelayInMilliseconds.toString()} milliseconds`,\n        {\n          fn\n        }\n      );\n      retryWithTimeoutDebugger.printStackTrace(stackTrace);\n      await sleep(fullOptions.retryDelayInMilliseconds);\n    }\n  });\n}\n\n/**\n * Executes a function with a timeout. If the function does not complete within the specified time, it is considered to have timed out.\n *\n * If `DEBUG=obsidian-dev-utils:Async:runWithTimeout` is set, the execution is not terminated after the timeout and the function is allowed to run indefinitely.\n *\n * @typeParam R - The type of the result from the asynchronous function.\n * @param timeoutInMilliseconds - The maximum time to wait in milliseconds.\n * @param fn - The function to execute.\n * @returns A Promise that resolves with the result of the asynchronous function or rejects if it times out.\n */\nexport async function runWithTimeout<R>(timeoutInMilliseconds: number, fn: () => Promisable<R>): Promise<R> {\n  let isTimedOut = true;\n  let result: R = null as R;\n  const startTime = performance.now();\n  await Promise.race([run(), innerTimeout()]);\n  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n  if (isTimedOut) {\n    throw new Error('Timed out');\n  }\n  return result;\n\n  async function run(): Promise<void> {\n    result = await fn();\n    isTimedOut = false;\n    const duration = performance.now() - startTime;\n    getLibDebugger('Async:runWithTimeout')(`Execution time: ${duration.toString()} milliseconds`, { fn });\n  }\n\n  async function innerTimeout(): Promise<void> {\n    if (!isTimedOut) {\n      return;\n    }\n    await sleep(timeoutInMilliseconds);\n    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition\n    if (!isTimedOut) {\n      return;\n    }\n    const duration = performance.now() - startTime;\n    console.warn(`Timed out in ${duration.toString()} milliseconds`, { fn });\n    const _debugger = getLibDebugger('Async:runWithTimeout:timeout');\n    if (_debugger.enabled) {\n      _debugger(\n        `The execution is not terminated because debugger ${_debugger.namespace} is enabled. See https://github.com/mnaoumov/obsidian-dev-utils/?tab=readme-ov-file#debugging for more information`\n      );\n      await innerTimeout();\n    }\n  }\n}\n\n/**\n * Delays execution for a specified number of milliseconds.\n *\n * @param milliseconds - The time to wait in milliseconds.\n * @returns A Promise that resolves after the specified delay.\n */\nexport async function sleep(milliseconds: number): Promise<void> {\n  await new Promise((resolve) => {\n    setTimeout(resolve, milliseconds);\n  });\n}\n\n/**\n * Returns a Promise that rejects when the abort signal is aborted.\n *\n * @param abortSignal - The abort signal to listen to.\n * @returns A Promise that rejects when the abort signal is aborted.\n */\nexport function throwOnAbort(abortSignal: AbortSignal): Promise<void> {\n  return new Promise((_resolve, reject) => {\n    if (handleAbort()) {\n      return;\n    }\n    abortSignal.addEventListener('abort', handleAbort, { once: true });\n\n    function handleAbort(): boolean {\n      try {\n        abortSignal.throwIfAborted();\n        return false;\n      } catch (e) {\n        reject(e as Error);\n        return true;\n      }\n    }\n  });\n}\n\n/**\n * Returns a Promise that rejects after the specified timeout period.\n *\n * @param timeoutInMilliseconds - The timeout period in milliseconds.\n * @returns A Promise that always rejects with a timeout error.\n */\nexport async function timeout(timeoutInMilliseconds: number): Promise<never> {\n  await sleep(timeoutInMilliseconds);\n  throw new Error(`Timed out in ${timeoutInMilliseconds.toString()} milliseconds`);\n}\n\n/**\n * Converts an AsyncIterableIterator to an array by consuming all its elements.\n *\n * @typeParam T - The type of elements produced by the AsyncIterableIterator.\n * @param iter - The AsyncIterableIterator to convert.\n * @returns A Promise that resolves with an array of all the elements in the AsyncIterableIterator.\n */\nexport async function toArray<T>(iter: AsyncIterableIterator<T>): Promise<T[]> {\n  const arr: T[] = [];\n  for await (const item of iter) {\n    arr.push(item);\n  }\n  return arr;\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/dist/lib/esm/Library.mjs
CHANGED
@@ -5,7 +5,7 @@ if you want to view the source, please visit the github repository of this plugi
|
|
5
5
|
|
6
6
|
(function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd:__name(()=>"/","cwd"),env:{},platform:"android"};globalThis.process=browserProcess})();
|
7
7
|
|
8
|
-
const LIBRARY_VERSION = "22.
|
8
|
+
const LIBRARY_VERSION = "22.1.1-beta.1";
|
9
9
|
const LIBRARY_NAME = "obsidian-dev-utils";
|
10
10
|
const LIBRARY_STYLES = ".obsidian-dev-utils input[type=url] {\n height: var(--input-height);\n}\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=url],\n.obsidian-dev-utils input[type=week] {\n -webkit-app-region: no-drag;\n background: var(--background-modifier-form-field);\n border: var(--input-border-width) solid var(--background-modifier-border);\n color: var(--text-normal);\n font-family: inherit;\n padding: var(--size-4-1) var(--size-4-2);\n font-size: var(--font-ui-small);\n border-radius: var(--input-radius);\n outline: none;\n}\n@media (hover: hover) {\n .obsidian-dev-utils input[type=month]:hover,\n .obsidian-dev-utils input[type=time]:hover,\n .obsidian-dev-utils input[type=url]:hover,\n .obsidian-dev-utils input[type=week]:hover {\n border-color: var(--background-modifier-border-hover);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n }\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus {\n border-color: var(--background-modifier-border-focus);\n transition: box-shadow 0.15s ease-in-out, border 0.15s ease-in-out;\n}\n.obsidian-dev-utils input[type=month]:active, .obsidian-dev-utils input[type=month]:focus, .obsidian-dev-utils input[type=month]:focus-visible,\n.obsidian-dev-utils input[type=time]:active,\n.obsidian-dev-utils input[type=time]:focus,\n.obsidian-dev-utils input[type=time]:focus-visible,\n.obsidian-dev-utils input[type=url]:active,\n.obsidian-dev-utils input[type=url]:focus,\n.obsidian-dev-utils input[type=url]:focus-visible,\n.obsidian-dev-utils input[type=week]:active,\n.obsidian-dev-utils input[type=week]:focus,\n.obsidian-dev-utils input[type=week]:focus-visible {\n box-shadow: 0 0 0 2px var(--background-modifier-border-focus);\n}\n.obsidian-dev-utils input[type=month]::placeholder,\n.obsidian-dev-utils input[type=time]::placeholder,\n.obsidian-dev-utils input[type=url]::placeholder,\n.obsidian-dev-utils input[type=week]::placeholder {\n color: var(--text-faint);\n}\n.mod-rtl input[type=month],\n.mod-rtl input[type=time],\n.mod-rtl input[type=week],\n.is-rtl input[type=month],\n.is-rtl input[type=time],\n.is-rtl input[type=week],\n.rtl input[type=month],\n.rtl input[type=time],\n.rtl input[type=week] {\n direction: rtl;\n}\n.mod-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=month]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl input[type=week]::-webkit-calendar-picker-indicator,\n.rtl input[type=month]::-webkit-calendar-picker-indicator,\n.rtl input[type=time]::-webkit-calendar-picker-indicator,\n.rtl input[type=week]::-webkit-calendar-picker-indicator {\n right: var(--size-4-1);\n left: auto;\n}\n\n.obsidian-dev-utils input[type=month],\n.obsidian-dev-utils input[type=time],\n.obsidian-dev-utils input[type=week] {\n font-variant-numeric: tabular-nums;\n position: relative;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-text,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-text {\n color: var(--text-faint);\n padding-inline-end: 0;\n}\n.obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n position: absolute;\n left: var(--size-4-1);\n right: auto;\n opacity: 0.5;\n}\n.obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-month-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-day-field:focus, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:active, .obsidian-dev-utils input[type=month]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=time]::-webkit-datetime-edit-year-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-month-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-day-field:focus,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:active,\n.obsidian-dev-utils input[type=week]::-webkit-datetime-edit-year-field:focus {\n background-color: var(--text-selection);\n color: var(--text-normal);\n cursor: text;\n}\n.mod-rtl .obsidian-dev-utils input[type=month], .is-rtl .obsidian-dev-utils input[type=month], .rtl .obsidian-dev-utils input[type=month],\n.mod-rtl .obsidian-dev-utils input[type=time],\n.is-rtl .obsidian-dev-utils input[type=time],\n.rtl .obsidian-dev-utils input[type=time],\n.mod-rtl .obsidian-dev-utils input[type=week],\n.is-rtl .obsidian-dev-utils input[type=week],\n.rtl .obsidian-dev-utils input[type=week] {\n direction: rtl;\n}\n.mod-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .is-rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator, .rtl .obsidian-dev-utils input[type=month]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator,\n.mod-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.is-rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator,\n.rtl .obsidian-dev-utils input[type=week]::-webkit-calendar-picker-indicator {\n left: auto;\n right: var(--size-4-1);\n}\n\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=month],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=time],\nbody:not(.is-ios):not(.is-android) .obsidian-dev-utils input[type=week] {\n padding-inline-start: var(--size-4-6);\n}\n\n.obsidian-dev-utils input[type=time]::-webkit-calendar-picker-indicator {\n margin-inline-start: 0;\n}\n\n.obsidian-dev-utils.modal-container .ok-button {\n margin-right: 10px;\n margin-top: 20px;\n}\n\n.obsidian-dev-utils .multiple-dropdown-component select,\n.obsidian-dev-utils .multiple-dropdown-component select:focus,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown {\n height: auto;\n}\n.obsidian-dev-utils .multiple-dropdown-component select option:checked,\n.obsidian-dev-utils .multiple-dropdown-component select:focus option:checked,\n.obsidian-dev-utils .multiple-dropdown-component .dropdown option:checked {\n background-color: #1967d2;\n color: #fff;\n}\n\n.obsidian-dev-utils.prompt-modal .text-box {\n width: 100%;\n}\n\n.obsidian-dev-utils :invalid {\n box-shadow: 0 0 0 2px var(--text-error);\n}\n\n/*# sourceMappingURL=data:application/json;charset=utf-8,%7B%22version%22:3,%22sourceRoot%22:%22%22,%22sources%22:%5B%22../src/styles/input.scss%22,%22../src/styles/input-time.scss%22,%22../src/styles/modal-container.scss%22,%22../src/styles/multiple-dropdown-component.scss%22,%22../src/styles/prompt-modal.scss%22,%22../src/styles/validation.scss%22%5D,%22names%22:%5B%5D,%22mappings%22:%22AACE;EACE;;AAGF;AAAA;AAAA;AAAA;EAIE;EACA;EACA;EACA;EACA;EACA;EACA;EACA;EACA;;AAGE;EACE;AAAA;AAAA;AAAA;IACE;IACA,YACE;;;AAMR;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA,YACE;;AAIJ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAGF;AAAA;AAAA;AAAA;EACE;;AASE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AC5DV;AAAA;AAAA;EAGE;EACA;;AAEA;AAAA;AAAA;EACE;EACA;;AAGF;AAAA;AAAA;EACE;EACA;EACA;EACA;;AAMA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAEE;EACA;EACA;;AAIK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EAGP;;AAEA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;EACE;EACA;;;AAKF;AAAA;AAAA;EACE;;;AAMJ;EACE;;;AChDF;EACE;EACA;;;ACFF;AAAA;AAAA;EAGE;;AAEA;AAAA;AAAA;EACE;EACA;;;ACPJ;EACE;;;ACFJ;EACE%22,%22file%22:%22styles.css%22,%22sourcesContent%22:%5B%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='url'%5D%20%7B%5Cn%20%20%20%20height:%20var(--input-height)%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='url'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20-webkit-app-region:%20no-drag;%5Cn%20%20%20%20background:%20var(--background-modifier-form-field);%5Cn%20%20%20%20border:%20var(--input-border-width)%20solid%20var(--background-modifier-border);%5Cn%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20font-family:%20inherit;%5Cn%20%20%20%20padding:%20var(--size-4-1)%20var(--size-4-2);%5Cn%20%20%20%20font-size:%20var(--font-ui-small);%5Cn%20%20%20%20border-radius:%20var(--input-radius);%5Cn%20%20%20%20outline:%20none;%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20@media%20(hover:%20hover)%20%7B%5Cn%20%20%20%20%20%20%20%20&:hover%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20border-color:%20var(--background-modifier-border-hover);%5Cn%20%20%20%20%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20border-color:%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%20%20transition:%5Cn%20%20%20%20%20%20%20%20box-shadow%200.15s%20ease-in-out,%5Cn%20%20%20%20%20%20%20%20border%200.15s%20ease-in-out;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&:active,%5Cn%20%20%20%20&:focus,%5Cn%20%20%20%20&:focus-visible%20%7B%5Cn%20%20%20%20%20%20box-shadow:%200%200%200%202px%20var(--background-modifier-border-focus);%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::placeholder%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20@at-root%20%7B%5Cn%20%20%20%20.mod-rtl,%5Cn%20%20%20%20.is-rtl,%5Cn%20%20%20%20.rtl%20%7B%5Cn%20%20%20%20%20%20&%20%7B%5Cn%20%20%20%20%20%20%20%20input%5Btype='month'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='time'%5D,%5Cn%20%20%20%20%20%20%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%20%20%7D%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20input%5Btype='month'%5D,%5Cn%20%20input%5Btype='time'%5D,%5Cn%20%20input%5Btype='week'%5D%20%7B%5Cn%20%20%20%20font-variant-numeric:%20tabular-nums;%5Cn%20%20%20%20position:%20relative;%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-text%20%7B%5Cn%20%20%20%20%20%20color:%20var(--text-faint);%5Cn%20%20%20%20%20%20padding-inline-end:%200;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20position:%20absolute;%5Cn%20%20%20%20%20%20left:%20var(--size-4-1);%5Cn%20%20%20%20%20%20right:%20auto;%5Cn%20%20%20%20%20%20opacity:%200.5;%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20&::-webkit-datetime-edit-month-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-day-field,%5Cn%20%20%20%20&::-webkit-datetime-edit-year-field%20%7B%5Cn%20%20%20%20%20%20&:active,%5Cn%20%20%20%20%20%20&:focus%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20var(--text-selection);%5Cn%20%20%20%20%20%20%20%20color:%20var(--text-normal);%5Cn%20%20%20%20%20%20%20%20cursor:%20text;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20.mod-rtl%20&,%5Cn%20%20%20%20%20%20.is-rtl%20&,%5Cn%20%20%20%20%20%20.rtl%20&%20%7B%5Cn%20%20%20%20%20%20direction:%20rtl;%5Cn%5Cn%20%20%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20%20%20left:%20auto;%5Cn%20%20%20%20%20%20%20%20right:%20var(--size-4-1);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%5Cn%20%20%20%20@at-root%20%7B%5Cn%20%20%20%20%20%20body:not(.is-ios):not(.is-android)%20&%20%7B%5Cn%20%20%20%20%20%20%20%20padding-inline-start:%20var(--size-4-6);%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%5Cn%20%20input%5Btype='time'%5D%20%7B%5Cn%20%20%20%20&::-webkit-calendar-picker-indicator%20%7B%5Cn%20%20%20%20%20%20margin-inline-start:%200;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.modal-container%20%7B%5Cn%20%20%20%20.ok-button%20%7B%5Cn%20%20%20%20%20%20margin-right:%2010px;%5Cn%20%20%20%20%20%20margin-top:%2020px;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20.multiple-dropdown-component%20%7B%5Cn%20%20%20%20select,%5Cn%20%20%20%20select:focus,%5Cn%20%20%20%20.dropdown%20%7B%5Cn%20%20%20%20%20%20height:%20auto;%5Cn%5Cn%20%20%20%20%20%20option:checked%20%7B%5Cn%20%20%20%20%20%20%20%20background-color:%20%231967d2;%5Cn%20%20%20%20%20%20%20%20color:%20%23fff;%5Cn%20%20%20%20%20%20%7D%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20&.prompt-modal%20%7B%5Cn%20%20%20%20.text-box%20%7B%5Cn%20%20%20%20%20%20width:%20100%25;%5Cn%20%20%20%20%7D%5Cn%20%20%7D%5Cn%7D%5Cn%22,%22.obsidian-dev-utils%20%7B%5Cn%20%20:invalid%20%7B%5Cn%20%20%20%20box-shadow:%200%200%200%202px%20var(--text-error);%5Cn%20%20%7D%5Cn%7D%5Cn%22%5D%7D */\n";
|
11
11
|
export {
|
@@ -20,6 +20,12 @@ export declare enum FunctionHandlingMode {
|
|
20
20
|
*/
|
21
21
|
NameOnly = "nameOnly"
|
22
22
|
}
|
23
|
+
/**
|
24
|
+
* A type that represents the keys of an object as strings.
|
25
|
+
*
|
26
|
+
* @typeParam T - The type of the object.
|
27
|
+
*/
|
28
|
+
export type StringKeys<T extends object> = Extract<keyof T, string>;
|
23
29
|
/**
|
24
30
|
* Options for converting an object to JSON.
|
25
31
|
*/
|
package/dist/lib/esm/Object.mjs
CHANGED
@@ -283,4 +283,4 @@ export {
|
|
283
283
|
setNestedPropertyValue,
|
284
284
|
toJson
|
285
285
|
};
|
286
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Object.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Object\n * Contains utility functions for Objects.\n */\n\nimport type { UndefinedOnPartialDeep } from 'type-fest';\n\nimport {\n  errorToString,\n  throwExpression\n} from './Error.ts';\nimport { replaceAll } from './String.ts';\n\n/**\n * Specifies how functions should be handled in the JSON output.\n */\nexport enum FunctionHandlingMode {\n  /**\n   * Excludes functions from the JSON output.\n   */\n  Exclude = 'exclude',\n  /**\n   * Includes the full function definition in the JSON output.\n   */\n  Full = 'full',\n  /**\n   * Includes only the function name in the JSON output.\n   */\n  NameOnly = 'nameOnly'\n}\n\nenum TokenSubstitutionKey {\n  CircularReference = 'CircularReference',\n  Function = 'Function',\n  MaxDepthLimitReached = 'MaxDepthLimitReached',\n  MaxDepthLimitReachedArray = 'MaxDepthLimitReachedArray',\n  ToJSONFailed = 'ToJSONFailed',\n  Undefined = 'Undefined'\n}\n\n/**\n * Options for converting an object to JSON.\n */\nexport interface ToJsonOptions {\n  /**\n   * Specifies how functions should be handled in the JSON output (default: `exclude`).\n   */\n  functionHandlingMode: FunctionHandlingMode;\n  /**\n   * Specifies the maximum depth of nested objects to include in the JSON output.\n   * Use `-1` for no limit.\n   * Defaults to `-1`.\n   */\n  maxDepth: number;\n  /**\n   * Specifies whether to catch errors in `toJSON()` and replace them with a placeholder.\n   * Defaults to `false`.\n   */\n  shouldCatchToJSONErrors: boolean;\n  /**\n   * Specifies whether to handle circular references in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleCircularReferences: boolean;\n  /**\n   * Specifies whether to handle errors in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleErrors: boolean;\n  /**\n   * Specifies whether to handle undefined values in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleUndefined: boolean;\n  /**\n   * Specifies whether to sort the keys of the JSON output.\n   * Defaults to `false`.\n   */\n  shouldSortKeys: boolean;\n  /**\n   * Specifies the indentation of the JSON output. This can be a number of spaces or a string. Defaults to `2`.\n   */\n  space: number | string;\n  /**\n   * Specifies the substitutions to use in the JSON output.\n   */\n  tokenSubstitutions: Partial<TokenSubstitutions>;\n}\n\ninterface ApplySubstitutionsOptions {\n  functionTexts: string[];\n  index: number;\n  key: TokenSubstitutionKey;\n  substitutions: TokenSubstitutions;\n}\n\ninterface JSONSerializable {\n  toJSON(...args: unknown[]): unknown;\n}\n\ninterface TokenSubstitutions {\n  circularReference: string;\n  maxDepthLimitReached: string;\n  toJSONFailed: string;\n}\n\nconst KEY_SEPARATOR = '.';\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param source - The source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U>(target: T, source: U): T & U;\n\n/**\n * @param target - The target object to assign properties to.\n * @param source1 - The first source object to assign properties from.\n * @param source2 - The second source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U, V>(target: T, source1: U, source2: V): T & U & V;\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param source1 - The first source object to assign properties from.\n * @param source2 - The second source object to assign properties from.\n * @param source3 - The third source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U, V, W>(target: T, source1: U, source2: V, source3: W): T & U & V & W;\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param sources - The source objects to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties(target: object, ...sources: object[]): object {\n  return _assignWithNonEnumerableProperties(target, ...sources);\n}\n\n/**\n * Clones an object, including non-enumerable properties.\n *\n * @param obj - The object to clone.\n * @returns A new object with the same properties as the original object, including non-enumerable properties.\n */\nexport function cloneWithNonEnumerableProperties<T extends object>(obj: T): T {\n  return Object.create(getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj)) as T;\n}\n\n/**\n * Compares two values to determine if they are deeply equal.\n *\n * @param a - The first value to compare.\n * @param b - The second value to compare.\n * @returns `true` if the values are deeply equal, otherwise `false`.\n */\nexport function deepEqual(a: unknown, b: unknown): boolean {\n  if (a === b) {\n    return true;\n  }\n\n  if (typeof a !== 'object' || typeof b !== 'object' || a === null || b === null) {\n    return false;\n  }\n\n  const keysA = Object.keys(a);\n  const keysB = Object.keys(b);\n\n  if (keysA.length !== keysB.length) {\n    return false;\n  }\n\n  const aRecord = a as Record<string, unknown>;\n  const bRecord = b as Record<string, unknown>;\n\n  for (const key of keysA) {\n    if (!keysB.includes(key) || !deepEqual(aRecord[key], bRecord[key])) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\n/**\n * Deletes multiple properties from an object.\n *\n * @typeParam T - The type of the object.\n * @param obj - The object to delete the properties from.\n * @param propertyNames - The names of the properties to delete.\n * @returns `true` if any of the properties were present, otherwise `false`.\n */\nexport function deleteProperties<T extends object>(obj: T, propertyNames: (keyof T)[]): boolean {\n  let ans = false;\n\n  for (const propertyName of propertyNames) {\n    ans = deleteProperty(obj, propertyName) || ans;\n  }\n\n  return ans;\n}\n\n/**\n * Deletes a property from an object.\n *\n * @typeParam T - The type of the object.\n * @param obj - The object to delete the property from.\n * @param propertyName - The name of the property to delete.\n * @returns `true` if the property was present, otherwise `false`.\n */\nexport function deleteProperty<T extends object>(obj: T, propertyName: keyof T): boolean {\n  if (!Object.hasOwn(obj, propertyName)) {\n    return false;\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n  delete obj[propertyName];\n  return true;\n}\n\n/**\n * Gets the value of a nested property from an object.\n *\n * @param obj - The object to get the nested property value from.\n * @param path - The path to the nested property.\n * @returns The value of the nested property.\n */\nexport function getNestedPropertyValue(obj: Record<string, unknown>, path: string): unknown {\n  let node: Record<string, unknown> | undefined = obj;\n  const keys = path.split(KEY_SEPARATOR);\n  for (const key of keys) {\n    if (node === undefined) {\n      return undefined;\n    }\n    node = node[key] as Record<string, unknown> | undefined;\n  }\n\n  return node;\n}\n\n/**\n * Gets the prototype of the specified object.\n *\n * @typeParam T - The type of the object.\n * @param instance - The object instance to retrieve the prototype of.\n * @returns The prototype of the object.\n */\nexport function getPrototypeOf<T>(instance: T): T {\n  if (instance === undefined || instance === null) {\n    return instance;\n  }\n  return Object.getPrototypeOf(instance) as T;\n}\n\n/**\n * Retrieves the name of a property of a given type `T`.\n *\n * @typeParam T - The type of the object containing the property.\n * @param name - The name of the property as a string.\n * @returns The name of the property.\n */\nexport function nameof<T>(name: Extract<keyof T, string>): string {\n  return name;\n}\n\n/**\n * Normalizes optional properties to allow `undefined` assignment in strict mode.\n *\n * This utility provides a workaround for the `exactOptionalPropertyTypes` TypeScript flag,\n * which prohibits directly assigning `undefined` to optional properties when the type\n * explicitly omits `undefined`.\n *\n * Example:\n * ```typescript\n * // With `exactOptionalPropertyTypes: true`\n * const x: { prop?: string } = { prop: undefined }; // Compiler error\n *\n * // Using this utility:\n * const y: { prop?: string } = normalizeOptionalProperties<{ prop?: string }>({ prop: undefined }); // Works\n * ```\n *\n * @typeParam T - The target type with optional properties to normalize.\n * @param obj - The object to normalize, allowing explicit `undefined` for optional properties.\n * @returns The normalized object, compatible with `exactOptionalPropertyTypes`.\n */\nexport function normalizeOptionalProperties<T>(obj: UndefinedOnPartialDeep<T>): T {\n  return obj as T;\n}\n\n/**\n * Sets the value of a nested property in an object.\n *\n * @param obj - The object to set the nested property value in.\n * @param path - The path to the nested property.\n * @param value - The value to set.\n */\nexport function setNestedPropertyValue(obj: Record<string, unknown>, path: string, value: unknown): void {\n  const error = new Error(`Property path ${path} not found`);\n  let node: Record<string, unknown> | undefined = obj;\n  const keys = path.split(KEY_SEPARATOR);\n  for (const key of keys.slice(0, -1)) {\n    if (node === undefined) {\n      throw error;\n    }\n    node = node[key] as Record<string, unknown> | undefined;\n  }\n\n  const lastKey = keys.at(-1);\n  if (node === undefined || lastKey === undefined) {\n    throw error;\n  }\n\n  node[lastKey] = value;\n}\n\n/**\n * Converts a given value to a JSON string.\n *\n * @param value - The value to be converted to JSON. This can be of any type.\n * @param options - Options for customizing the JSON conversion process.\n * @returns The JSON string representation of the input value.\n */\nexport function toJson(value: unknown, options: Partial<ToJsonOptions> = {}): string {\n  const DEFAULT_OPTIONS: { tokenSubstitutions: TokenSubstitutions } & ToJsonOptions = {\n    functionHandlingMode: FunctionHandlingMode.Exclude,\n    maxDepth: -1,\n    shouldCatchToJSONErrors: false,\n    shouldHandleCircularReferences: false,\n    shouldHandleErrors: false,\n    shouldHandleUndefined: false,\n    shouldSortKeys: false,\n    // eslint-disable-next-line no-magic-numbers\n    space: 2,\n    tokenSubstitutions: {\n      circularReference: makeObjectTokenSubstitution(TokenSubstitutionKey.CircularReference),\n      maxDepthLimitReached: makeObjectTokenSubstitution(TokenSubstitutionKey.MaxDepthLimitReached),\n      toJSONFailed: makeObjectTokenSubstitution(TokenSubstitutionKey.ToJSONFailed)\n    }\n  };\n\n  const fullOptions = {\n    ...DEFAULT_OPTIONS,\n    ...options,\n    tokenSubstitutions: {\n      ...DEFAULT_OPTIONS.tokenSubstitutions,\n      ...options.tokenSubstitutions\n    }\n  };\n\n  if (fullOptions.maxDepth === -1) {\n    fullOptions.maxDepth = Infinity;\n  }\n\n  const functionTexts: string[] = [];\n  const usedObjects = new WeakSet<object>();\n\n  const plainObject = toPlainObject(value, '', 0, true, fullOptions, functionTexts, usedObjects);\n  let json = JSON.stringify(plainObject, null, fullOptions.space) ?? '';\n  json = replaceAll(json, /\"\\[\\[(?<Key>[A-Za-z]+)(?<Index>\\d*)\\]\\]\"/g, (_, key, indexStr) =>\n    applySubstitutions({\n      functionTexts,\n      index: indexStr ? parseInt(indexStr, 10) : 0,\n      key: key as TokenSubstitutionKey,\n      substitutions: fullOptions.tokenSubstitutions\n    }));\n  return json;\n}\n\nfunction _assignWithNonEnumerableProperties(target: object, ...sources: object[]): object {\n  for (const source of sources) {\n    const descriptors = Object.getOwnPropertyDescriptors(source);\n\n    for (const [key, descriptor] of Object.entries(descriptors)) {\n      try {\n        // Avoid redefining read-only properties (especially `prototype`)\n        if (\n          key === 'prototype'\n          || (Object.getOwnPropertyDescriptor(target, key)?.writable === false\n            && !Object.getOwnPropertyDescriptor(target, key)?.configurable)\n        ) {\n          continue;\n        }\n\n        Object.defineProperty(target, key, descriptor);\n      } catch {\n        // Silently ignore if defineProperty fails\n      }\n    }\n  }\n\n  const sourcePrototypes = sources\n    .map((source) => getPrototypeOf<object | undefined>(source))\n    .filter((proto): proto is object => !!proto);\n\n  if (sourcePrototypes.length > 0) {\n    const targetPrototype = _assignWithNonEnumerableProperties({}, getPrototypeOf(target), ...sourcePrototypes);\n\n    try {\n      Object.setPrototypeOf(target, targetPrototype);\n    } catch {\n      // Silently ignore if setPrototypeOf fails\n    }\n  }\n\n  return target;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-invalid-void-type\nfunction applySubstitutions(options: ApplySubstitutionsOptions): string | void {\n  switch (options.key) {\n    case TokenSubstitutionKey.CircularReference:\n      return options.substitutions.circularReference;\n    case TokenSubstitutionKey.Function:\n      return options.functionTexts[options.index] ?? throwExpression(new Error(`Function with index ${options.index.toString()} not found`));\n    case TokenSubstitutionKey.MaxDepthLimitReached:\n      return options.substitutions.maxDepthLimitReached;\n    case TokenSubstitutionKey.MaxDepthLimitReachedArray:\n      return `Array(${options.index.toString()})`;\n    case TokenSubstitutionKey.ToJSONFailed:\n      return options.substitutions.toJSONFailed;\n    case TokenSubstitutionKey.Undefined:\n      return 'undefined';\n    default:\n      break;\n  }\n}\n\nfunction handleArray(\n  value: unknown[],\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (depth > fullOptions.maxDepth) {\n    return makePlaceholder(TokenSubstitutionKey.MaxDepthLimitReachedArray, value.length);\n  }\n\n  return value.map((item, index) => toPlainObject(item, index.toString(), depth + 1, canUseToJSON, fullOptions, functionTexts, usedObjects));\n}\n\nfunction handleCircularReference(value: object, key: string, fullOptions: ToJsonOptions): unknown {\n  if (fullOptions.shouldHandleCircularReferences) {\n    return makePlaceholder(TokenSubstitutionKey.CircularReference);\n  }\n  const valueConstructorName = value.constructor.name || 'Object';\n  throw new TypeError(`Converting circular structure to JSON\n--> starting at object with constructor '${valueConstructorName}'\n--- property '${key}' closes the circle`);\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type\nfunction handleFunction(value: Function, functionTexts: string[], fullOptions: ToJsonOptions): unknown {\n  if (fullOptions.functionHandlingMode === FunctionHandlingMode.Exclude) {\n    return undefined;\n  }\n  const index = functionTexts.length;\n  const functionText = fullOptions.functionHandlingMode === FunctionHandlingMode.Full\n    ? value.toString()\n    : `function ${value.name || 'anonymous'}() { /* ... */ }`;\n  functionTexts.push(functionText);\n  return makePlaceholder(TokenSubstitutionKey.Function, index);\n}\n\nfunction handleObject(\n  value: object,\n  key: string,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (usedObjects.has(value)) {\n    return handleCircularReference(value, key, fullOptions);\n  }\n\n  usedObjects.add(value);\n\n  if (canUseToJSON) {\n    const toJSONResult = tryHandleToJSON(value, key, depth, fullOptions, functionTexts, usedObjects);\n    if (toJSONResult !== undefined) {\n      return toJSONResult;\n    }\n  }\n\n  if (Array.isArray(value)) {\n    return handleArray(value, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n  }\n\n  if (depth > fullOptions.maxDepth) {\n    return makePlaceholder(TokenSubstitutionKey.MaxDepthLimitReached);\n  }\n\n  if (value instanceof Error && fullOptions.shouldHandleErrors) {\n    return errorToString(value);\n  }\n\n  return handlePlainObject(value, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n}\n\nfunction handlePlainObject(\n  value: object,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  const entries = Object.entries(value);\n  if (fullOptions.shouldSortKeys) {\n    entries.sort(([key1], [key2]) => key1.localeCompare(key2));\n  }\n\n  return Object.fromEntries(\n    entries.map(([key2, value2]) => [\n      key2,\n      toPlainObject(value2, key2, depth + 1, canUseToJSON, fullOptions, functionTexts, usedObjects)\n    ])\n  );\n}\n\nfunction makeObjectTokenSubstitution(key: TokenSubstitutionKey): string {\n  return `{ \"[[${key}]]\": null }`;\n}\n\nfunction makePlaceholder(key: TokenSubstitutionKey, index?: number): string {\n  return `[[${key}${index?.toString() ?? ''}]]`;\n}\n\nfunction toPlainObject(\n  value: unknown,\n  key: string,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (value === undefined) {\n    return (depth === 0 || fullOptions.shouldHandleUndefined)\n      ? makePlaceholder(TokenSubstitutionKey.Undefined)\n      : undefined;\n  }\n\n  if (typeof value === 'function') {\n    return handleFunction(value, functionTexts, fullOptions);\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return value;\n  }\n\n  return handleObject(value, key, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n}\n\nfunction tryHandleToJSON(\n  value: object,\n  key: string,\n  depth: number,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  const toJSON = (value as Partial<JSONSerializable>).toJSON;\n  if (typeof toJSON === 'function') {\n    try {\n      const newValue = toJSON.call(value, key);\n      return toPlainObject(newValue, key, depth, false, fullOptions, functionTexts, usedObjects);\n    } catch (e) {\n      if (fullOptions.shouldCatchToJSONErrors) {\n        return makePlaceholder(TokenSubstitutionKey.ToJSONFailed);\n      }\n      throw e;\n    }\n  }\n  return undefined;\n}\n"],
  "mappings": "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",
  "names": ["FunctionHandlingMode"]
}

|
286
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/Object.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation Object\n * Contains utility functions for Objects.\n */\n\nimport type { UndefinedOnPartialDeep } from 'type-fest';\n\nimport {\n  errorToString,\n  throwExpression\n} from './Error.ts';\nimport { replaceAll } from './String.ts';\n\n/**\n * Specifies how functions should be handled in the JSON output.\n */\nexport enum FunctionHandlingMode {\n  /**\n   * Excludes functions from the JSON output.\n   */\n  Exclude = 'exclude',\n  /**\n   * Includes the full function definition in the JSON output.\n   */\n  Full = 'full',\n  /**\n   * Includes only the function name in the JSON output.\n   */\n  NameOnly = 'nameOnly'\n}\n\nenum TokenSubstitutionKey {\n  CircularReference = 'CircularReference',\n  Function = 'Function',\n  MaxDepthLimitReached = 'MaxDepthLimitReached',\n  MaxDepthLimitReachedArray = 'MaxDepthLimitReachedArray',\n  ToJSONFailed = 'ToJSONFailed',\n  Undefined = 'Undefined'\n}\n\n/**\n * A type that represents the keys of an object as strings.\n *\n * @typeParam T - The type of the object.\n */\nexport type StringKeys<T extends object> = Extract<keyof T, string>;\n\n/**\n * Options for converting an object to JSON.\n */\nexport interface ToJsonOptions {\n  /**\n   * Specifies how functions should be handled in the JSON output (default: `exclude`).\n   */\n  functionHandlingMode: FunctionHandlingMode;\n  /**\n   * Specifies the maximum depth of nested objects to include in the JSON output.\n   * Use `-1` for no limit.\n   * Defaults to `-1`.\n   */\n  maxDepth: number;\n  /**\n   * Specifies whether to catch errors in `toJSON()` and replace them with a placeholder.\n   * Defaults to `false`.\n   */\n  shouldCatchToJSONErrors: boolean;\n  /**\n   * Specifies whether to handle circular references in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleCircularReferences: boolean;\n  /**\n   * Specifies whether to handle errors in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleErrors: boolean;\n  /**\n   * Specifies whether to handle undefined values in the JSON output.\n   * Defaults to `false`.\n   */\n  shouldHandleUndefined: boolean;\n  /**\n   * Specifies whether to sort the keys of the JSON output.\n   * Defaults to `false`.\n   */\n  shouldSortKeys: boolean;\n  /**\n   * Specifies the indentation of the JSON output. This can be a number of spaces or a string. Defaults to `2`.\n   */\n  space: number | string;\n  /**\n   * Specifies the substitutions to use in the JSON output.\n   */\n  tokenSubstitutions: Partial<TokenSubstitutions>;\n}\n\ninterface ApplySubstitutionsOptions {\n  functionTexts: string[];\n  index: number;\n  key: TokenSubstitutionKey;\n  substitutions: TokenSubstitutions;\n}\n\ninterface JSONSerializable {\n  toJSON(...args: unknown[]): unknown;\n}\n\ninterface TokenSubstitutions {\n  circularReference: string;\n  maxDepthLimitReached: string;\n  toJSONFailed: string;\n}\n\nconst KEY_SEPARATOR = '.';\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param source - The source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U>(target: T, source: U): T & U;\n\n/**\n * @param target - The target object to assign properties to.\n * @param source1 - The first source object to assign properties from.\n * @param source2 - The second source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U, V>(target: T, source1: U, source2: V): T & U & V;\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param source1 - The first source object to assign properties from.\n * @param source2 - The second source object to assign properties from.\n * @param source3 - The third source object to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties<T extends object, U, V, W>(target: T, source1: U, source2: V, source3: W): T & U & V & W;\n\n/**\n * Assigns properties from one or more source objects to a target object, including non-enumerable properties.\n *\n * @param target - The target object to assign properties to.\n * @param sources - The source objects to assign properties from.\n * @returns The target object with the assigned properties.\n */\nexport function assignWithNonEnumerableProperties(target: object, ...sources: object[]): object {\n  return _assignWithNonEnumerableProperties(target, ...sources);\n}\n\n/**\n * Clones an object, including non-enumerable properties.\n *\n * @param obj - The object to clone.\n * @returns A new object with the same properties as the original object, including non-enumerable properties.\n */\nexport function cloneWithNonEnumerableProperties<T extends object>(obj: T): T {\n  return Object.create(getPrototypeOf(obj), Object.getOwnPropertyDescriptors(obj)) as T;\n}\n\n/**\n * Compares two values to determine if they are deeply equal.\n *\n * @param a - The first value to compare.\n * @param b - The second value to compare.\n * @returns `true` if the values are deeply equal, otherwise `false`.\n */\nexport function deepEqual(a: unknown, b: unknown): boolean {\n  if (a === b) {\n    return true;\n  }\n\n  if (typeof a !== 'object' || typeof b !== 'object' || a === null || b === null) {\n    return false;\n  }\n\n  const keysA = Object.keys(a);\n  const keysB = Object.keys(b);\n\n  if (keysA.length !== keysB.length) {\n    return false;\n  }\n\n  const aRecord = a as Record<string, unknown>;\n  const bRecord = b as Record<string, unknown>;\n\n  for (const key of keysA) {\n    if (!keysB.includes(key) || !deepEqual(aRecord[key], bRecord[key])) {\n      return false;\n    }\n  }\n\n  return true;\n}\n\n/**\n * Deletes multiple properties from an object.\n *\n * @typeParam T - The type of the object.\n * @param obj - The object to delete the properties from.\n * @param propertyNames - The names of the properties to delete.\n * @returns `true` if any of the properties were present, otherwise `false`.\n */\nexport function deleteProperties<T extends object>(obj: T, propertyNames: (keyof T)[]): boolean {\n  let ans = false;\n\n  for (const propertyName of propertyNames) {\n    ans = deleteProperty(obj, propertyName) || ans;\n  }\n\n  return ans;\n}\n\n/**\n * Deletes a property from an object.\n *\n * @typeParam T - The type of the object.\n * @param obj - The object to delete the property from.\n * @param propertyName - The name of the property to delete.\n * @returns `true` if the property was present, otherwise `false`.\n */\nexport function deleteProperty<T extends object>(obj: T, propertyName: keyof T): boolean {\n  if (!Object.hasOwn(obj, propertyName)) {\n    return false;\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n  delete obj[propertyName];\n  return true;\n}\n\n/**\n * Gets the value of a nested property from an object.\n *\n * @param obj - The object to get the nested property value from.\n * @param path - The path to the nested property.\n * @returns The value of the nested property.\n */\nexport function getNestedPropertyValue(obj: Record<string, unknown>, path: string): unknown {\n  let node: Record<string, unknown> | undefined = obj;\n  const keys = path.split(KEY_SEPARATOR);\n  for (const key of keys) {\n    if (node === undefined) {\n      return undefined;\n    }\n    node = node[key] as Record<string, unknown> | undefined;\n  }\n\n  return node;\n}\n\n/**\n * Gets the prototype of the specified object.\n *\n * @typeParam T - The type of the object.\n * @param instance - The object instance to retrieve the prototype of.\n * @returns The prototype of the object.\n */\nexport function getPrototypeOf<T>(instance: T): T {\n  if (instance === undefined || instance === null) {\n    return instance;\n  }\n  return Object.getPrototypeOf(instance) as T;\n}\n\n/**\n * Retrieves the name of a property of a given type `T`.\n *\n * @typeParam T - The type of the object containing the property.\n * @param name - The name of the property as a string.\n * @returns The name of the property.\n */\nexport function nameof<T>(name: Extract<keyof T, string>): string {\n  return name;\n}\n\n/**\n * Normalizes optional properties to allow `undefined` assignment in strict mode.\n *\n * This utility provides a workaround for the `exactOptionalPropertyTypes` TypeScript flag,\n * which prohibits directly assigning `undefined` to optional properties when the type\n * explicitly omits `undefined`.\n *\n * Example:\n * ```typescript\n * // With `exactOptionalPropertyTypes: true`\n * const x: { prop?: string } = { prop: undefined }; // Compiler error\n *\n * // Using this utility:\n * const y: { prop?: string } = normalizeOptionalProperties<{ prop?: string }>({ prop: undefined }); // Works\n * ```\n *\n * @typeParam T - The target type with optional properties to normalize.\n * @param obj - The object to normalize, allowing explicit `undefined` for optional properties.\n * @returns The normalized object, compatible with `exactOptionalPropertyTypes`.\n */\nexport function normalizeOptionalProperties<T>(obj: UndefinedOnPartialDeep<T>): T {\n  return obj as T;\n}\n\n/**\n * Sets the value of a nested property in an object.\n *\n * @param obj - The object to set the nested property value in.\n * @param path - The path to the nested property.\n * @param value - The value to set.\n */\nexport function setNestedPropertyValue(obj: Record<string, unknown>, path: string, value: unknown): void {\n  const error = new Error(`Property path ${path} not found`);\n  let node: Record<string, unknown> | undefined = obj;\n  const keys = path.split(KEY_SEPARATOR);\n  for (const key of keys.slice(0, -1)) {\n    if (node === undefined) {\n      throw error;\n    }\n    node = node[key] as Record<string, unknown> | undefined;\n  }\n\n  const lastKey = keys.at(-1);\n  if (node === undefined || lastKey === undefined) {\n    throw error;\n  }\n\n  node[lastKey] = value;\n}\n\n/**\n * Converts a given value to a JSON string.\n *\n * @param value - The value to be converted to JSON. This can be of any type.\n * @param options - Options for customizing the JSON conversion process.\n * @returns The JSON string representation of the input value.\n */\nexport function toJson(value: unknown, options: Partial<ToJsonOptions> = {}): string {\n  const DEFAULT_OPTIONS: { tokenSubstitutions: TokenSubstitutions } & ToJsonOptions = {\n    functionHandlingMode: FunctionHandlingMode.Exclude,\n    maxDepth: -1,\n    shouldCatchToJSONErrors: false,\n    shouldHandleCircularReferences: false,\n    shouldHandleErrors: false,\n    shouldHandleUndefined: false,\n    shouldSortKeys: false,\n    // eslint-disable-next-line no-magic-numbers\n    space: 2,\n    tokenSubstitutions: {\n      circularReference: makeObjectTokenSubstitution(TokenSubstitutionKey.CircularReference),\n      maxDepthLimitReached: makeObjectTokenSubstitution(TokenSubstitutionKey.MaxDepthLimitReached),\n      toJSONFailed: makeObjectTokenSubstitution(TokenSubstitutionKey.ToJSONFailed)\n    }\n  };\n\n  const fullOptions = {\n    ...DEFAULT_OPTIONS,\n    ...options,\n    tokenSubstitutions: {\n      ...DEFAULT_OPTIONS.tokenSubstitutions,\n      ...options.tokenSubstitutions\n    }\n  };\n\n  if (fullOptions.maxDepth === -1) {\n    fullOptions.maxDepth = Infinity;\n  }\n\n  const functionTexts: string[] = [];\n  const usedObjects = new WeakSet<object>();\n\n  const plainObject = toPlainObject(value, '', 0, true, fullOptions, functionTexts, usedObjects);\n  let json = JSON.stringify(plainObject, null, fullOptions.space) ?? '';\n  json = replaceAll(json, /\"\\[\\[(?<Key>[A-Za-z]+)(?<Index>\\d*)\\]\\]\"/g, (_, key, indexStr) =>\n    applySubstitutions({\n      functionTexts,\n      index: indexStr ? parseInt(indexStr, 10) : 0,\n      key: key as TokenSubstitutionKey,\n      substitutions: fullOptions.tokenSubstitutions\n    }));\n  return json;\n}\n\nfunction _assignWithNonEnumerableProperties(target: object, ...sources: object[]): object {\n  for (const source of sources) {\n    const descriptors = Object.getOwnPropertyDescriptors(source);\n\n    for (const [key, descriptor] of Object.entries(descriptors)) {\n      try {\n        // Avoid redefining read-only properties (especially `prototype`)\n        if (\n          key === 'prototype'\n          || (Object.getOwnPropertyDescriptor(target, key)?.writable === false\n            && !Object.getOwnPropertyDescriptor(target, key)?.configurable)\n        ) {\n          continue;\n        }\n\n        Object.defineProperty(target, key, descriptor);\n      } catch {\n        // Silently ignore if defineProperty fails\n      }\n    }\n  }\n\n  const sourcePrototypes = sources\n    .map((source) => getPrototypeOf<object | undefined>(source))\n    .filter((proto): proto is object => !!proto);\n\n  if (sourcePrototypes.length > 0) {\n    const targetPrototype = _assignWithNonEnumerableProperties({}, getPrototypeOf(target), ...sourcePrototypes);\n\n    try {\n      Object.setPrototypeOf(target, targetPrototype);\n    } catch {\n      // Silently ignore if setPrototypeOf fails\n    }\n  }\n\n  return target;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-invalid-void-type\nfunction applySubstitutions(options: ApplySubstitutionsOptions): string | void {\n  switch (options.key) {\n    case TokenSubstitutionKey.CircularReference:\n      return options.substitutions.circularReference;\n    case TokenSubstitutionKey.Function:\n      return options.functionTexts[options.index] ?? throwExpression(new Error(`Function with index ${options.index.toString()} not found`));\n    case TokenSubstitutionKey.MaxDepthLimitReached:\n      return options.substitutions.maxDepthLimitReached;\n    case TokenSubstitutionKey.MaxDepthLimitReachedArray:\n      return `Array(${options.index.toString()})`;\n    case TokenSubstitutionKey.ToJSONFailed:\n      return options.substitutions.toJSONFailed;\n    case TokenSubstitutionKey.Undefined:\n      return 'undefined';\n    default:\n      break;\n  }\n}\n\nfunction handleArray(\n  value: unknown[],\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (depth > fullOptions.maxDepth) {\n    return makePlaceholder(TokenSubstitutionKey.MaxDepthLimitReachedArray, value.length);\n  }\n\n  return value.map((item, index) => toPlainObject(item, index.toString(), depth + 1, canUseToJSON, fullOptions, functionTexts, usedObjects));\n}\n\nfunction handleCircularReference(value: object, key: string, fullOptions: ToJsonOptions): unknown {\n  if (fullOptions.shouldHandleCircularReferences) {\n    return makePlaceholder(TokenSubstitutionKey.CircularReference);\n  }\n  const valueConstructorName = value.constructor.name || 'Object';\n  throw new TypeError(`Converting circular structure to JSON\n--> starting at object with constructor '${valueConstructorName}'\n--- property '${key}' closes the circle`);\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unsafe-function-type\nfunction handleFunction(value: Function, functionTexts: string[], fullOptions: ToJsonOptions): unknown {\n  if (fullOptions.functionHandlingMode === FunctionHandlingMode.Exclude) {\n    return undefined;\n  }\n  const index = functionTexts.length;\n  const functionText = fullOptions.functionHandlingMode === FunctionHandlingMode.Full\n    ? value.toString()\n    : `function ${value.name || 'anonymous'}() { /* ... */ }`;\n  functionTexts.push(functionText);\n  return makePlaceholder(TokenSubstitutionKey.Function, index);\n}\n\nfunction handleObject(\n  value: object,\n  key: string,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (usedObjects.has(value)) {\n    return handleCircularReference(value, key, fullOptions);\n  }\n\n  usedObjects.add(value);\n\n  if (canUseToJSON) {\n    const toJSONResult = tryHandleToJSON(value, key, depth, fullOptions, functionTexts, usedObjects);\n    if (toJSONResult !== undefined) {\n      return toJSONResult;\n    }\n  }\n\n  if (Array.isArray(value)) {\n    return handleArray(value, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n  }\n\n  if (depth > fullOptions.maxDepth) {\n    return makePlaceholder(TokenSubstitutionKey.MaxDepthLimitReached);\n  }\n\n  if (value instanceof Error && fullOptions.shouldHandleErrors) {\n    return errorToString(value);\n  }\n\n  return handlePlainObject(value, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n}\n\nfunction handlePlainObject(\n  value: object,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  const entries = Object.entries(value);\n  if (fullOptions.shouldSortKeys) {\n    entries.sort(([key1], [key2]) => key1.localeCompare(key2));\n  }\n\n  return Object.fromEntries(\n    entries.map(([key2, value2]) => [\n      key2,\n      toPlainObject(value2, key2, depth + 1, canUseToJSON, fullOptions, functionTexts, usedObjects)\n    ])\n  );\n}\n\nfunction makeObjectTokenSubstitution(key: TokenSubstitutionKey): string {\n  return `{ \"[[${key}]]\": null }`;\n}\n\nfunction makePlaceholder(key: TokenSubstitutionKey, index?: number): string {\n  return `[[${key}${index?.toString() ?? ''}]]`;\n}\n\nfunction toPlainObject(\n  value: unknown,\n  key: string,\n  depth: number,\n  canUseToJSON: boolean,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  if (value === undefined) {\n    return (depth === 0 || fullOptions.shouldHandleUndefined)\n      ? makePlaceholder(TokenSubstitutionKey.Undefined)\n      : undefined;\n  }\n\n  if (typeof value === 'function') {\n    return handleFunction(value, functionTexts, fullOptions);\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return value;\n  }\n\n  return handleObject(value, key, depth, canUseToJSON, fullOptions, functionTexts, usedObjects);\n}\n\nfunction tryHandleToJSON(\n  value: object,\n  key: string,\n  depth: number,\n  fullOptions: ToJsonOptions,\n  functionTexts: string[],\n  usedObjects: WeakSet<object>\n): unknown {\n  const toJSON = (value as Partial<JSONSerializable>).toJSON;\n  if (typeof toJSON === 'function') {\n    try {\n      const newValue = toJSON.call(value, key);\n      return toPlainObject(newValue, key, depth, false, fullOptions, functionTexts, usedObjects);\n    } catch (e) {\n      if (fullOptions.shouldCatchToJSONErrors) {\n        return makePlaceholder(TokenSubstitutionKey.ToJSONFailed);\n      }\n      throw e;\n    }\n  }\n  return undefined;\n}\n"],
  "mappings": "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",
  "names": ["FunctionHandlingMode"]
}

|