obsidian-dev-utils 49.0.2 → 50.0.0
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 +11 -0
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/ScriptUtils/ESLint/eslint.config.cjs +2 -2
- package/dist/lib/cjs/obsidian/FileChange.cjs +1 -1
- package/dist/lib/cjs/obsidian/Markdown.cjs +3 -2
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +7 -7
- package/dist/lib/cjs/obsidian/Vault.cjs +46 -5
- package/dist/lib/cjs/obsidian/Vault.d.cts +24 -0
- package/dist/lib/cjs/obsidian/VaultDelete.cjs +171 -0
- package/dist/lib/cjs/obsidian/VaultDelete.d.cts +18 -0
- package/dist/lib/cjs/obsidian/index.cjs +4 -4
- package/dist/lib/cjs/obsidian/index.d.cts +1 -1
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/ScriptUtils/ESLint/eslint.config.mjs +2 -2
- package/dist/lib/esm/obsidian/FileChange.mjs +1 -1
- package/dist/lib/esm/obsidian/Markdown.mjs +3 -2
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +9 -10
- package/dist/lib/esm/obsidian/Vault.d.mts +24 -0
- package/dist/lib/esm/obsidian/Vault.mjs +42 -4
- package/dist/lib/esm/obsidian/VaultDelete.d.mts +18 -0
- package/dist/lib/esm/obsidian/VaultDelete.mjs +73 -0
- package/dist/lib/esm/obsidian/index.d.mts +1 -1
- package/dist/lib/esm/obsidian/index.mjs +3 -3
- package/obsidian/VaultDelete/package.json +6 -0
- package/package.json +1 -1
- package/dist/lib/cjs/obsidian/VaultEx.cjs +0 -198
- package/dist/lib/cjs/obsidian/VaultEx.d.cts +0 -36
- package/dist/lib/esm/obsidian/VaultEx.d.mts +0 -36
- package/dist/lib/esm/obsidian/VaultEx.mjs +0 -98
- package/obsidian/VaultEx/package.json +0 -6
|
@@ -108,7 +108,7 @@ async function createTempFile(app, path) {
|
|
|
108
108
|
return async () => {
|
|
109
109
|
file = getFile(app, path);
|
|
110
110
|
if (!file.deleted) {
|
|
111
|
-
await app
|
|
111
|
+
await trashSafe(app, file);
|
|
112
112
|
}
|
|
113
113
|
await folderCleanup();
|
|
114
114
|
};
|
|
@@ -125,11 +125,32 @@ async function createTempFolder(app, path) {
|
|
|
125
125
|
return async () => {
|
|
126
126
|
folder = getFolder(app, path);
|
|
127
127
|
if (!folder.deleted) {
|
|
128
|
-
await app
|
|
128
|
+
await trashSafe(app, folder);
|
|
129
129
|
}
|
|
130
130
|
await folderCleanup();
|
|
131
131
|
};
|
|
132
132
|
}
|
|
133
|
+
async function deleteEmptyFolder(app, pathOrFolder) {
|
|
134
|
+
const folder = getFolderOrNull(app, pathOrFolder);
|
|
135
|
+
if (!folder) {
|
|
136
|
+
return;
|
|
137
|
+
}
|
|
138
|
+
if (!await isEmptyFolder(app, folder)) {
|
|
139
|
+
return;
|
|
140
|
+
}
|
|
141
|
+
await trashSafe(app, folder);
|
|
142
|
+
}
|
|
143
|
+
async function deleteEmptyFolderHierarchy(app, pathOrFolder) {
|
|
144
|
+
let folder = getFolderOrNull(app, pathOrFolder);
|
|
145
|
+
while (folder) {
|
|
146
|
+
if (!await isEmptyFolder(app, folder)) {
|
|
147
|
+
return;
|
|
148
|
+
}
|
|
149
|
+
const parent = folder.parent;
|
|
150
|
+
await deleteEmptyFolder(app, folder);
|
|
151
|
+
folder = parent;
|
|
152
|
+
}
|
|
153
|
+
}
|
|
133
154
|
function getAbstractFilePathSafe(app, path, type) {
|
|
134
155
|
const abstractFile = getAbstractFileOrNull(app, path);
|
|
135
156
|
if (abstractFile && getFileSystemType(abstractFile) === type) {
|
|
@@ -357,6 +378,20 @@ async function saveNote(app, pathOrFile) {
|
|
|
357
378
|
}
|
|
358
379
|
}
|
|
359
380
|
}
|
|
381
|
+
async function trashSafe(app, pathOrFile) {
|
|
382
|
+
const file = getAbstractFileOrNull(app, pathOrFile);
|
|
383
|
+
if (!file) {
|
|
384
|
+
return;
|
|
385
|
+
}
|
|
386
|
+
try {
|
|
387
|
+
await app.fileManager.trashFile(file);
|
|
388
|
+
} catch (e) {
|
|
389
|
+
if (await app.vault.exists(file.path)) {
|
|
390
|
+
throw e;
|
|
391
|
+
}
|
|
392
|
+
getLibDebugger("Vault:trashSafe")(`An error occurred while trashing ${file.path}, but the file no longer exists.`, { error: e, path: file.path });
|
|
393
|
+
}
|
|
394
|
+
}
|
|
360
395
|
async function invokeFileActionSafe(app, pathOrFile, fileAction) {
|
|
361
396
|
const path = getPath(app, pathOrFile);
|
|
362
397
|
let file = getFileOrNull(app, path);
|
|
@@ -379,6 +414,8 @@ export {
|
|
|
379
414
|
createFolderSafe,
|
|
380
415
|
createTempFile,
|
|
381
416
|
createTempFolder,
|
|
417
|
+
deleteEmptyFolder,
|
|
418
|
+
deleteEmptyFolderHierarchy,
|
|
382
419
|
getAbstractFilePathSafe,
|
|
383
420
|
getAvailablePath,
|
|
384
421
|
getFilePathSafe,
|
|
@@ -397,6 +434,7 @@ export {
|
|
|
397
434
|
process,
|
|
398
435
|
readSafe,
|
|
399
436
|
renameSafe,
|
|
400
|
-
saveNote
|
|
437
|
+
saveNote,
|
|
438
|
+
trashSafe
|
|
401
439
|
};
|
|
402
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  EventRef,\n  ListedFiles\n} from 'obsidian';\n\nimport {\n  MarkdownView,\n  TAbstractFile,\n  TFile,\n  TFolder\n} from 'obsidian';\nimport {\n  parentFolderPath,\n  ViewType\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrAbstractFile,\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { getLibDebugger } from '../Debug.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 { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  lockEditor,\n  unlockEditor\n} from './Editor.ts';\nimport {\n  asFile,\n  asFolder,\n  FileSystemType,\n  getAbstractFile,\n  getAbstractFileOrNull,\n  getFile,\n  getFileOrNull,\n  getFileSystemType,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport { t } from './i18n/i18n.ts';\n\n/**\n * Options for {@link process}.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * Whether to fail if the file is missing or deleted. Default is `true`.\n   */\n  readonly shouldFailOnMissingFile?: boolean;\n\n  /**\n   * Whether to lock the editor while processing the file. Applicable only for markdown files. Default is `true`.\n   */\n  readonly shouldLockEditorWhileProcessing?: boolean;\n\n  /**\n   * Whether to show a timeout notice. Default is `true`.\n   */\n  readonly shouldShowTimeoutNotice?: 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 {@link 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  if (file.path === newPath) {\n    return newPath;\n  }\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 {@link 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 *\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A {@link 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 *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link 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 folderPath = parentFolderPath(path);\n  await createTempFolder(app, folderPath);\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 a safe path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @param type - The type of the file system object.\n * @returns The safe path for the file or folder.\n */\nexport function getAbstractFilePathSafe(app: App, path: string, type: FileSystemType): string {\n  const abstractFile = getAbstractFileOrNull(app, path);\n\n  if (abstractFile && getFileSystemType(abstractFile) === type) {\n    return path;\n  }\n\n  return getAvailablePath(app, path);\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 * Gets a safe file path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @returns The safe path for the file or folder.\n */\nexport function getFilePathSafe(app: App, path: string): string {\n  return getAbstractFilePathSafe(app, path, FileSystemType.File);\n}\n\n/**\n * Gets a safe folder path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @returns The safe path for the file or folder.\n */\nexport function getFolderPathSafe(app: App, path: string): string {\n  return getAbstractFilePathSafe(app, path, FileSystemType.Folder);\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 *\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 or creates an abstract file safely in the specified path.\n *\n * If the file already exists, it will be returned.\n * If the file does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the abstract file to get or create.\n * @param type - The type of the abstract file to get or create.\n * @returns A {@link Promise} that resolves to the abstract file.\n */\nexport async function getOrCreateAbstractFileSafe(app: App, path: string, type: FileSystemType): Promise<TAbstractFile> {\n  path = getAbstractFilePathSafe(app, path, type);\n  const abstractFile = getAbstractFileOrNull(app, path);\n  if (abstractFile) {\n    return abstractFile;\n  }\n\n  switch (type) {\n    case FileSystemType.File:\n      return await app.vault.create(path, '');\n    case FileSystemType.Folder:\n      return await app.vault.createFolder(path);\n    default:\n      throw new Error(`Invalid file system type: ${type as string}`);\n  }\n}\n\n/**\n * Gets or creates a file safely in the specified path.\n *\n * If the file already exists, it will be returned.\n * If the file does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get or create.\n * @returns A {@link Promise} that resolves to the file.\n */\nexport async function getOrCreateFileSafe(app: App, path: string): Promise<TFile> {\n  return asFile(await getOrCreateAbstractFileSafe(app, path, FileSystemType.File));\n}\n\n/**\n * Gets or creates a folder safely in the specified path.\n *\n * If the folder already exists, it will be returned.\n * If the folder does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to get or create.\n * @returns A {@link Promise} that resolves to the folder.\n */\nexport async function getOrCreateFolderSafe(app: App, path: string): Promise<TFolder> {\n  return asFolder(await getOrCreateAbstractFileSafe(app, path, FileSystemType.Folder));\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrAbstractFile - The old path or abstract file to rename.\n * @param newPath - The new path to rename the abstract file to.\n * @returns The safe rename path for the abstract file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrAbstractFile: PathOrAbstractFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrAbstractFile);\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 -- There is no elegant way to perform infinite loops.\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 * Invokes a function with the file system lock.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to execute the function with the file system lock of.\n * @param fn - The function to execute.\n */\nexport async function invokeWithFileSystemLock(app: App, pathOrFile: PathOrFile, fn: (content: string) => void): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  await app.vault.process(file, (content) => {\n    fn(content);\n    return content;\n  });\n}\n\n/**\n * Checks if a path or file is a child of another path or file.\n *\n * @param app - The application instance.\n * @param a - The first path or file.\n * @param b - The second path or file.\n * @returns A boolean indicating whether the first path or file is a child of the second path or file.\n */\nexport function isChild(app: App, a: PathOrAbstractFile, b: PathOrAbstractFile): boolean {\n  const aPath = getPath(app, a);\n  const bPath = getPath(app, b);\n\n  if (aPath === bPath) {\n    return false;\n  }\n\n  if (bPath === '/') {\n    return true;\n  }\n\n  return aPath.startsWith(`${bPath}/`);\n}\n\n/**\n * Checks if a path or file is a child or self of another path or file.\n *\n * @param app - The application instance.\n * @param a - The first path or file.\n * @param b - The second path or file.\n * @returns A boolean indicating whether the first path or file is a child or self of the second path or file.\n */\nexport function isChildOrSelf(app: App, a: PathOrAbstractFile, b: PathOrAbstractFile): boolean {\n  const aPath = getPath(app, a);\n  const bPath = getPath(app, b);\n  return aPath === bPath || isChild(app, a, b);\n}\n\n/**\n * Checks if a folder is empty.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A {@link 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 {@link Promise} that resolves to a {@link 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 {@link 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    shouldLockEditorWhileProcessing: true,\n    shouldShowTimeoutNotice: true,\n    // eslint-disable-next-line no-magic-numbers -- Default value.\n    timeoutInMilliseconds: 500\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  const abortController = new AbortController();\n  fullOptions.abortSignal = abortSignalAny(fullOptions.abortSignal, abortController.signal);\n  const path = getPath(app, pathOrFile);\n\n  let activeLeafChangeEventRef: EventRef | null = null;\n\n  if (fullOptions.shouldLockEditorWhileProcessing) {\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    }\n\n    activeLeafChangeEventRef = app.workspace.on('active-leaf-change', (leaf) => {\n      if (leaf?.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    });\n  }\n\n  try {\n    await retryWithTimeoutNotice({\n      async operationFn(abortSignal) {\n        abortSignal.throwIfAborted();\n\n        const oldContent = await readSafe(app, pathOrFile);\n        abortSignal.throwIfAborted();\n\n        if (oldContent === null) {\n          return handleMissingFile();\n        }\n\n        const newContent = await resolveValue(newContentProvider, abortSignal, oldContent);\n        abortSignal.throwIfAborted();\n\n        if (newContent === null) {\n          return false;\n        }\n\n        let isSuccess = true;\n        const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n          abortSignal.throwIfAborted();\n          await app.vault.process(file, (content) => {\n            abortSignal.throwIfAborted();\n            if (content !== oldContent) {\n              getLibDebugger('Vault:process')('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          abortSignal.throwIfAborted();\n        });\n\n        if (!doesFileExist) {\n          return handleMissingFile();\n        }\n\n        return isSuccess;\n\n        function handleMissingFile(): boolean {\n          if (fullOptions.shouldFailOnMissingFile) {\n            throw new Error(`File '${path}' not found`);\n          }\n          return true;\n        }\n      },\n      operationName: t(($) => $.obsidianDevUtils.vault.processFile, { filePath: path }),\n      retryOptions: fullOptions,\n      shouldShowTimeoutNotice: fullOptions.shouldShowTimeoutNotice\n    });\n  } finally {\n    activeLeafChangeEventRef?.e.offref(activeLeafChangeEventRef);\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        unlockEditor(leaf.view.editor);\n      }\n    }\n  }\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 {@link 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    await saveNote(app, 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 oldPathOrAbstractFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A {@link Promise} that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrAbstractFile: PathOrAbstractFile, newPath: string): Promise<string> {\n  const oldAbstractFile = getAbstractFile(app, oldPathOrAbstractFile);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrAbstractFile, newPath);\n\n  if (oldAbstractFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldAbstractFile.path !== newPath) {\n      await app.fileManager.renameFile(oldAbstractFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.fileManager.renameFile(oldAbstractFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldAbstractFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Saves the specified note in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The note to be saved.\n * @returns A {@link Promise} that resolves when the note is saved.\n */\nexport async function saveNote(app: App, pathOrFile: PathOrFile): Promise<void> {\n  if (!isMarkdownFile(app, pathOrFile)) {\n    return;\n  }\n\n  const path = getPath(app, pathOrFile);\n\n  for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n    if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path && leaf.view.dirty) {\n      await leaf.view.save();\n    }\n  }\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": []
}

|
|
440
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport type {\n  App,\n  EventRef,\n  ListedFiles\n} from 'obsidian';\n\nimport {\n  MarkdownView,\n  TAbstractFile,\n  TFile,\n  TFolder\n} from 'obsidian';\nimport {\n  parentFolderPath,\n  ViewType\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type {\n  PathOrAbstractFile,\n  PathOrFile,\n  PathOrFolder\n} from './FileSystem.ts';\n\nimport { abortSignalAny } from '../AbortController.ts';\nimport { getLibDebugger } from '../Debug.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 { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  lockEditor,\n  unlockEditor\n} from './Editor.ts';\nimport {\n  asFile,\n  asFolder,\n  FileSystemType,\n  getAbstractFile,\n  getAbstractFileOrNull,\n  getFile,\n  getFileOrNull,\n  getFileSystemType,\n  getFolder,\n  getFolderOrNull,\n  getPath,\n  isFile,\n  isMarkdownFile,\n  isNote\n} from './FileSystem.ts';\nimport { t } from './i18n/i18n.ts';\n\n/**\n * Options for {@link process}.\n */\nexport interface ProcessOptions extends RetryOptions {\n  /**\n   * Whether to fail if the file is missing or deleted. Default is `true`.\n   */\n  readonly shouldFailOnMissingFile?: boolean;\n\n  /**\n   * Whether to lock the editor while processing the file. Applicable only for markdown files. Default is `true`.\n   */\n  readonly shouldLockEditorWhileProcessing?: boolean;\n\n  /**\n   * Whether to show a timeout notice. Default is `true`.\n   */\n  readonly shouldShowTimeoutNotice?: 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 {@link 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  if (file.path === newPath) {\n    return newPath;\n  }\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 {@link 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 *\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A {@link 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 trashSafe(app, file);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A {@link 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 folderPath = parentFolderPath(path);\n  await createTempFolder(app, folderPath);\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 trashSafe(app, folder);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Deletes an empty folder.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The folder to delete.\n * @returns A {@link Promise} that resolves when the folder is deleted.\n */\nexport async function deleteEmptyFolder(app: App, pathOrFolder: null | PathOrFolder): Promise<void> {\n  const folder = getFolderOrNull(app, pathOrFolder);\n  if (!folder) {\n    return;\n  }\n  if (!await isEmptyFolder(app, folder)) {\n    return;\n  }\n  await trashSafe(app, folder);\n}\n\n/**\n * Removes empty folder hierarchy starting from the given folder.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The folder to start removing empty hierarchy from.\n * @returns A {@link Promise} that resolves when the empty hierarchy is deleted.\n */\nexport async function deleteEmptyFolderHierarchy(app: App, pathOrFolder: null | PathOrFolder): Promise<void> {\n  let folder = getFolderOrNull(app, pathOrFolder);\n\n  while (folder) {\n    if (!await isEmptyFolder(app, folder)) {\n      return;\n    }\n    const parent = folder.parent;\n    await deleteEmptyFolder(app, folder);\n    folder = parent;\n  }\n}\n\n/**\n * Gets a safe path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @param type - The type of the file system object.\n * @returns The safe path for the file or folder.\n */\nexport function getAbstractFilePathSafe(app: App, path: string, type: FileSystemType): string {\n  const abstractFile = getAbstractFileOrNull(app, path);\n\n  if (abstractFile && getFileSystemType(abstractFile) === type) {\n    return path;\n  }\n\n  return getAvailablePath(app, path);\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 * Gets a safe file path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @returns The safe path for the file or folder.\n */\nexport function getFilePathSafe(app: App, path: string): string {\n  return getAbstractFilePathSafe(app, path, FileSystemType.File);\n}\n\n/**\n * Gets a safe folder path for a file or folder.\n *\n * @param app - The application instance.\n * @param path - The path of the file or folder to get a safe path for.\n * @returns The safe path for the file or folder.\n */\nexport function getFolderPathSafe(app: App, path: string): string {\n  return getAbstractFilePathSafe(app, path, FileSystemType.Folder);\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 *\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 or creates an abstract file safely in the specified path.\n *\n * If the file already exists, it will be returned.\n * If the file does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the abstract file to get or create.\n * @param type - The type of the abstract file to get or create.\n * @returns A {@link Promise} that resolves to the abstract file.\n */\nexport async function getOrCreateAbstractFileSafe(app: App, path: string, type: FileSystemType): Promise<TAbstractFile> {\n  path = getAbstractFilePathSafe(app, path, type);\n  const abstractFile = getAbstractFileOrNull(app, path);\n  if (abstractFile) {\n    return abstractFile;\n  }\n\n  switch (type) {\n    case FileSystemType.File:\n      return await app.vault.create(path, '');\n    case FileSystemType.Folder:\n      return await app.vault.createFolder(path);\n    default:\n      throw new Error(`Invalid file system type: ${type as string}`);\n  }\n}\n\n/**\n * Gets or creates a file safely in the specified path.\n *\n * If the file already exists, it will be returned.\n * If the file does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the file to get or create.\n * @returns A {@link Promise} that resolves to the file.\n */\nexport async function getOrCreateFileSafe(app: App, path: string): Promise<TFile> {\n  return asFile(await getOrCreateAbstractFileSafe(app, path, FileSystemType.File));\n}\n\n/**\n * Gets or creates a folder safely in the specified path.\n *\n * If the folder already exists, it will be returned.\n * If the folder does not exist, it will be created and returned.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to get or create.\n * @returns A {@link Promise} that resolves to the folder.\n */\nexport async function getOrCreateFolderSafe(app: App, path: string): Promise<TFolder> {\n  return asFolder(await getOrCreateAbstractFileSafe(app, path, FileSystemType.Folder));\n}\n\n/**\n * Gets a safe rename path for a file.\n *\n * @param app - The application instance.\n * @param oldPathOrAbstractFile - The old path or abstract file to rename.\n * @param newPath - The new path to rename the abstract file to.\n * @returns The safe rename path for the abstract file.\n */\nexport function getSafeRenamePath(app: App, oldPathOrAbstractFile: PathOrAbstractFile, newPath: string): string {\n  const oldPath = getPath(app, oldPathOrAbstractFile);\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 -- There is no elegant way to perform infinite loops.\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 * Invokes a function with the file system lock.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or file to execute the function with the file system lock of.\n * @param fn - The function to execute.\n */\nexport async function invokeWithFileSystemLock(app: App, pathOrFile: PathOrFile, fn: (content: string) => void): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  await app.vault.process(file, (content) => {\n    fn(content);\n    return content;\n  });\n}\n\n/**\n * Checks if a path or file is a child of another path or file.\n *\n * @param app - The application instance.\n * @param a - The first path or file.\n * @param b - The second path or file.\n * @returns A boolean indicating whether the first path or file is a child of the second path or file.\n */\nexport function isChild(app: App, a: PathOrAbstractFile, b: PathOrAbstractFile): boolean {\n  const aPath = getPath(app, a);\n  const bPath = getPath(app, b);\n\n  if (aPath === bPath) {\n    return false;\n  }\n\n  if (bPath === '/') {\n    return true;\n  }\n\n  return aPath.startsWith(`${bPath}/`);\n}\n\n/**\n * Checks if a path or file is a child or self of another path or file.\n *\n * @param app - The application instance.\n * @param a - The first path or file.\n * @param b - The second path or file.\n * @returns A boolean indicating whether the first path or file is a child or self of the second path or file.\n */\nexport function isChildOrSelf(app: App, a: PathOrAbstractFile, b: PathOrAbstractFile): boolean {\n  const aPath = getPath(app, a);\n  const bPath = getPath(app, b);\n  return aPath === bPath || isChild(app, a, b);\n}\n\n/**\n * Checks if a folder is empty.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The path or folder to check.\n * @returns A {@link 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 {@link Promise} that resolves to a {@link 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 {@link 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    shouldLockEditorWhileProcessing: true,\n    shouldShowTimeoutNotice: true,\n    // eslint-disable-next-line no-magic-numbers -- Default value.\n    timeoutInMilliseconds: 500\n  };\n  const fullOptions = { ...DEFAULT_RETRY_OPTIONS, ...options };\n  const abortController = new AbortController();\n  fullOptions.abortSignal = abortSignalAny(fullOptions.abortSignal, abortController.signal);\n  const path = getPath(app, pathOrFile);\n\n  let activeLeafChangeEventRef: EventRef | null = null;\n\n  if (fullOptions.shouldLockEditorWhileProcessing) {\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    }\n\n    activeLeafChangeEventRef = app.workspace.on('active-leaf-change', (leaf) => {\n      if (leaf?.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        lockEditor(leaf.view.editor);\n      }\n    });\n  }\n\n  try {\n    await retryWithTimeoutNotice({\n      async operationFn(abortSignal) {\n        abortSignal.throwIfAborted();\n\n        const oldContent = await readSafe(app, pathOrFile);\n        abortSignal.throwIfAborted();\n\n        if (oldContent === null) {\n          return handleMissingFile();\n        }\n\n        const newContent = await resolveValue(newContentProvider, abortSignal, oldContent);\n        abortSignal.throwIfAborted();\n\n        if (newContent === null) {\n          return false;\n        }\n\n        let isSuccess = true;\n        const doesFileExist = await invokeFileActionSafe(app, pathOrFile, async (file) => {\n          abortSignal.throwIfAborted();\n          await app.vault.process(file, (content) => {\n            abortSignal.throwIfAborted();\n            if (content !== oldContent) {\n              getLibDebugger('Vault:process')('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          abortSignal.throwIfAborted();\n        });\n\n        if (!doesFileExist) {\n          return handleMissingFile();\n        }\n\n        return isSuccess;\n\n        function handleMissingFile(): boolean {\n          if (fullOptions.shouldFailOnMissingFile) {\n            throw new Error(`File '${path}' not found`);\n          }\n          return true;\n        }\n      },\n      operationName: t(($) => $.obsidianDevUtils.vault.processFile, { filePath: path }),\n      retryOptions: fullOptions,\n      shouldShowTimeoutNotice: fullOptions.shouldShowTimeoutNotice\n    });\n  } finally {\n    activeLeafChangeEventRef?.e.offref(activeLeafChangeEventRef);\n    for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n      if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path) {\n        unlockEditor(leaf.view.editor);\n      }\n    }\n  }\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 {@link 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    await saveNote(app, 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 oldPathOrAbstractFile - The old path or file to rename.\n * @param newPath - The new path to rename the file to.\n * @returns A {@link Promise} that resolves to the new path of the file.\n */\nexport async function renameSafe(app: App, oldPathOrAbstractFile: PathOrAbstractFile, newPath: string): Promise<string> {\n  const oldAbstractFile = getAbstractFile(app, oldPathOrAbstractFile);\n\n  const newAvailablePath = getSafeRenamePath(app, oldPathOrAbstractFile, newPath);\n\n  if (oldAbstractFile.path.toLowerCase() === newAvailablePath.toLowerCase()) {\n    if (oldAbstractFile.path !== newPath) {\n      await app.fileManager.renameFile(oldAbstractFile, newAvailablePath);\n    }\n    return newAvailablePath;\n  }\n\n  const newFolderPath = parentFolderPath(newAvailablePath);\n  await createFolderSafe(app, newFolderPath);\n\n  try {\n    await app.fileManager.renameFile(oldAbstractFile, newAvailablePath);\n  } catch (e) {\n    if (!await app.vault.exists(newAvailablePath) || await app.vault.exists(oldAbstractFile.path)) {\n      throw e;\n    }\n  }\n\n  return newAvailablePath;\n}\n\n/**\n * Saves the specified note in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The note to be saved.\n * @returns A {@link Promise} that resolves when the note is saved.\n */\nexport async function saveNote(app: App, pathOrFile: PathOrFile): Promise<void> {\n  if (!isMarkdownFile(app, pathOrFile)) {\n    return;\n  }\n\n  const path = getPath(app, pathOrFile);\n\n  for (const leaf of app.workspace.getLeavesOfType(ViewType.Markdown)) {\n    if (leaf.view instanceof MarkdownView && leaf.view.file?.path === path && leaf.view.dirty) {\n      await leaf.view.save();\n    }\n  }\n}\n\n/**\n * Trashes an abstract file safely from the vault.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or abstract file to trash.\n * @returns A {@link Promise} that resolves when the file is trashed.\n */\nexport async function trashSafe(app: App, pathOrFile: PathOrAbstractFile): Promise<void> {\n  const file = getAbstractFileOrNull(app, pathOrFile);\n  if (!file) {\n    return;\n  }\n\n  try {\n    await app.fileManager.trashFile(file);\n  } catch (e) {\n    if (await app.vault.exists(file.path)) {\n      throw e;\n    }\n\n    getLibDebugger('Vault:trashSafe')(`An error occurred while trashing ${file.path}, but the file no longer exists.`, { error: e, path: file.path });\n  }\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": []
}

|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @packageDocumentation
|
|
3
|
+
*
|
|
4
|
+
* This module provides deletion utilities that require metadata cache access.
|
|
5
|
+
*/
|
|
6
|
+
import type { App } from 'obsidian';
|
|
7
|
+
import type { PathOrAbstractFile } from './FileSystem.mjs';
|
|
8
|
+
/**
|
|
9
|
+
* Deletes an abstract file safely from the vault, but only if it is not referenced by other notes.
|
|
10
|
+
*
|
|
11
|
+
* @param app - The Obsidian application instance.
|
|
12
|
+
* @param pathOrFile - The path or abstract file to delete.
|
|
13
|
+
* @param deletedNotePath - Optional. The path of the note that triggered the removal.
|
|
14
|
+
* @param shouldReportUsedAttachments - Optional. If `true`, a notice will be shown for each attachment that is still used by other notes.
|
|
15
|
+
* @param shouldDeleteEmptyFolders - Optional. If `true`, empty folders will be deleted.
|
|
16
|
+
* @returns A {@link Promise} that resolves to a boolean indicating whether the removal was successful.
|
|
17
|
+
*/
|
|
18
|
+
export declare function deleteIfNotUsed(app: App, pathOrFile: PathOrAbstractFile, deletedNotePath?: string, shouldReportUsedAttachments?: boolean, shouldDeleteEmptyFolders?: boolean): Promise<boolean>;
|
|
@@ -0,0 +1,73 @@
|
|
|
1
|
+
/*
|
|
2
|
+
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
|
3
|
+
if you want to view the source, please visit the github repository of this plugin
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
(function initEsm() {
|
|
7
|
+
if (globalThis.process) {
|
|
8
|
+
return;
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
const browserProcess = {
|
|
12
|
+
browser: true,
|
|
13
|
+
cwd() {
|
|
14
|
+
return '/';
|
|
15
|
+
},
|
|
16
|
+
env: {},
|
|
17
|
+
platform: 'android'
|
|
18
|
+
};
|
|
19
|
+
globalThis.process = browserProcess;
|
|
20
|
+
})();
|
|
21
|
+
|
|
22
|
+
import { Notice } from "obsidian";
|
|
23
|
+
import { printError } from "../Error.mjs";
|
|
24
|
+
import {
|
|
25
|
+
getAbstractFileOrNull,
|
|
26
|
+
isFile,
|
|
27
|
+
isFolder
|
|
28
|
+
} from "./FileSystem.mjs";
|
|
29
|
+
import { t } from "./i18n/i18n.mjs";
|
|
30
|
+
import { getBacklinksForFileSafe } from "./MetadataCache.mjs";
|
|
31
|
+
import {
|
|
32
|
+
isEmptyFolder,
|
|
33
|
+
listSafe,
|
|
34
|
+
trashSafe
|
|
35
|
+
} from "./Vault.mjs";
|
|
36
|
+
async function deleteIfNotUsed(app, pathOrFile, deletedNotePath, shouldReportUsedAttachments, shouldDeleteEmptyFolders) {
|
|
37
|
+
const file = getAbstractFileOrNull(app, pathOrFile);
|
|
38
|
+
if (!file) {
|
|
39
|
+
return false;
|
|
40
|
+
}
|
|
41
|
+
let canDelete = isFile(file) || (shouldDeleteEmptyFolders ?? true);
|
|
42
|
+
if (isFile(file)) {
|
|
43
|
+
const backlinks = await getBacklinksForFileSafe(app, file);
|
|
44
|
+
if (deletedNotePath) {
|
|
45
|
+
backlinks.clear(deletedNotePath);
|
|
46
|
+
}
|
|
47
|
+
if (backlinks.count() !== 0) {
|
|
48
|
+
if (shouldReportUsedAttachments) {
|
|
49
|
+
new Notice(t(($) => $.obsidianDevUtils.notices.attachmentIsStillUsed, { attachmentPath: file.path }));
|
|
50
|
+
}
|
|
51
|
+
canDelete = false;
|
|
52
|
+
}
|
|
53
|
+
} else if (isFolder(file)) {
|
|
54
|
+
const listedFiles = await listSafe(app, file);
|
|
55
|
+
for (const child of [...listedFiles.files, ...listedFiles.folders]) {
|
|
56
|
+
canDelete &&= await deleteIfNotUsed(app, child, deletedNotePath, shouldReportUsedAttachments);
|
|
57
|
+
}
|
|
58
|
+
canDelete &&= await isEmptyFolder(app, file);
|
|
59
|
+
}
|
|
60
|
+
if (canDelete) {
|
|
61
|
+
try {
|
|
62
|
+
await trashSafe(app, file);
|
|
63
|
+
} catch (e) {
|
|
64
|
+
printError(new Error(`Failed to delete ${file.path}`, { cause: e }));
|
|
65
|
+
canDelete = false;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return canDelete;
|
|
69
|
+
}
|
|
70
|
+
export {
|
|
71
|
+
deleteIfNotUsed
|
|
72
|
+
};
|
|
73
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1ZhdWx0RGVsZXRlLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIEBwYWNrYWdlRG9jdW1lbnRhdGlvblxuICpcbiAqIFRoaXMgbW9kdWxlIHByb3ZpZGVzIGRlbGV0aW9uIHV0aWxpdGllcyB0aGF0IHJlcXVpcmUgbWV0YWRhdGEgY2FjaGUgYWNjZXNzLlxuICovXG5cbmltcG9ydCB0eXBlIHsgQXBwIH0gZnJvbSAnb2JzaWRpYW4nO1xuXG5pbXBvcnQgeyBOb3RpY2UgfSBmcm9tICdvYnNpZGlhbic7XG5cbmltcG9ydCB0eXBlIHsgUGF0aE9yQWJzdHJhY3RGaWxlIH0gZnJvbSAnLi9GaWxlU3lzdGVtLnRzJztcblxuaW1wb3J0IHsgcHJpbnRFcnJvciB9IGZyb20gJy4uL0Vycm9yLnRzJztcbmltcG9ydCB7XG4gIGdldEFic3RyYWN0RmlsZU9yTnVsbCxcbiAgaXNGaWxlLFxuICBpc0ZvbGRlclxufSBmcm9tICcuL0ZpbGVTeXN0ZW0udHMnO1xuaW1wb3J0IHsgdCB9IGZyb20gJy4vaTE4bi9pMThuLnRzJztcbmltcG9ydCB7IGdldEJhY2tsaW5rc0ZvckZpbGVTYWZlIH0gZnJvbSAnLi9NZXRhZGF0YUNhY2hlLnRzJztcbmltcG9ydCB7XG4gIGlzRW1wdHlGb2xkZXIsXG4gIGxpc3RTYWZlLFxuICB0cmFzaFNhZmVcbn0gZnJvbSAnLi9WYXVsdC50cyc7XG5cbi8qKlxuICogRGVsZXRlcyBhbiBhYnN0cmFjdCBmaWxlIHNhZmVseSBmcm9tIHRoZSB2YXVsdCwgYnV0IG9ubHkgaWYgaXQgaXMgbm90IHJlZmVyZW5jZWQgYnkgb3RoZXIgbm90ZXMuXG4gKlxuICogQHBhcmFtIGFwcCAtIFRoZSBPYnNpZGlhbiBhcHBsaWNhdGlvbiBpbnN0YW5jZS5cbiAqIEBwYXJhbSBwYXRoT3JGaWxlIC0gVGhlIHBhdGggb3IgYWJzdHJhY3QgZmlsZSB0byBkZWxldGUuXG4gKiBAcGFyYW0gZGVsZXRlZE5vdGVQYXRoIC0gT3B0aW9uYWwuIFRoZSBwYXRoIG9mIHRoZSBub3RlIHRoYXQgdHJpZ2dlcmVkIHRoZSByZW1vdmFsLlxuICogQHBhcmFtIHNob3VsZFJlcG9ydFVzZWRBdHRhY2htZW50cyAtIE9wdGlvbmFsLiBJZiBgdHJ1ZWAsIGEgbm90aWNlIHdpbGwgYmUgc2hvd24gZm9yIGVhY2ggYXR0YWNobWVudCB0aGF0IGlzIHN0aWxsIHVzZWQgYnkgb3RoZXIgbm90ZXMuXG4gKiBAcGFyYW0gc2hvdWxkRGVsZXRlRW1wdHlGb2xkZXJzIC0gT3B0aW9uYWwuIElmIGB0cnVlYCwgZW1wdHkgZm9sZGVycyB3aWxsIGJlIGRlbGV0ZWQuXG4gKiBAcmV0dXJucyBBIHtAbGluayBQcm9taXNlfSB0aGF0IHJlc29sdmVzIHRvIGEgYm9vbGVhbiBpbmRpY2F0aW5nIHdoZXRoZXIgdGhlIHJlbW92YWwgd2FzIHN1Y2Nlc3NmdWwuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBkZWxldGVJZk5vdFVzZWQoXG4gIGFwcDogQXBwLFxuICBwYXRoT3JGaWxlOiBQYXRoT3JBYnN0cmFjdEZpbGUsXG4gIGRlbGV0ZWROb3RlUGF0aD86IHN0cmluZyxcbiAgc2hvdWxkUmVwb3J0VXNlZEF0dGFjaG1lbnRzPzogYm9vbGVhbixcbiAgc2hvdWxkRGVsZXRlRW1wdHlGb2xkZXJzPzogYm9vbGVhblxuKTogUHJvbWlzZTxib29sZWFuPiB7XG4gIGNvbnN0IGZpbGUgPSBnZXRBYnN0cmFjdEZpbGVPck51bGwoYXBwLCBwYXRoT3JGaWxlKTtcblxuICBpZiAoIWZpbGUpIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICBsZXQgY2FuRGVsZXRlID0gaXNGaWxlKGZpbGUpIHx8IChzaG91bGREZWxldGVFbXB0eUZvbGRlcnMgPz8gdHJ1ZSk7XG5cbiAgLyogdjggaWdub3JlIHN0YXJ0IC0tIFRBYnN0cmFjdEZpbGUgaXMgYWx3YXlzIFRGaWxlIG9yIFRGb2xkZXIgaW4gT2JzaWRpYW47IHRoZSBmYWxzZSBicmFuY2ggb2YgaXNGaWxlIGxlYWRzIHRvIGlzRm9sZGVyLiAqL1xuICBpZiAoaXNGaWxlKGZpbGUpKSB7XG4gICAgLyogdjggaWdub3JlIHN0b3AgKi9cbiAgICBjb25zdCBiYWNrbGlua3MgPSBhd2FpdCBnZXRCYWNrbGlua3NGb3JGaWxlU2FmZShhcHAsIGZpbGUpO1xuICAgIGlmIChkZWxldGVkTm90ZVBhdGgpIHtcbiAgICAgIGJhY2tsaW5rcy5jbGVhcihkZWxldGVkTm90ZVBhdGgpO1xuICAgIH1cbiAgICBpZiAoYmFja2xpbmtzLmNvdW50KCkgIT09IDApIHtcbiAgICAgIGlmIChzaG91bGRSZXBvcnRVc2VkQXR0YWNobWVudHMpIHtcbiAgICAgICAgbmV3IE5vdGljZSh0KCgkKSA9PiAkLm9ic2lkaWFuRGV2VXRpbHMubm90aWNlcy5hdHRhY2htZW50SXNTdGlsbFVzZWQsIHsgYXR0YWNobWVudFBhdGg6IGZpbGUucGF0aCB9KSk7XG4gICAgICB9XG4gICAgICBjYW5EZWxldGUgPSBmYWxzZTtcbiAgICB9XG4gICAgLyogdjggaWdub3JlIHN0YXJ0IC0tIFRBYnN0cmFjdEZpbGUgaXMgYWx3YXlzIFRGaWxlIG9yIFRGb2xkZXIgaW4gT2JzaWRpYW47IGRlZmVuc2l2ZSBmYWxsYmFjay4gKi9cbiAgfSBlbHNlIGlmIChpc0ZvbGRlcihmaWxlKSkge1xuICAgIC8qIHY4IGlnbm9yZSBzdG9wICovXG4gICAgY29uc3QgbGlzdGVkRmlsZXMgPSBhd2FpdCBsaXN0U2FmZShhcHAsIGZpbGUpO1xuICAgIGZvciAoY29uc3QgY2hpbGQgb2YgWy4uLmxpc3RlZEZpbGVzLmZpbGVzLCAuLi5saXN0ZWRGaWxlcy5mb2xkZXJzXSkge1xuICAgICAgY2FuRGVsZXRlICYmPSBhd2FpdCBkZWxldGVJZk5vdFVzZWQoYXBwLCBjaGlsZCwgZGVsZXRlZE5vdGVQYXRoLCBzaG91bGRSZXBvcnRVc2VkQXR0YWNobWVudHMpO1xuICAgIH1cblxuICAgIGNhbkRlbGV0ZSAmJj0gYXdhaXQgaXNFbXB0eUZvbGRlcihhcHAsIGZpbGUpO1xuICB9XG5cbiAgaWYgKGNhbkRlbGV0ZSkge1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCB0cmFzaFNhZmUoYXBwLCBmaWxlKTtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBwcmludEVycm9yKG5ldyBFcnJvcihgRmFpbGVkIHRvIGRlbGV0ZSAke2ZpbGUucGF0aH1gLCB7IGNhdXNlOiBlIH0pKTtcbiAgICAgIGNhbkRlbGV0ZSA9IGZhbHNlO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiBjYW5EZWxldGU7XG59XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFRQSxTQUFTLGNBQWM7QUFJdkIsU0FBUyxrQkFBa0I7QUFDM0I7QUFBQSxFQUNFO0FBQUEsRUFDQTtBQUFBLEVBQ0E7QUFBQSxPQUNLO0FBQ1AsU0FBUyxTQUFTO0FBQ2xCLFNBQVMsK0JBQStCO0FBQ3hDO0FBQUEsRUFDRTtBQUFBLEVBQ0E7QUFBQSxFQUNBO0FBQUEsT0FDSztBQVlQLGVBQXNCLGdCQUNwQixLQUNBLFlBQ0EsaUJBQ0EsNkJBQ0EsMEJBQ2tCO0FBQ2xCLFFBQU0sT0FBTyxzQkFBc0IsS0FBSyxVQUFVO0FBRWxELE1BQUksQ0FBQyxNQUFNO0FBQ1QsV0FBTztBQUFBLEVBQ1Q7QUFFQSxNQUFJLFlBQVksT0FBTyxJQUFJLE1BQU0sNEJBQTRCO0FBRzdELE1BQUksT0FBTyxJQUFJLEdBQUc7QUFFaEIsVUFBTSxZQUFZLE1BQU0sd0JBQXdCLEtBQUssSUFBSTtBQUN6RCxRQUFJLGlCQUFpQjtBQUNuQixnQkFBVSxNQUFNLGVBQWU7QUFBQSxJQUNqQztBQUNBLFFBQUksVUFBVSxNQUFNLE1BQU0sR0FBRztBQUMzQixVQUFJLDZCQUE2QjtBQUMvQixZQUFJLE9BQU8sRUFBRSxDQUFDLE1BQU0sRUFBRSxpQkFBaUIsUUFBUSx1QkFBdUIsRUFBRSxnQkFBZ0IsS0FBSyxLQUFLLENBQUMsQ0FBQztBQUFBLE1BQ3RHO0FBQ0Esa0JBQVk7QUFBQSxJQUNkO0FBQUEsRUFFRixXQUFXLFNBQVMsSUFBSSxHQUFHO0FBRXpCLFVBQU0sY0FBYyxNQUFNLFNBQVMsS0FBSyxJQUFJO0FBQzVDLGVBQVcsU0FBUyxDQUFDLEdBQUcsWUFBWSxPQUFPLEdBQUcsWUFBWSxPQUFPLEdBQUc7QUFDbEUsb0JBQWMsTUFBTSxnQkFBZ0IsS0FBSyxPQUFPLGlCQUFpQiwyQkFBMkI7QUFBQSxJQUM5RjtBQUVBLGtCQUFjLE1BQU0sY0FBYyxLQUFLLElBQUk7QUFBQSxFQUM3QztBQUVBLE1BQUksV0FBVztBQUNiLFFBQUk7QUFDRixZQUFNLFVBQVUsS0FBSyxJQUFJO0FBQUEsSUFDM0IsU0FBUyxHQUFHO0FBQ1YsaUJBQVcsSUFBSSxNQUFNLG9CQUFvQixLQUFLLElBQUksSUFBSSxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7QUFDbkUsa0JBQVk7QUFBQSxJQUNkO0FBQUEsRUFDRjtBQUVBLFNBQU87QUFDVDsiLAogICJuYW1lcyI6IFtdCn0K
|
|
@@ -36,5 +36,5 @@ export * as SettingEx from './SettingEx.mjs';
|
|
|
36
36
|
export * as SettingGroupEx from './SettingGroupEx.mjs';
|
|
37
37
|
export * as Validation from './Validation.mjs';
|
|
38
38
|
export * as Vault from './Vault.mjs';
|
|
39
|
-
export * as
|
|
39
|
+
export * as VaultDelete from './VaultDelete.mjs';
|
|
40
40
|
export * as Workspace from './Workspace.mjs';
|
|
@@ -57,7 +57,7 @@ import * as SettingEx from "./SettingEx.mjs";
|
|
|
57
57
|
import * as SettingGroupEx from "./SettingGroupEx.mjs";
|
|
58
58
|
import * as Validation from "./Validation.mjs";
|
|
59
59
|
import * as Vault from "./Vault.mjs";
|
|
60
|
-
import * as
|
|
60
|
+
import * as VaultDelete from "./VaultDelete.mjs";
|
|
61
61
|
import * as Workspace from "./Workspace.mjs";
|
|
62
62
|
export {
|
|
63
63
|
App,
|
|
@@ -97,8 +97,8 @@ export {
|
|
|
97
97
|
SettingGroupEx,
|
|
98
98
|
Validation,
|
|
99
99
|
Vault,
|
|
100
|
-
|
|
100
|
+
VaultDelete,
|
|
101
101
|
Workspace,
|
|
102
102
|
i18n
|
|
103
103
|
};
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL2luZGV4LnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKiBUSElTIElTIEEgR0VORVJBVEVEL0JVTkRMRUQgRklMRSBCWSBCVUlMRCBTQ1JJUFQgKi9cblxuZXhwb3J0ICogYXMgQXBwIGZyb20gJy4vQXBwLnRzJztcbmV4cG9ydCAqIGFzIEFzeW5jV2l0aE5vdGljZSBmcm9tICcuL0FzeW5jV2l0aE5vdGljZS50cyc7XG5leHBvcnQgKiBhcyBBdHRhY2htZW50UGF0aCBmcm9tICcuL0F0dGFjaG1lbnRQYXRoLnRzJztcbmV4cG9ydCAqIGFzIEJhY2tsaW5rIGZyb20gJy4vQmFja2xpbmsudHMnO1xuZXhwb3J0ICogYXMgQ2FsbG91dCBmcm9tICcuL0NhbGxvdXQudHMnO1xuZXhwb3J0ICogYXMgQ29kZUJsb2NrTWFya2Rvd25JbmZvcm1hdGlvbiBmcm9tICcuL0NvZGVCbG9ja01hcmtkb3duSW5mb3JtYXRpb24udHMnO1xuZXhwb3J0ICogYXMgQ29tbWFuZHMgZnJvbSAnLi9Db21tYW5kcy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBDb21wb25lbnRzIGZyb20gJy4vQ29tcG9uZW50cy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBEYXRhdmlldyBmcm9tICcuL0RhdGF2aWV3LnRzJztcbmV4cG9ydCAqIGFzIERhdGF2aWV3TGluayBmcm9tICcuL0RhdGF2aWV3TGluay50cyc7XG5leHBvcnQgKiBhcyBFZGl0b3IgZnJvbSAnLi9FZGl0b3IudHMnO1xuZXhwb3J0ICogYXMgRmlsZUNoYW5nZSBmcm9tICcuL0ZpbGVDaGFuZ2UudHMnO1xuZXhwb3J0ICogYXMgRmlsZU1hbmFnZXIgZnJvbSAnLi9GaWxlTWFuYWdlci50cyc7XG5leHBvcnQgKiBhcyBGaWxlU3lzdGVtIGZyb20gJy4vRmlsZVN5c3RlbS50cyc7XG5leHBvcnQgKiBhcyBGcm9udG1hdHRlciBmcm9tICcuL0Zyb250bWF0dGVyLnRzJztcbmV4cG9ydCAqIGFzIEZyb250bWF0dGVyTGlua0NhY2hlV2l0aE9mZnNldHMgZnJvbSAnLi9Gcm9udG1hdHRlckxpbmtDYWNoZVdpdGhPZmZzZXRzLnRzJztcbmV4cG9ydCAqIGFzIGkxOG4gZnJvbSAnLi9pMThuL2luZGV4LnRzJztcbmV4cG9ydCAqIGFzIExpbmsgZnJvbSAnLi9MaW5rLnRzJztcbmV4cG9ydCAqIGFzIExvZ2dlciBmcm9tICcuL0xvZ2dlci50cyc7XG5leHBvcnQgKiBhcyBMb29wIGZyb20gJy4vTG9vcC50cyc7XG5leHBvcnQgKiBhcyBNYXJrZG93biBmcm9tICcuL01hcmtkb3duLnRzJztcbmV4cG9ydCAqIGFzIE1hcmtkb3duQ29kZUJsb2NrUHJvY2Vzc29yIGZyb20gJy4vTWFya2Rvd25Db2RlQmxvY2tQcm9jZXNzb3IudHMnO1xuZXhwb3J0ICogYXMgTWFya2Rvd25WaWV3IGZyb20gJy4vTWFya2Rvd25WaWV3LnRzJztcbmV4cG9ydCAqIGFzIE1ldGFkYXRhQ2FjaGUgZnJvbSAnLi9NZXRhZGF0YUNhY2hlLnRzJztcbmV4cG9ydCAqIGFzIE1vZGFscyBmcm9tICcuL01vZGFscy9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBNb25rZXlBcm91bmQgZnJvbSAnLi9Nb25rZXlBcm91bmQudHMnO1xuZXhwb3J0ICogYXMgT2JzaWRpYW5TZXR0aW5ncyBmcm9tICcuL09ic2lkaWFuU2V0dGluZ3MudHMnO1xuZXhwb3J0ICogYXMgUGRmIGZyb20gJy4vUGRmLnRzJztcbmV4cG9ydCAqIGFzIFBsdWdpbiBmcm9tICcuL1BsdWdpbi9pbmRleC50cyc7XG5leHBvcnQgKiBhcyBRdWV1ZSBmcm9tICcuL1F1ZXVlLnRzJztcbmV4cG9ydCAqIGFzIFJlYWN0IGZyb20gJy4vUmVhY3QvaW5kZXgudHMnO1xuZXhwb3J0ICogYXMgUmVmZXJlbmNlIGZyb20gJy4vUmVmZXJlbmNlLnRzJztcbmV4cG9ydCAqIGFzIFJlbmFtZURlbGV0ZUhhbmRsZXIgZnJvbSAnLi9SZW5hbWVEZWxldGVIYW5kbGVyLnRzJztcbmV4cG9ydCAqIGFzIFJlc291cmNlVXJsIGZyb20gJy4vUmVzb3VyY2VVcmwudHMnO1xuZXhwb3J0ICogYXMgU2V0dGluZ0V4IGZyb20gJy4vU2V0dGluZ0V4LnRzJztcbmV4cG9ydCAqIGFzIFNldHRpbmdHcm91cEV4IGZyb20gJy4vU2V0dGluZ0dyb3VwRXgudHMnO1xuZXhwb3J0ICogYXMgVmFsaWRhdGlvbiBmcm9tICcuL1ZhbGlkYXRpb24udHMnO1xuZXhwb3J0ICogYXMgVmF1bHQgZnJvbSAnLi9WYXVsdC50cyc7XG5leHBvcnQgKiBhcyBWYXVsdERlbGV0ZSBmcm9tICcuL1ZhdWx0RGVsZXRlLnRzJztcbmV4cG9ydCAqIGFzIFdvcmtzcGFjZSBmcm9tICcuL1dvcmtzcGFjZS50cyc7XG4iXSwKICAibWFwcGluZ3MiOiAiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFQSxZQUFZLFNBQVM7QUFDckIsWUFBWSxxQkFBcUI7QUFDakMsWUFBWSxvQkFBb0I7QUFDaEMsWUFBWSxjQUFjO0FBQzFCLFlBQVksYUFBYTtBQUN6QixZQUFZLGtDQUFrQztBQUM5QyxZQUFZLGNBQWM7QUFDMUIsWUFBWSxnQkFBZ0I7QUFDNUIsWUFBWSxjQUFjO0FBQzFCLFlBQVksa0JBQWtCO0FBQzlCLFlBQVksWUFBWTtBQUN4QixZQUFZLGdCQUFnQjtBQUM1QixZQUFZLGlCQUFpQjtBQUM3QixZQUFZLGdCQUFnQjtBQUM1QixZQUFZLGlCQUFpQjtBQUM3QixZQUFZLHFDQUFxQztBQUNqRCxZQUFZLFVBQVU7QUFDdEIsWUFBWSxVQUFVO0FBQ3RCLFlBQVksWUFBWTtBQUN4QixZQUFZLFVBQVU7QUFDdEIsWUFBWSxjQUFjO0FBQzFCLFlBQVksZ0NBQWdDO0FBQzVDLFlBQVksa0JBQWtCO0FBQzlCLFlBQVksbUJBQW1CO0FBQy9CLFlBQVksWUFBWTtBQUN4QixZQUFZLGtCQUFrQjtBQUM5QixZQUFZLHNCQUFzQjtBQUNsQyxZQUFZLFNBQVM7QUFDckIsWUFBWSxZQUFZO0FBQ3hCLFlBQVksV0FBVztBQUN2QixZQUFZLFdBQVc7QUFDdkIsWUFBWSxlQUFlO0FBQzNCLFlBQVkseUJBQXlCO0FBQ3JDLFlBQVksaUJBQWlCO0FBQzdCLFlBQVksZUFBZTtBQUMzQixZQUFZLG9CQUFvQjtBQUNoQyxZQUFZLGdCQUFnQjtBQUM1QixZQUFZLFdBQVc7QUFDdkIsWUFBWSxpQkFBaUI7QUFDN0IsWUFBWSxlQUFlOyIsCiAgIm5hbWVzIjogW10KfQo=
|
package/package.json
CHANGED
|
@@ -1,198 +0,0 @@
|
|
|
1
|
-
/*
|
|
2
|
-
THIS IS A GENERATED/BUNDLED FILE BY ESBUILD
|
|
3
|
-
if you want to view the source, please visit the github repository of this plugin
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
(function initCjs() {
|
|
7
|
-
const globalThisRecord = globalThis;
|
|
8
|
-
globalThisRecord['__name'] ??= name;
|
|
9
|
-
const originalRequire = require;
|
|
10
|
-
if (originalRequire && !originalRequire.__isPatched) {
|
|
11
|
-
// eslint-disable-next-line no-global-assign, no-implicit-globals -- We need to patch the `require()` function.
|
|
12
|
-
require = Object.assign(
|
|
13
|
-
(id) => requirePatched(id),
|
|
14
|
-
originalRequire,
|
|
15
|
-
{
|
|
16
|
-
__isPatched: true
|
|
17
|
-
}
|
|
18
|
-
);
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
const newFuncs = {
|
|
22
|
-
__extractDefault() {
|
|
23
|
-
return extractDefault;
|
|
24
|
-
},
|
|
25
|
-
process() {
|
|
26
|
-
const browserProcess = {
|
|
27
|
-
browser: true,
|
|
28
|
-
cwd() {
|
|
29
|
-
return '/';
|
|
30
|
-
},
|
|
31
|
-
env: {},
|
|
32
|
-
platform: 'android'
|
|
33
|
-
};
|
|
34
|
-
return browserProcess;
|
|
35
|
-
}
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
for (const key of Object.keys(newFuncs)) {
|
|
39
|
-
globalThisRecord[key] ??= newFuncs[key]?.();
|
|
40
|
-
}
|
|
41
|
-
|
|
42
|
-
function name(obj) {
|
|
43
|
-
return obj;
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
function extractDefault(module) {
|
|
47
|
-
return module && module.__esModule && 'default' in module ? module.default : module;
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
const OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
51
|
-
'obsidian',
|
|
52
|
-
'@codemirror/autocomplete',
|
|
53
|
-
'@codemirror/collab',
|
|
54
|
-
'@codemirror/commands',
|
|
55
|
-
'@codemirror/language',
|
|
56
|
-
'@codemirror/lint',
|
|
57
|
-
'@codemirror/search',
|
|
58
|
-
'@codemirror/state',
|
|
59
|
-
'@codemirror/text',
|
|
60
|
-
'@codemirror/view',
|
|
61
|
-
'@lezer/common',
|
|
62
|
-
'@lezer/lr',
|
|
63
|
-
'@lezer/highlight'];
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
const DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES = [
|
|
67
|
-
'@codemirror/closebrackets',
|
|
68
|
-
'@codemirror/comment',
|
|
69
|
-
'@codemirror/fold',
|
|
70
|
-
'@codemirror/gutter',
|
|
71
|
-
'@codemirror/highlight',
|
|
72
|
-
'@codemirror/history',
|
|
73
|
-
'@codemirror/matchbrackets',
|
|
74
|
-
'@codemirror/panel',
|
|
75
|
-
'@codemirror/rangeset',
|
|
76
|
-
'@codemirror/rectangular-selection',
|
|
77
|
-
'@codemirror/stream-parser',
|
|
78
|
-
'@codemirror/tooltip'];
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
function requirePatched(id) {
|
|
82
|
-
if (OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id) || DEPRECATED_OBSIDIAN_BUILT_IN_MODULE_NAMES.includes(id)) {
|
|
83
|
-
return originalRequire?.(id);
|
|
84
|
-
}
|
|
85
|
-
|
|
86
|
-
// eslint-disable-next-line @typescript-eslint/no-deprecated, @typescript-eslint/no-unnecessary-condition -- We need access to app here which might not be available yet.
|
|
87
|
-
if (globalThis?.app?.isMobile) {
|
|
88
|
-
if (id === 'process' || id === 'node:process') {
|
|
89
|
-
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Fake process object is returned instead.`);
|
|
90
|
-
return globalThis.process;
|
|
91
|
-
}
|
|
92
|
-
} else {
|
|
93
|
-
const module = originalRequire?.(id);
|
|
94
|
-
if (module) {
|
|
95
|
-
return extractDefault(module);
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
console.debug(`The most likely you can safely ignore this error. Module not found: ${id}. Empty object is returned instead.`);
|
|
100
|
-
return {};
|
|
101
|
-
}
|
|
102
|
-
})();
|
|
103
|
-
|
|
104
|
-
"use strict";
|
|
105
|
-
var __defProp = Object.defineProperty;
|
|
106
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
107
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
108
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
109
|
-
var __export = (target, all) => {
|
|
110
|
-
for (var name in all)
|
|
111
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
112
|
-
};
|
|
113
|
-
var __copyProps = (to, from, except, desc) => {
|
|
114
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
115
|
-
for (let key of __getOwnPropNames(from))
|
|
116
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
117
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
118
|
-
}
|
|
119
|
-
return to;
|
|
120
|
-
};
|
|
121
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
122
|
-
var VaultEx_exports = {};
|
|
123
|
-
__export(VaultEx_exports, {
|
|
124
|
-
deleteEmptyFolder: () => deleteEmptyFolder,
|
|
125
|
-
deleteEmptyFolderHierarchy: () => deleteEmptyFolderHierarchy,
|
|
126
|
-
deleteSafe: () => deleteSafe
|
|
127
|
-
});
|
|
128
|
-
module.exports = __toCommonJS(VaultEx_exports);
|
|
129
|
-
var import_obsidian = require('obsidian');
|
|
130
|
-
var import_Error = require('../Error.cjs');
|
|
131
|
-
var import_FileSystem = require('./FileSystem.cjs');
|
|
132
|
-
var import_i18n = require('./i18n/i18n.cjs');
|
|
133
|
-
var import_MetadataCache = require('./MetadataCache.cjs');
|
|
134
|
-
var import_Vault = require('./Vault.cjs');
|
|
135
|
-
async function deleteEmptyFolder(app, pathOrFolder) {
|
|
136
|
-
const folder = (0, import_FileSystem.getFolderOrNull)(app, pathOrFolder);
|
|
137
|
-
if (!folder) {
|
|
138
|
-
return;
|
|
139
|
-
}
|
|
140
|
-
if (!await (0, import_Vault.isEmptyFolder)(app, folder)) {
|
|
141
|
-
return;
|
|
142
|
-
}
|
|
143
|
-
await deleteSafe(app, folder);
|
|
144
|
-
}
|
|
145
|
-
async function deleteEmptyFolderHierarchy(app, pathOrFolder) {
|
|
146
|
-
let folder = (0, import_FileSystem.getFolderOrNull)(app, pathOrFolder);
|
|
147
|
-
while (folder) {
|
|
148
|
-
if (!await (0, import_Vault.isEmptyFolder)(app, folder)) {
|
|
149
|
-
return;
|
|
150
|
-
}
|
|
151
|
-
const parent = folder.parent;
|
|
152
|
-
await deleteEmptyFolder(app, folder);
|
|
153
|
-
folder = parent;
|
|
154
|
-
}
|
|
155
|
-
}
|
|
156
|
-
async function deleteSafe(app, pathOrFile, deletedNotePath, shouldReportUsedAttachments, shouldDeleteEmptyFolders) {
|
|
157
|
-
const file = (0, import_FileSystem.getAbstractFileOrNull)(app, pathOrFile);
|
|
158
|
-
if (!file) {
|
|
159
|
-
return false;
|
|
160
|
-
}
|
|
161
|
-
let canDelete = (0, import_FileSystem.isFile)(file) || (shouldDeleteEmptyFolders ?? true);
|
|
162
|
-
if ((0, import_FileSystem.isFile)(file)) {
|
|
163
|
-
const backlinks = await (0, import_MetadataCache.getBacklinksForFileSafe)(app, file);
|
|
164
|
-
if (deletedNotePath) {
|
|
165
|
-
backlinks.clear(deletedNotePath);
|
|
166
|
-
}
|
|
167
|
-
if (backlinks.count() !== 0) {
|
|
168
|
-
if (shouldReportUsedAttachments) {
|
|
169
|
-
new import_obsidian.Notice((0, import_i18n.t)(($) => $.obsidianDevUtils.notices.attachmentIsStillUsed, { attachmentPath: file.path }));
|
|
170
|
-
}
|
|
171
|
-
canDelete = false;
|
|
172
|
-
}
|
|
173
|
-
} else if ((0, import_FileSystem.isFolder)(file)) {
|
|
174
|
-
const listedFiles = await (0, import_Vault.listSafe)(app, file);
|
|
175
|
-
for (const child of [...listedFiles.files, ...listedFiles.folders]) {
|
|
176
|
-
canDelete &&= await deleteSafe(app, child, deletedNotePath, shouldReportUsedAttachments);
|
|
177
|
-
}
|
|
178
|
-
canDelete &&= await (0, import_Vault.isEmptyFolder)(app, file);
|
|
179
|
-
}
|
|
180
|
-
if (canDelete) {
|
|
181
|
-
try {
|
|
182
|
-
await app.fileManager.trashFile(file);
|
|
183
|
-
} catch (e) {
|
|
184
|
-
if (await app.vault.exists(file.path)) {
|
|
185
|
-
(0, import_Error.printError)(new Error(`Failed to delete ${file.path}`, { cause: e }));
|
|
186
|
-
canDelete = false;
|
|
187
|
-
}
|
|
188
|
-
}
|
|
189
|
-
}
|
|
190
|
-
return canDelete;
|
|
191
|
-
}
|
|
192
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
193
|
-
0 && (module.exports = {
|
|
194
|
-
deleteEmptyFolder,
|
|
195
|
-
deleteEmptyFolderHierarchy,
|
|
196
|
-
deleteSafe
|
|
197
|
-
});
|
|
198
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL29ic2lkaWFuL1ZhdWx0RXgudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgYWRkaXRpb25hbCB1dGlsaXRpZXMgZm9yIHdvcmtpbmcgd2l0aCB0aGUgT2JzaWRpYW4gVmF1bHQuXG4gKlxuICogSXQgaGFzIHRvIGJlIGV4dHJhY3RlZCBmcm9tIGBWYXVsdGAgYmVjYXVzZSBvZiBjaXJjdWxhciBkZXBlbmRlbmNpZXMuXG4gKi9cblxuaW1wb3J0IHR5cGUgeyBBcHAgfSBmcm9tICdvYnNpZGlhbic7XG5cbmltcG9ydCB7IE5vdGljZSB9IGZyb20gJ29ic2lkaWFuJztcblxuaW1wb3J0IHR5cGUge1xuICBQYXRoT3JBYnN0cmFjdEZpbGUsXG4gIFBhdGhPckZvbGRlclxufSBmcm9tICcuL0ZpbGVTeXN0ZW0udHMnO1xuXG5pbXBvcnQgeyBwcmludEVycm9yIH0gZnJvbSAnLi4vRXJyb3IudHMnO1xuaW1wb3J0IHtcbiAgZ2V0QWJzdHJhY3RGaWxlT3JOdWxsLFxuICBnZXRGb2xkZXJPck51bGwsXG4gIGlzRmlsZSxcbiAgaXNGb2xkZXJcbn0gZnJvbSAnLi9GaWxlU3lzdGVtLnRzJztcbmltcG9ydCB7IHQgfSBmcm9tICcuL2kxOG4vaTE4bi50cyc7XG5pbXBvcnQgeyBnZXRCYWNrbGlua3NGb3JGaWxlU2FmZSB9IGZyb20gJy4vTWV0YWRhdGFDYWNoZS50cyc7XG5pbXBvcnQge1xuICBpc0VtcHR5Rm9sZGVyLFxuICBsaXN0U2FmZVxufSBmcm9tICcuL1ZhdWx0LnRzJztcblxuLyoqXG4gKiBEZWxldGVzIGFuIGVtcHR5IGZvbGRlci5cbiAqXG4gKiBAcGFyYW0gYXBwIC0gVGhlIGFwcGxpY2F0aW9uIGluc3RhbmNlLlxuICogQHBhcmFtIHBhdGhPckZvbGRlciAtIFRoZSBmb2xkZXIgdG8gZGVsZXRlLlxuICogQHJldHVybnMgQSB7QGxpbmsgUHJvbWlzZX0gdGhhdCByZXNvbHZlcyB3aGVuIHRoZSBmb2xkZXIgaXMgZGVsZXRlZC5cbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGRlbGV0ZUVtcHR5Rm9sZGVyKGFwcDogQXBwLCBwYXRoT3JGb2xkZXI6IG51bGwgfCBQYXRoT3JGb2xkZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgZm9sZGVyID0gZ2V0Rm9sZGVyT3JOdWxsKGFwcCwgcGF0aE9yRm9sZGVyKTtcbiAgaWYgKCFmb2xkZXIpIHtcbiAgICByZXR1cm47XG4gIH1cbiAgaWYgKCFhd2FpdCBpc0VtcHR5Rm9sZGVyKGFwcCwgZm9sZGVyKSkge1xuICAgIHJldHVybjtcbiAgfVxuICBhd2FpdCBkZWxldGVTYWZlKGFwcCwgZm9sZGVyKTtcbn1cblxuLyoqXG4gKiBSZW1vdmVzIGVtcHR5IGZvbGRlciBoaWVyYXJjaHkgc3RhcnRpbmcgZnJvbSB0aGUgZ2l2ZW4gZm9sZGVyLlxuICpcbiAqIEBwYXJhbSBhcHAgLSBUaGUgYXBwbGljYXRpb24gaW5zdGFuY2UuXG4gKiBAcGFyYW0gcGF0aE9yRm9sZGVyIC0gVGhlIGZvbGRlciB0byBzdGFydCByZW1vdmluZyBlbXB0eSBoaWVyYXJjaHkgZnJvbS5cbiAqIEByZXR1cm5zIEEge0BsaW5rIFByb21pc2V9IHRoYXQgcmVzb2x2ZXMgd2hlbiB0aGUgZW1wdHkgaGllcmFyY2h5IGlzIGRlbGV0ZWQuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBkZWxldGVFbXB0eUZvbGRlckhpZXJhcmNoeShhcHA6IEFwcCwgcGF0aE9yRm9sZGVyOiBudWxsIHwgUGF0aE9yRm9sZGVyKTogUHJvbWlzZTx2b2lkPiB7XG4gIGxldCBmb2xkZXIgPSBnZXRGb2xkZXJPck51bGwoYXBwLCBwYXRoT3JGb2xkZXIpO1xuXG4gIHdoaWxlIChmb2xkZXIpIHtcbiAgICBpZiAoIWF3YWl0IGlzRW1wdHlGb2xkZXIoYXBwLCBmb2xkZXIpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGNvbnN0IHBhcmVudCA9IGZvbGRlci5wYXJlbnQ7XG4gICAgYXdhaXQgZGVsZXRlRW1wdHlGb2xkZXIoYXBwLCBmb2xkZXIpO1xuICAgIGZvbGRlciA9IHBhcmVudDtcbiAgfVxufVxuXG4vKipcbiAqIERlbGV0ZXMgYWJzdHJhY3QgZmlsZSBzYWZlbHkgZnJvbSB0aGUgdmF1bHQuXG4gKlxuICogQHBhcmFtIGFwcCAtIFRoZSBPYnNpZGlhbiBhcHBsaWNhdGlvbiBpbnN0YW5jZS5cbiAqIEBwYXJhbSBwYXRoT3JGaWxlIC0gVGhlIHBhdGggb3IgYWJzdHJhY3QgZmlsZSB0byBkZWxldGUuXG4gKiBAcGFyYW0gZGVsZXRlZE5vdGVQYXRoIC0gT3B0aW9uYWwuIFRoZSBwYXRoIG9mIHRoZSBub3RlIHRoYXQgdHJpZ2dlcmVkIHRoZSByZW1vdmFsLlxuICogQHBhcmFtIHNob3VsZFJlcG9ydFVzZWRBdHRhY2htZW50cyAtIE9wdGlvbmFsLiBJZiBgdHJ1ZWAsIGEgbm90aWNlIHdpbGwgYmUgc2hvd24gZm9yIGVhY2ggYXR0YWNobWVudCB0aGF0IGlzIHN0aWxsIHVzZWQgYnkgb3RoZXIgbm90ZXMuXG4gKiBAcGFyYW0gc2hvdWxkRGVsZXRlRW1wdHlGb2xkZXJzIC0gT3B0aW9uYWwuIElmIGB0cnVlYCwgZW1wdHkgZm9sZGVycyB3aWxsIGJlIGRlbGV0ZWQuXG4gKiBAcmV0dXJucyBBIHtAbGluayBQcm9taXNlfSB0aGF0IHJlc29sdmVzIHRvIGEgYm9vbGVhbiBpbmRpY2F0aW5nIHdoZXRoZXIgdGhlIHJlbW92YWwgd2FzIHN1Y2Nlc3NmdWwuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBkZWxldGVTYWZlKFxuICBhcHA6IEFwcCxcbiAgcGF0aE9yRmlsZTogUGF0aE9yQWJzdHJhY3RGaWxlLFxuICBkZWxldGVkTm90ZVBhdGg/OiBzdHJpbmcsXG4gIHNob3VsZFJlcG9ydFVzZWRBdHRhY2htZW50cz86IGJvb2xlYW4sXG4gIHNob3VsZERlbGV0ZUVtcHR5Rm9sZGVycz86IGJvb2xlYW5cbik6IFByb21pc2U8Ym9vbGVhbj4ge1xuICBjb25zdCBmaWxlID0gZ2V0QWJzdHJhY3RGaWxlT3JOdWxsKGFwcCwgcGF0aE9yRmlsZSk7XG5cbiAgaWYgKCFmaWxlKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgbGV0IGNhbkRlbGV0ZSA9IGlzRmlsZShmaWxlKSB8fCAoc2hvdWxkRGVsZXRlRW1wdHlGb2xkZXJzID8/IHRydWUpO1xuXG4gIC8qIHY4IGlnbm9yZSBzdGFydCAtLSBUQWJzdHJhY3RGaWxlIGlzIGFsd2F5cyBURmlsZSBvciBURm9sZGVyIGluIE9ic2lkaWFuOyB0aGUgZmFsc2UgYnJhbmNoIG9mIGlzRmlsZSBsZWFkcyB0byBpc0ZvbGRlci4gKi9cbiAgaWYgKGlzRmlsZShmaWxlKSkge1xuICAgIC8qIHY4IGlnbm9yZSBzdG9wICovXG4gICAgY29uc3QgYmFja2xpbmtzID0gYXdhaXQgZ2V0QmFja2xpbmtzRm9yRmlsZVNhZmUoYXBwLCBmaWxlKTtcbiAgICBpZiAoZGVsZXRlZE5vdGVQYXRoKSB7XG4gICAgICBiYWNrbGlua3MuY2xlYXIoZGVsZXRlZE5vdGVQYXRoKTtcbiAgICB9XG4gICAgaWYgKGJhY2tsaW5rcy5jb3VudCgpICE9PSAwKSB7XG4gICAgICBpZiAoc2hvdWxkUmVwb3J0VXNlZEF0dGFjaG1lbnRzKSB7XG4gICAgICAgIG5ldyBOb3RpY2UodCgoJCkgPT4gJC5vYnNpZGlhbkRldlV0aWxzLm5vdGljZXMuYXR0YWNobWVudElzU3RpbGxVc2VkLCB7IGF0dGFjaG1lbnRQYXRoOiBmaWxlLnBhdGggfSkpO1xuICAgICAgfVxuICAgICAgY2FuRGVsZXRlID0gZmFsc2U7XG4gICAgfVxuICAgIC8qIHY4IGlnbm9yZSBzdGFydCAtLSBUQWJzdHJhY3RGaWxlIGlzIGFsd2F5cyBURmlsZSBvciBURm9sZGVyIGluIE9ic2lkaWFuOyBkZWZlbnNpdmUgZmFsbGJhY2suICovXG4gIH0gZWxzZSBpZiAoaXNGb2xkZXIoZmlsZSkpIHtcbiAgICAvKiB2OCBpZ25vcmUgc3RvcCAqL1xuICAgIGNvbnN0IGxpc3RlZEZpbGVzID0gYXdhaXQgbGlzdFNhZmUoYXBwLCBmaWxlKTtcbiAgICBmb3IgKGNvbnN0IGNoaWxkIG9mIFsuLi5saXN0ZWRGaWxlcy5maWxlcywgLi4ubGlzdGVkRmlsZXMuZm9sZGVyc10pIHtcbiAgICAgIGNhbkRlbGV0ZSAmJj0gYXdhaXQgZGVsZXRlU2FmZShhcHAsIGNoaWxkLCBkZWxldGVkTm90ZVBhdGgsIHNob3VsZFJlcG9ydFVzZWRBdHRhY2htZW50cyk7XG4gICAgfVxuXG4gICAgY2FuRGVsZXRlICYmPSBhd2FpdCBpc0VtcHR5Rm9sZGVyKGFwcCwgZmlsZSk7XG4gIH1cblxuICBpZiAoY2FuRGVsZXRlKSB7XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGFwcC5maWxlTWFuYWdlci50cmFzaEZpbGUoZmlsZSk7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgaWYgKGF3YWl0IGFwcC52YXVsdC5leGlzdHMoZmlsZS5wYXRoKSkge1xuICAgICAgICBwcmludEVycm9yKG5ldyBFcnJvcihgRmFpbGVkIHRvIGRlbGV0ZSAke2ZpbGUucGF0aH1gLCB7IGNhdXNlOiBlIH0pKTtcbiAgICAgICAgY2FuRGVsZXRlID0gZmFsc2U7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIGNhbkRlbGV0ZTtcbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQVVBLHNCQUF1QjtBQU92QixtQkFBMkI7QUFDM0Isd0JBS087QUFDUCxrQkFBa0I7QUFDbEIsMkJBQXdDO0FBQ3hDLG1CQUdPO0FBU1AsZUFBc0Isa0JBQWtCLEtBQVUsY0FBa0Q7QUFDbEcsUUFBTSxhQUFTLG1DQUFnQixLQUFLLFlBQVk7QUFDaEQsTUFBSSxDQUFDLFFBQVE7QUFDWDtBQUFBLEVBQ0Y7QUFDQSxNQUFJLENBQUMsVUFBTSw0QkFBYyxLQUFLLE1BQU0sR0FBRztBQUNyQztBQUFBLEVBQ0Y7QUFDQSxRQUFNLFdBQVcsS0FBSyxNQUFNO0FBQzlCO0FBU0EsZUFBc0IsMkJBQTJCLEtBQVUsY0FBa0Q7QUFDM0csTUFBSSxhQUFTLG1DQUFnQixLQUFLLFlBQVk7QUFFOUMsU0FBTyxRQUFRO0FBQ2IsUUFBSSxDQUFDLFVBQU0sNEJBQWMsS0FBSyxNQUFNLEdBQUc7QUFDckM7QUFBQSxJQUNGO0FBQ0EsVUFBTSxTQUFTLE9BQU87QUFDdEIsVUFBTSxrQkFBa0IsS0FBSyxNQUFNO0FBQ25DLGFBQVM7QUFBQSxFQUNYO0FBQ0Y7QUFZQSxlQUFzQixXQUNwQixLQUNBLFlBQ0EsaUJBQ0EsNkJBQ0EsMEJBQ2tCO0FBQ2xCLFFBQU0sV0FBTyx5Q0FBc0IsS0FBSyxVQUFVO0FBRWxELE1BQUksQ0FBQyxNQUFNO0FBQ1QsV0FBTztBQUFBLEVBQ1Q7QUFFQSxNQUFJLGdCQUFZLDBCQUFPLElBQUksTUFBTSw0QkFBNEI7QUFHN0QsVUFBSSwwQkFBTyxJQUFJLEdBQUc7QUFFaEIsVUFBTSxZQUFZLFVBQU0sOENBQXdCLEtBQUssSUFBSTtBQUN6RCxRQUFJLGlCQUFpQjtBQUNuQixnQkFBVSxNQUFNLGVBQWU7QUFBQSxJQUNqQztBQUNBLFFBQUksVUFBVSxNQUFNLE1BQU0sR0FBRztBQUMzQixVQUFJLDZCQUE2QjtBQUMvQixZQUFJLDJCQUFPLGVBQUUsQ0FBQyxNQUFNLEVBQUUsaUJBQWlCLFFBQVEsdUJBQXVCLEVBQUUsZ0JBQWdCLEtBQUssS0FBSyxDQUFDLENBQUM7QUFBQSxNQUN0RztBQUNBLGtCQUFZO0FBQUEsSUFDZDtBQUFBLEVBRUYsZUFBVyw0QkFBUyxJQUFJLEdBQUc7QUFFekIsVUFBTSxjQUFjLFVBQU0sdUJBQVMsS0FBSyxJQUFJO0FBQzVDLGVBQVcsU0FBUyxDQUFDLEdBQUcsWUFBWSxPQUFPLEdBQUcsWUFBWSxPQUFPLEdBQUc7QUFDbEUsb0JBQWMsTUFBTSxXQUFXLEtBQUssT0FBTyxpQkFBaUIsMkJBQTJCO0FBQUEsSUFDekY7QUFFQSxrQkFBYyxVQUFNLDRCQUFjLEtBQUssSUFBSTtBQUFBLEVBQzdDO0FBRUEsTUFBSSxXQUFXO0FBQ2IsUUFBSTtBQUNGLFlBQU0sSUFBSSxZQUFZLFVBQVUsSUFBSTtBQUFBLElBQ3RDLFNBQVMsR0FBRztBQUNWLFVBQUksTUFBTSxJQUFJLE1BQU0sT0FBTyxLQUFLLElBQUksR0FBRztBQUNyQyxxQ0FBVyxJQUFJLE1BQU0sb0JBQW9CLEtBQUssSUFBSSxJQUFJLEVBQUUsT0FBTyxFQUFFLENBQUMsQ0FBQztBQUNuRSxvQkFBWTtBQUFBLE1BQ2Q7QUFBQSxJQUNGO0FBQUEsRUFDRjtBQUVBLFNBQU87QUFDVDsiLAogICJuYW1lcyI6IFtdCn0K
|