obsidian-dev-utils 44.2.4 → 45.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.
Files changed (57) hide show
  1. package/CHANGELOG.md +8 -0
  2. package/dist/lib/cjs/Async.cjs +53 -32
  3. package/dist/lib/cjs/Async.d.cts +92 -10
  4. package/dist/lib/cjs/Library.cjs +1 -1
  5. package/dist/lib/cjs/obsidian/AsyncWithNotice.cjs +167 -0
  6. package/dist/lib/cjs/obsidian/AsyncWithNotice.d.cts +74 -0
  7. package/dist/lib/cjs/obsidian/Backlink.cjs +2 -3
  8. package/dist/lib/cjs/obsidian/Callout.cjs +8 -2
  9. package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.cjs +56 -41
  10. package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.d.cts +45 -12
  11. package/dist/lib/cjs/obsidian/Commands/CommandBase.cjs +13 -1
  12. package/dist/lib/cjs/obsidian/Commands/CommandBase.d.cts +7 -0
  13. package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.cjs +3 -9
  14. package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.d.cts +7 -7
  15. package/dist/lib/cjs/obsidian/Commands/FileCommandBase.cjs +123 -26
  16. package/dist/lib/cjs/obsidian/Commands/FileCommandBase.d.cts +100 -9
  17. package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.cjs +115 -26
  18. package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.d.cts +94 -10
  19. package/dist/lib/cjs/obsidian/FileSystem.cjs +44 -1
  20. package/dist/lib/cjs/obsidian/FileSystem.d.cts +81 -33
  21. package/dist/lib/cjs/obsidian/MetadataCache.cjs +45 -41
  22. package/dist/lib/cjs/obsidian/Queue.cjs +34 -17
  23. package/dist/lib/cjs/obsidian/Queue.d.cts +62 -12
  24. package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +37 -27
  25. package/dist/lib/cjs/obsidian/Vault.cjs +45 -66
  26. package/dist/lib/cjs/obsidian/Vault.d.cts +0 -8
  27. package/dist/lib/cjs/obsidian/index.cjs +4 -1
  28. package/dist/lib/cjs/obsidian/index.d.cts +1 -0
  29. package/dist/lib/esm/Async.d.mts +92 -10
  30. package/dist/lib/esm/Async.mjs +53 -32
  31. package/dist/lib/esm/Library.mjs +1 -1
  32. package/dist/lib/esm/obsidian/AsyncWithNotice.d.mts +74 -0
  33. package/dist/lib/esm/obsidian/AsyncWithNotice.mjs +63 -0
  34. package/dist/lib/esm/obsidian/Backlink.mjs +4 -4
  35. package/dist/lib/esm/obsidian/Callout.mjs +8 -2
  36. package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.d.mts +45 -12
  37. package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.mjs +54 -40
  38. package/dist/lib/esm/obsidian/Commands/CommandBase.d.mts +7 -0
  39. package/dist/lib/esm/obsidian/Commands/CommandBase.mjs +13 -1
  40. package/dist/lib/esm/obsidian/Commands/EditorCommandBase.d.mts +7 -7
  41. package/dist/lib/esm/obsidian/Commands/EditorCommandBase.mjs +3 -9
  42. package/dist/lib/esm/obsidian/Commands/FileCommandBase.d.mts +100 -9
  43. package/dist/lib/esm/obsidian/Commands/FileCommandBase.mjs +127 -26
  44. package/dist/lib/esm/obsidian/Commands/FolderCommandBase.d.mts +94 -10
  45. package/dist/lib/esm/obsidian/Commands/FolderCommandBase.mjs +119 -26
  46. package/dist/lib/esm/obsidian/FileSystem.d.mts +81 -33
  47. package/dist/lib/esm/obsidian/FileSystem.mjs +38 -1
  48. package/dist/lib/esm/obsidian/MetadataCache.mjs +46 -41
  49. package/dist/lib/esm/obsidian/Queue.d.mts +62 -12
  50. package/dist/lib/esm/obsidian/Queue.mjs +35 -19
  51. package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +37 -27
  52. package/dist/lib/esm/obsidian/Vault.d.mts +0 -8
  53. package/dist/lib/esm/obsidian/Vault.mjs +46 -70
  54. package/dist/lib/esm/obsidian/index.d.mts +1 -0
  55. package/dist/lib/esm/obsidian/index.mjs +3 -1
  56. package/obsidian/AsyncWithNotice/package.json +6 -0
  57. package/package.json +7 -7
@@ -124,6 +124,12 @@ __export(FileSystem_exports, {
124
124
  BASE_FILE_EXTENSION: () => BASE_FILE_EXTENSION,
125
125
  CANVAS_FILE_EXTENSION: () => CANVAS_FILE_EXTENSION,
126
126
  MARKDOWN_FILE_EXTENSION: () => MARKDOWN_FILE_EXTENSION,
127
+ asArrayOfFiles: () => asArrayOfFiles,
128
+ asArrayOfFolders: () => asArrayOfFolders,
129
+ asFile: () => asFile,
130
+ asFileOrNull: () => asFileOrNull,
131
+ asFolder: () => asFolder,
132
+ asFolderOrNull: () => asFolderOrNull,
127
133
  checkExtension: () => checkExtension,
128
134
  getAbstractFile: () => getAbstractFile,
129
135
  getAbstractFileOrNull: () => getAbstractFileOrNull,
@@ -147,11 +153,42 @@ __export(FileSystem_exports, {
147
153
  module.exports = __toCommonJS(FileSystem_exports);
148
154
  var import_obsidian = require('obsidian');
149
155
  var import_implementations = require('obsidian-typings/implementations');
156
+ var import_Error = require('../Error.cjs');
150
157
  var import_Path = require('../Path.cjs');
151
158
  var import_String = require('../String.cjs');
152
159
  const BASE_FILE_EXTENSION = "base";
153
160
  const CANVAS_FILE_EXTENSION = "canvas";
154
161
  const MARKDOWN_FILE_EXTENSION = "md";
162
+ function asArrayOfFiles(abstractFiles) {
163
+ return abstractFiles.map((abstractFile) => asFile(abstractFile));
164
+ }
165
+ function asArrayOfFolders(abstractFiles) {
166
+ return abstractFiles.map((abstractFile) => asFolder(abstractFile));
167
+ }
168
+ function asFile(abstractFile) {
169
+ return asFileOrNull(abstractFile) ?? (0, import_Error.throwExpression)(new Error("Abstract file is not a file"));
170
+ }
171
+ function asFileOrNull(abstractFile) {
172
+ if (abstractFile === null) {
173
+ return null;
174
+ }
175
+ if (abstractFile instanceof import_obsidian.TFile) {
176
+ return abstractFile;
177
+ }
178
+ throw new Error("Abstract file is not a file");
179
+ }
180
+ function asFolder(abstractFile) {
181
+ return asFolderOrNull(abstractFile) ?? (0, import_Error.throwExpression)(new Error("Abstract file is not a folder"));
182
+ }
183
+ function asFolderOrNull(abstractFile) {
184
+ if (abstractFile === null) {
185
+ return null;
186
+ }
187
+ if (abstractFile instanceof import_obsidian.TFolder) {
188
+ return abstractFile;
189
+ }
190
+ throw new Error("Abstract file is not a folder");
191
+ }
155
192
  function checkExtension(app, pathOrFile, extension) {
156
193
  if (isFile(pathOrFile)) {
157
194
  return pathOrFile.extension === extension;
@@ -307,6 +344,12 @@ function getResolvedPath(path) {
307
344
  BASE_FILE_EXTENSION,
308
345
  CANVAS_FILE_EXTENSION,
309
346
  MARKDOWN_FILE_EXTENSION,
347
+ asArrayOfFiles,
348
+ asArrayOfFolders,
349
+ asFile,
350
+ asFileOrNull,
351
+ asFolder,
352
+ asFolderOrNull,
310
353
  checkExtension,
311
354
  getAbstractFile,
312
355
  getAbstractFileOrNull,
@@ -327,4 +370,4 @@ function getResolvedPath(path) {
327
370
  isNote,
328
371
  trimMarkdownExtension
329
372
  });
330
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileSystem.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with TAbstractFile, TFile, and TFolder instances in Obsidian.\n */\n\nimport type { App } from 'obsidian';\n\nimport {\n  normalizePath,\n  TAbstractFile,\n  TFile,\n  TFolder,\n  Vault\n} from 'obsidian';\nimport {\n  createTFileInstance,\n  createTFolderInstance,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport {\n  extname,\n  resolve\n} from '../Path.ts';\nimport { trimEnd } from '../String.ts';\n\n/**\n * A file extension for `base` files.\n */\nexport const BASE_FILE_EXTENSION = 'base';\n\n/**\n * A file extension for `canvas` files.\n */\nexport const CANVAS_FILE_EXTENSION = 'canvas';\n\n/**\n * A file extension for `markdown` files.\n */\nexport const MARKDOWN_FILE_EXTENSION = 'md';\n\n/**\n * A path or an instance of {@link TAbstractFile}.\n */\nexport type PathOrAbstractFile = string | TAbstractFile;\n\n/**\n * A path or a {@link TFile}.\n */\nexport type PathOrFile = string | TFile;\n\n/**\n * A path or a {@link TFolder}.\n */\nexport type PathOrFolder = string | TFolder;\n\n/**\n * Checks if the given path or file has the specified extension.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or abstract file to check.\n * @param extension - The extension to compare against.\n * @returns Returns `true` if the path or file has the specified extension, `false` otherwise.\n */\nexport function checkExtension(app: App, pathOrFile: null | PathOrAbstractFile, extension: string): boolean {\n  if (isFile(pathOrFile)) {\n    return pathOrFile.extension === extension;\n  }\n\n  if (typeof pathOrFile === 'string') {\n    const file = getFileOrNull(app, pathOrFile);\n    if (file) {\n      return file.extension === extension;\n    }\n\n    return extname(pathOrFile).slice(1) === extension;\n  }\n\n  return false;\n}\n\n/**\n * Retrieves the TAbstractFile object for the given path or abstract file.\n *\n * @param app - The App instance.\n * @param pathOrFile - The path or abstract file to retrieve the TAbstractFile for.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The TAbstractFile object.\n * @throws Error if the abstract file is not found.\n */\nexport function getAbstractFile(app: App, pathOrFile: PathOrAbstractFile, isCaseInsensitive?: boolean): TAbstractFile {\n  const file = getAbstractFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (!file) {\n    throw new Error(`Abstract file not found: ${pathOrFile as string}`);\n  }\n\n  return file;\n}\n\n/**\n * Retrieves an instance of TAbstractFile or null based on the provided path or abstract file.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or abstract file to retrieve.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The instance of TAbstractFile if found, otherwise null.\n */\nexport function getAbstractFileOrNull(app: App, pathOrFile: null | PathOrAbstractFile, isCaseInsensitive?: boolean): null | TAbstractFile {\n  if (pathOrFile === null) {\n    return null;\n  }\n\n  if (isAbstractFile(pathOrFile)) {\n    return app.vault.fileMap[pathOrFile.path] ?? pathOrFile;\n  }\n\n  const file = getFileInternal(app, pathOrFile, isCaseInsensitive);\n\n  if (file) {\n    return file;\n  }\n\n  const resolvedPath = getResolvedPath(pathOrFile);\n\n  if (resolvedPath === pathOrFile) {\n    return null;\n  }\n\n  return getFileInternal(app, resolvedPath, isCaseInsensitive);\n}\n\n/**\n * Retrieves a TFile object based on the provided path or file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to retrieve the TFile object for.\n * @param shouldIncludeNonExisting - Whether to include a non-existing file.\n *  If `true`, a new TFile object is created for the provided path.\n *  If `false`, an error is thrown if the file is not found.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The TFile object corresponding to the provided path or file.\n * @throws Error if the file is not found.\n */\nexport function getFile(app: App, pathOrFile: PathOrFile, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFile {\n  let file = getFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (!file) {\n    if (shouldIncludeNonExisting) {\n      file = createTFileInstance(app, pathOrFile as string);\n    } else {\n      throw new Error(`File not found: ${pathOrFile as string}`);\n    }\n  }\n\n  return file;\n}\n\n/**\n * Retrieves a TFile object based on the provided path or file.\n * If the provided argument is already a TFile object, it is returned as is.\n * Otherwise, the function uses the app's vault to retrieve the TFile object by its path.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or TFile object.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The TFile object if found, otherwise null.\n */\nexport function getFileOrNull(app: App, pathOrFile: null | PathOrFile, isCaseInsensitive?: boolean): null | TFile {\n  const file = getAbstractFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (isFile(file)) {\n    return file;\n  }\n  return null;\n}\n\n/**\n * Retrieves a TFolder object based on the provided app and pathOrFolder.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFolder - The path or folder identifier.\n * @param shouldIncludeNonExisting - Whether to allow the folder to not exist.\n *  If `true`, a new TFolder object is created for the provided path.\n *  If `false`, an error is thrown if the folder is not found.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The retrieved TFolder object.\n * @throws If the folder is not found.\n */\nexport function getFolder(app: App, pathOrFolder: PathOrFolder, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFolder {\n  let folder = getFolderOrNull(app, pathOrFolder, isCaseInsensitive);\n  if (!folder) {\n    if (shouldIncludeNonExisting) {\n      folder = createTFolderInstance(app, pathOrFolder as string);\n    } else {\n      throw new Error(`Folder not found: ${pathOrFolder as string}`);\n    }\n  }\n\n  return folder;\n}\n\n/**\n * Retrieves a TFolder object or null based on the provided path or folder.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to retrieve the TFolder from.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The TFolder object if found, otherwise null.\n */\nexport function getFolderOrNull(app: App, pathOrFolder: null | PathOrFolder, isCaseInsensitive?: boolean): null | TFolder {\n  const folder = getAbstractFileOrNull(app, pathOrFolder, isCaseInsensitive);\n  if (isFolder(folder)) {\n    return folder;\n  }\n  return null;\n}\n\n/**\n * Retrieves an array of TFile objects representing the markdown files within a specified folder or path.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFolder - The path or folder to retrieve the markdown files from.\n * @param isRecursive - Optional. Specifies whether to recursively search for markdown files within subfolders. Default is false.\n * @returns An array of TFile objects representing the markdown files.\n */\nexport function getMarkdownFiles(app: App, pathOrFolder: PathOrFolder, isRecursive?: boolean): TFile[] {\n  const folder = getFolder(app, pathOrFolder);\n\n  let markdownFiles: TFile[] = [];\n\n  if (isRecursive) {\n    Vault.recurseChildren(folder, (abstractFile) => {\n      if (isMarkdownFile(app, abstractFile) && abstractFile instanceof TFile) {\n        markdownFiles.push(abstractFile);\n      }\n    });\n  } else {\n    markdownFiles = folder.children.filter((file) => isMarkdownFile(app, file)) as TFile[];\n  }\n\n  markdownFiles = markdownFiles.sort((a, b) => a.path.localeCompare(b.path));\n  return markdownFiles;\n}\n\n/**\n * Retrieves the TFile object for the given path or creates a new one if it does not exist.\n *\n * @param app - The Obsidian App instance.\n * @param path - The path of the file to retrieve or create.\n * @returns The TFile object representing the file\n */\nexport async function getOrCreateFile(app: App, path: string): Promise<TFile> {\n  const file = getFileOrNull(app, path);\n  if (file) {\n    return file;\n  }\n\n  const folderPath = parentFolderPath(path);\n  await getOrCreateFolder(app, folderPath);\n\n  return await app.vault.create(path, '');\n}\n\n/**\n * Retrieves the TFolder object for the given path or creates a new one if it does not exist.\n *\n * @param app - The Obsidian App instance.\n * @param path - The path of the folder to retrieve or create.\n * @returns The TFolder object representing the folder.\n */\nexport async function getOrCreateFolder(app: App, path: string): Promise<TFolder> {\n  const folder = getFolderOrNull(app, path);\n  if (folder) {\n    return folder;\n  }\n\n  return await app.vault.createFolder(path);\n}\n\n/**\n * Returns the path of the given `pathOrFile`.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or abstract file.\n * @returns The path of the `pathOrFile`.\n */\nexport function getPath(app: App, pathOrFile: PathOrAbstractFile): string {\n  if (isAbstractFile(pathOrFile)) {\n    return pathOrFile.path;\n  }\n\n  const file = getAbstractFileOrNull(app, pathOrFile);\n  if (file) {\n    return file.path;\n  }\n\n  return getResolvedPath(pathOrFile);\n}\n\n/**\n * Checks if the given file is an instance of TAbstractFile.\n *\n * @param file - The file to check.\n * @returns A boolean indicating whether the file is an instance of TAbstractFile.\n */\nexport function isAbstractFile(file: unknown): file is TAbstractFile {\n  return file instanceof TAbstractFile;\n}\n\n/**\n *   Checks if the given file is a base file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a base file.\n */\nexport function isBaseFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, BASE_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is a canvas file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a canvas file or not.\n */\nexport function isCanvasFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, CANVAS_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is an instance of TFile.\n *\n * @param file - The file to check.\n * @returns A boolean indicating whether the file is an instance of TFile.\n */\nexport function isFile(file: unknown): file is TFile {\n  return file instanceof TFile;\n}\n\n/**\n * Checks if the given file is a folder.\n *\n * @param file - The file to check.\n * @returns `true` if the file is a folder, `false` otherwise.\n */\nexport function isFolder(file: unknown): file is TFolder {\n  return file instanceof TFolder;\n}\n\n/**\n * Checks if the given file is a Markdown file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a Markdown file.\n */\nexport function isMarkdownFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, MARKDOWN_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is a note.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a note.\n */\nexport function isNote(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return isMarkdownFile(app, pathOrFile) || isCanvasFile(app, pathOrFile) || isBaseFile(app, pathOrFile);\n}\n\n/**\n * Trims the markdown extension from the file path if the file is a markdown file.\n * If the file is not a markdown file, the original file path is returned.\n *\n * @param app - The Obsidian App instance.\n * @param file - The file to trim the markdown extension from.\n * @returns The file path with the markdown extension trimmed.\n */\nexport function trimMarkdownExtension(app: App, file: TAbstractFile): string {\n  if (!isMarkdownFile(app, file)) {\n    return file.path;\n  }\n\n  return trimEnd(file.path, `.${MARKDOWN_FILE_EXTENSION}`);\n}\n\nfunction getFileInternal(app: App, path: string, isCaseInsensitive?: boolean): null | TAbstractFile {\n  if (isCaseInsensitive) {\n    return app.vault.getAbstractFileByPathInsensitive(path);\n  }\n\n  return app.vault.getAbstractFileByPath(path) as null | TFile;\n}\n\nfunction getResolvedPath(path: string): string {\n  return normalizePath(resolve('/', path));\n}\n"],
  "mappings": "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",
  "names": []
}

373
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileSystem.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with {@link TAbstractFile}, {@link TFile}, and {@link TFolder} instances in Obsidian.\n */\n\nimport type { App } from 'obsidian';\n\nimport {\n  normalizePath,\n  TAbstractFile,\n  TFile,\n  TFolder,\n  Vault\n} from 'obsidian';\nimport {\n  createTFileInstance,\n  createTFolderInstance,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport { throwExpression } from '../Error.ts';\nimport {\n  extname,\n  resolve\n} from '../Path.ts';\nimport { trimEnd } from '../String.ts';\n\n/**\n * A file extension for `base` files.\n */\nexport const BASE_FILE_EXTENSION = 'base';\n\n/**\n * A file extension for `canvas` files.\n */\nexport const CANVAS_FILE_EXTENSION = 'canvas';\n\n/**\n * A file extension for `markdown` files.\n */\nexport const MARKDOWN_FILE_EXTENSION = 'md';\n\n/**\n * A path or an abstract file.\n */\nexport type PathOrAbstractFile = string | TAbstractFile;\n\n/**\n * A path or a file.\n */\nexport type PathOrFile = string | TFile;\n\n/**\n * A path or a folder.\n */\nexport type PathOrFolder = string | TFolder;\n\n/**\n * Converts an array of abstract files to an array of files.\n *\n * @param abstractFiles - The abstract files to convert.\n * @returns The array of files.\n * @throws Error if any of the abstract files are not files.\n */\nexport function asArrayOfFiles(abstractFiles: TAbstractFile[]): TFile[] {\n  return abstractFiles.map((abstractFile) => asFile(abstractFile));\n}\n\n/**\n * Converts an array of abstract files to an array of folders.\n *\n * @param abstractFiles - The abstract files to convert.\n * @returns The array of folders.\n * @throws Error if any of the abstract files are not folders.\n */\nexport function asArrayOfFolders(abstractFiles: TAbstractFile[]): TFolder[] {\n  return abstractFiles.map((abstractFile) => asFolder(abstractFile));\n}\n\n/**\n * Converts an abstract file to a file.\n *\n * @param abstractFile - The abstract file to convert.\n * @returns The file.\n * @throws Error if the abstract file is not a file.\n */\nexport function asFile(abstractFile: null | TAbstractFile): TFile {\n  return asFileOrNull(abstractFile) ?? throwExpression(new Error('Abstract file is not a file'));\n}\n\n/**\n * Converts an abstract file to a file or `null`.\n *\n * @param abstractFile - The abstract file to convert.\n * @returns The file or `null`.\n * @throws Error if the abstract file is not a file.\n */\nexport function asFileOrNull(abstractFile: null | TAbstractFile): null | TFile {\n  if (abstractFile === null) {\n    return null;\n  }\n  if (abstractFile instanceof TFile) {\n    return abstractFile;\n  }\n  throw new Error('Abstract file is not a file');\n}\n\n/**\n * Converts an abstract file to a folder.\n *\n * @param abstractFile - The abstract file to convert.\n * @returns The folder.\n * @throws Error if the abstract file is not a folder.\n */\nexport function asFolder(abstractFile: null | TAbstractFile): TFolder {\n  return asFolderOrNull(abstractFile) ?? throwExpression(new Error('Abstract file is not a folder'));\n}\n\n/**\n * Converts an abstract file to a folder or `null`.\n *\n * @param abstractFile - The abstract file to convert.\n * @returns The folder or `null`.\n * @throws Error if the abstract file is not a folder.\n */\nexport function asFolderOrNull(abstractFile: null | TAbstractFile): null | TFolder {\n  if (abstractFile === null) {\n    return null;\n  }\n  if (abstractFile instanceof TFolder) {\n    return abstractFile;\n  }\n  throw new Error('Abstract file is not a folder');\n}\n\n/**\n * Checks if the given path or file has the specified extension.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or abstract file to check.\n * @param extension - The extension to compare against.\n * @returns Returns `true` if the path or file has the specified extension, `false` otherwise.\n */\nexport function checkExtension(app: App, pathOrFile: null | PathOrAbstractFile, extension: string): boolean {\n  if (isFile(pathOrFile)) {\n    return pathOrFile.extension === extension;\n  }\n\n  if (typeof pathOrFile === 'string') {\n    const file = getFileOrNull(app, pathOrFile);\n    if (file) {\n      return file.extension === extension;\n    }\n\n    return extname(pathOrFile).slice(1) === extension;\n  }\n\n  return false;\n}\n\n/**\n * Retrieves the TAbstractFile object for the given path or abstract file.\n *\n * @param app - The App instance.\n * @param pathOrFile - The path or abstract file to retrieve the abstract file for.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The abstract file.\n * @throws Error if the abstract file is not found.\n */\nexport function getAbstractFile(app: App, pathOrFile: PathOrAbstractFile, isCaseInsensitive?: boolean): TAbstractFile {\n  const file = getAbstractFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (!file) {\n    throw new Error(`Abstract file not found: ${pathOrFile as string}`);\n  }\n\n  return file;\n}\n\n/**\n * Retrieves an abstract file or `null` based on the provided path or abstract file.\n *\n * @param app - The application instance.\n * @param pathOrFile - The path or abstract file to retrieve.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The abstract file if found, otherwise `null`.\n */\nexport function getAbstractFileOrNull(app: App, pathOrFile: null | PathOrAbstractFile, isCaseInsensitive?: boolean): null | TAbstractFile {\n  if (pathOrFile === null) {\n    return null;\n  }\n\n  if (isAbstractFile(pathOrFile)) {\n    return app.vault.fileMap[pathOrFile.path] ?? pathOrFile;\n  }\n\n  const file = getFileInternal(app, pathOrFile, isCaseInsensitive);\n\n  if (file) {\n    return file;\n  }\n\n  const resolvedPath = getResolvedPath(pathOrFile);\n\n  if (resolvedPath === pathOrFile) {\n    return null;\n  }\n\n  return getFileInternal(app, resolvedPath, isCaseInsensitive);\n}\n\n/**\n * Retrieves a file based on the provided path or file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to retrieve the file for.\n * @param shouldIncludeNonExisting - Whether to include a non-existing file.\n *  If `true`, a new file is created for the provided path.\n *  If `false`, an error is thrown if the file is not found.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The file corresponding to the provided path or file.\n * @throws Error if the file is not found.\n */\nexport function getFile(app: App, pathOrFile: PathOrFile, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFile {\n  let file = getFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (!file) {\n    if (shouldIncludeNonExisting) {\n      file = createTFileInstance(app, pathOrFile as string);\n    } else {\n      throw new Error(`File not found: ${pathOrFile as string}`);\n    }\n  }\n\n  return file;\n}\n\n/**\n * Retrieves a file or `null` based on the provided path or file.\n * If the provided argument is already a file, it is returned as is.\n * Otherwise, the function uses the app's vault to retrieve the file by its path.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The file if found, otherwise `null`.\n */\nexport function getFileOrNull(app: App, pathOrFile: null | PathOrFile, isCaseInsensitive?: boolean): null | TFile {\n  const file = getAbstractFileOrNull(app, pathOrFile, isCaseInsensitive);\n  if (isFile(file)) {\n    return file;\n  }\n  return null;\n}\n\n/**\n * Retrieves a folder based on the provided app and pathOrFolder.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFolder - The path or folder identifier.\n * @param shouldIncludeNonExisting - Whether to allow the folder to not exist.\n *  If `true`, a new folder is created for the provided path.\n *  If `false`, an error is thrown if the folder is not found.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The retrieved folder.\n * @throws If the folder is not found.\n */\nexport function getFolder(app: App, pathOrFolder: PathOrFolder, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFolder {\n  let folder = getFolderOrNull(app, pathOrFolder, isCaseInsensitive);\n  if (!folder) {\n    if (shouldIncludeNonExisting) {\n      folder = createTFolderInstance(app, pathOrFolder as string);\n    } else {\n      throw new Error(`Folder not found: ${pathOrFolder as string}`);\n    }\n  }\n\n  return folder;\n}\n\n/**\n * Retrieves a folder or `null` based on the provided path or folder.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFolder - The path or folder to retrieve the folder from.\n * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.\n * @returns The folder if found, otherwise `null`.\n */\nexport function getFolderOrNull(app: App, pathOrFolder: null | PathOrFolder, isCaseInsensitive?: boolean): null | TFolder {\n  const folder = getAbstractFileOrNull(app, pathOrFolder, isCaseInsensitive);\n  if (isFolder(folder)) {\n    return folder;\n  }\n  return null;\n}\n\n/**\n * Retrieves an array of files representing the markdown files within a specified folder or path.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFolder - The path or folder to retrieve the markdown files from.\n * @param isRecursive - Optional. Specifies whether to recursively search for markdown files within subfolders. Default is false.\n * @returns An array of files representing the markdown files.\n */\nexport function getMarkdownFiles(app: App, pathOrFolder: PathOrFolder, isRecursive?: boolean): TFile[] {\n  const folder = getFolder(app, pathOrFolder);\n\n  let markdownFiles: TFile[] = [];\n\n  if (isRecursive) {\n    Vault.recurseChildren(folder, (abstractFile) => {\n      if (isMarkdownFile(app, abstractFile) && abstractFile instanceof TFile) {\n        markdownFiles.push(abstractFile);\n      }\n    });\n  } else {\n    markdownFiles = folder.children.filter((file) => isMarkdownFile(app, file)) as TFile[];\n  }\n\n  markdownFiles = markdownFiles.sort((a, b) => a.path.localeCompare(b.path));\n  return markdownFiles;\n}\n\n/**\n * Retrieves the file for the given path or creates a new one if it does not exist.\n *\n * @param app - The Obsidian App instance.\n * @param path - The path of the file to retrieve or create.\n * @returns The file representing the file\n */\nexport async function getOrCreateFile(app: App, path: string): Promise<TFile> {\n  const file = getFileOrNull(app, path);\n  if (file) {\n    return file;\n  }\n\n  const folderPath = parentFolderPath(path);\n  await getOrCreateFolder(app, folderPath);\n\n  return await app.vault.create(path, '');\n}\n\n/**\n * Retrieves the folder for the given path or creates a new one if it does not exist.\n *\n * @param app - The Obsidian App instance.\n * @param path - The path of the folder to retrieve or create.\n * @returns The folder representing the folder.\n */\nexport async function getOrCreateFolder(app: App, path: string): Promise<TFolder> {\n  const folder = getFolderOrNull(app, path);\n  if (folder) {\n    return folder;\n  }\n\n  return await app.vault.createFolder(path);\n}\n\n/**\n * Returns the path of the given `pathOrFile`.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or abstract file.\n * @returns The path of the `pathOrFile`.\n */\nexport function getPath(app: App, pathOrFile: PathOrAbstractFile): string {\n  if (isAbstractFile(pathOrFile)) {\n    return pathOrFile.path;\n  }\n\n  const file = getAbstractFileOrNull(app, pathOrFile);\n  if (file) {\n    return file.path;\n  }\n\n  return getResolvedPath(pathOrFile);\n}\n\n/**\n * Checks if the given file is an instance of abstract file.\n *\n * @param file - The file to check.\n * @returns A boolean indicating whether the file is an instance of abstract file.\n */\nexport function isAbstractFile(file: unknown): file is TAbstractFile {\n  return file instanceof TAbstractFile;\n}\n\n/**\n *   Checks if the given file is a base file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a base file.\n */\nexport function isBaseFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, BASE_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is a canvas file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a canvas file or not.\n */\nexport function isCanvasFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, CANVAS_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is an instance of file.\n *\n * @param file - The file to check.\n * @returns A boolean indicating whether the file is an instance of file.\n */\nexport function isFile(file: unknown): file is TFile {\n  return file instanceof TFile;\n}\n\n/**\n * Checks if the given file is a folder.\n *\n * @param file - The file to check.\n * @returns `true` if the file is a folder, `false` otherwise.\n */\nexport function isFolder(file: unknown): file is TFolder {\n  return file instanceof TFolder;\n}\n\n/**\n * Checks if the given file is a Markdown file.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a Markdown file.\n */\nexport function isMarkdownFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return checkExtension(app, pathOrFile, MARKDOWN_FILE_EXTENSION);\n}\n\n/**\n * Checks if the given file is a note.\n *\n * @param app - The Obsidian App instance.\n * @param pathOrFile - The path or file to check.\n * @returns A boolean indicating whether the file is a note.\n */\nexport function isNote(app: App, pathOrFile: null | PathOrAbstractFile): boolean {\n  return isMarkdownFile(app, pathOrFile) || isCanvasFile(app, pathOrFile) || isBaseFile(app, pathOrFile);\n}\n\n/**\n * Trims the markdown extension from the file path if the file is a markdown file.\n * If the file is not a markdown file, the original file path is returned.\n *\n * @param app - The Obsidian App instance.\n * @param file - The file to trim the markdown extension from.\n * @returns The file path with the markdown extension trimmed.\n */\nexport function trimMarkdownExtension(app: App, file: TAbstractFile): string {\n  if (!isMarkdownFile(app, file)) {\n    return file.path;\n  }\n\n  return trimEnd(file.path, `.${MARKDOWN_FILE_EXTENSION}`);\n}\n\nfunction getFileInternal(app: App, path: string, isCaseInsensitive?: boolean): null | TAbstractFile {\n  if (isCaseInsensitive) {\n    return app.vault.getAbstractFileByPathInsensitive(path);\n  }\n\n  return app.vault.getAbstractFileByPath(path) as null | TFile;\n}\n\nfunction getResolvedPath(path: string): string {\n  return normalizePath(resolve('/', path));\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * @packageDocumentation
3
3
  *
4
- * This module provides utility functions for working with TAbstractFile, TFile, and TFolder instances in Obsidian.
4
+ * This module provides utility functions for working with {@link TAbstractFile}, {@link TFile}, and {@link TFolder} instances in Obsidian.
5
5
  */
6
6
  import type { App } from 'obsidian';
7
7
  import { TAbstractFile, TFile, TFolder } from 'obsidian';
@@ -18,17 +18,65 @@ export declare const CANVAS_FILE_EXTENSION = "canvas";
18
18
  */
19
19
  export declare const MARKDOWN_FILE_EXTENSION = "md";
20
20
  /**
21
- * A path or an instance of {@link TAbstractFile}.
21
+ * A path or an abstract file.
22
22
  */
23
23
  export type PathOrAbstractFile = string | TAbstractFile;
24
24
  /**
25
- * A path or a {@link TFile}.
25
+ * A path or a file.
26
26
  */
27
27
  export type PathOrFile = string | TFile;
28
28
  /**
29
- * A path or a {@link TFolder}.
29
+ * A path or a folder.
30
30
  */
31
31
  export type PathOrFolder = string | TFolder;
32
+ /**
33
+ * Converts an array of abstract files to an array of files.
34
+ *
35
+ * @param abstractFiles - The abstract files to convert.
36
+ * @returns The array of files.
37
+ * @throws Error if any of the abstract files are not files.
38
+ */
39
+ export declare function asArrayOfFiles(abstractFiles: TAbstractFile[]): TFile[];
40
+ /**
41
+ * Converts an array of abstract files to an array of folders.
42
+ *
43
+ * @param abstractFiles - The abstract files to convert.
44
+ * @returns The array of folders.
45
+ * @throws Error if any of the abstract files are not folders.
46
+ */
47
+ export declare function asArrayOfFolders(abstractFiles: TAbstractFile[]): TFolder[];
48
+ /**
49
+ * Converts an abstract file to a file.
50
+ *
51
+ * @param abstractFile - The abstract file to convert.
52
+ * @returns The file.
53
+ * @throws Error if the abstract file is not a file.
54
+ */
55
+ export declare function asFile(abstractFile: null | TAbstractFile): TFile;
56
+ /**
57
+ * Converts an abstract file to a file or `null`.
58
+ *
59
+ * @param abstractFile - The abstract file to convert.
60
+ * @returns The file or `null`.
61
+ * @throws Error if the abstract file is not a file.
62
+ */
63
+ export declare function asFileOrNull(abstractFile: null | TAbstractFile): null | TFile;
64
+ /**
65
+ * Converts an abstract file to a folder.
66
+ *
67
+ * @param abstractFile - The abstract file to convert.
68
+ * @returns The folder.
69
+ * @throws Error if the abstract file is not a folder.
70
+ */
71
+ export declare function asFolder(abstractFile: null | TAbstractFile): TFolder;
72
+ /**
73
+ * Converts an abstract file to a folder or `null`.
74
+ *
75
+ * @param abstractFile - The abstract file to convert.
76
+ * @returns The folder or `null`.
77
+ * @throws Error if the abstract file is not a folder.
78
+ */
79
+ export declare function asFolderOrNull(abstractFile: null | TAbstractFile): null | TFolder;
32
80
  /**
33
81
  * Checks if the given path or file has the specified extension.
34
82
  *
@@ -42,90 +90,90 @@ export declare function checkExtension(app: App, pathOrFile: null | PathOrAbstra
42
90
  * Retrieves the TAbstractFile object for the given path or abstract file.
43
91
  *
44
92
  * @param app - The App instance.
45
- * @param pathOrFile - The path or abstract file to retrieve the TAbstractFile for.
93
+ * @param pathOrFile - The path or abstract file to retrieve the abstract file for.
46
94
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
47
- * @returns The TAbstractFile object.
95
+ * @returns The abstract file.
48
96
  * @throws Error if the abstract file is not found.
49
97
  */
50
98
  export declare function getAbstractFile(app: App, pathOrFile: PathOrAbstractFile, isCaseInsensitive?: boolean): TAbstractFile;
51
99
  /**
52
- * Retrieves an instance of TAbstractFile or null based on the provided path or abstract file.
100
+ * Retrieves an abstract file or `null` based on the provided path or abstract file.
53
101
  *
54
102
  * @param app - The application instance.
55
103
  * @param pathOrFile - The path or abstract file to retrieve.
56
104
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
57
- * @returns The instance of TAbstractFile if found, otherwise null.
105
+ * @returns The abstract file if found, otherwise `null`.
58
106
  */
59
107
  export declare function getAbstractFileOrNull(app: App, pathOrFile: null | PathOrAbstractFile, isCaseInsensitive?: boolean): null | TAbstractFile;
60
108
  /**
61
- * Retrieves a TFile object based on the provided path or file.
109
+ * Retrieves a file based on the provided path or file.
62
110
  *
63
111
  * @param app - The Obsidian App instance.
64
- * @param pathOrFile - The path or file to retrieve the TFile object for.
112
+ * @param pathOrFile - The path or file to retrieve the file for.
65
113
  * @param shouldIncludeNonExisting - Whether to include a non-existing file.
66
- * If `true`, a new TFile object is created for the provided path.
114
+ * If `true`, a new file is created for the provided path.
67
115
  * If `false`, an error is thrown if the file is not found.
68
116
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
69
- * @returns The TFile object corresponding to the provided path or file.
117
+ * @returns The file corresponding to the provided path or file.
70
118
  * @throws Error if the file is not found.
71
119
  */
72
120
  export declare function getFile(app: App, pathOrFile: PathOrFile, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFile;
73
121
  /**
74
- * Retrieves a TFile object based on the provided path or file.
75
- * If the provided argument is already a TFile object, it is returned as is.
76
- * Otherwise, the function uses the app's vault to retrieve the TFile object by its path.
122
+ * Retrieves a file or `null` based on the provided path or file.
123
+ * If the provided argument is already a file, it is returned as is.
124
+ * Otherwise, the function uses the app's vault to retrieve the file by its path.
77
125
  *
78
126
  * @param app - The Obsidian App instance.
79
- * @param pathOrFile - The path or TFile object.
127
+ * @param pathOrFile - The path or file.
80
128
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
81
- * @returns The TFile object if found, otherwise null.
129
+ * @returns The file if found, otherwise `null`.
82
130
  */
83
131
  export declare function getFileOrNull(app: App, pathOrFile: null | PathOrFile, isCaseInsensitive?: boolean): null | TFile;
84
132
  /**
85
- * Retrieves a TFolder object based on the provided app and pathOrFolder.
133
+ * Retrieves a folder based on the provided app and pathOrFolder.
86
134
  *
87
135
  * @param app - The Obsidian app instance.
88
136
  * @param pathOrFolder - The path or folder identifier.
89
137
  * @param shouldIncludeNonExisting - Whether to allow the folder to not exist.
90
- * If `true`, a new TFolder object is created for the provided path.
138
+ * If `true`, a new folder is created for the provided path.
91
139
  * If `false`, an error is thrown if the folder is not found.
92
140
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
93
- * @returns The retrieved TFolder object.
141
+ * @returns The retrieved folder.
94
142
  * @throws If the folder is not found.
95
143
  */
96
144
  export declare function getFolder(app: App, pathOrFolder: PathOrFolder, shouldIncludeNonExisting?: boolean, isCaseInsensitive?: boolean): TFolder;
97
145
  /**
98
- * Retrieves a TFolder object or null based on the provided path or folder.
146
+ * Retrieves a folder or `null` based on the provided path or folder.
99
147
  *
100
148
  * @param app - The Obsidian application instance.
101
- * @param pathOrFolder - The path or folder to retrieve the TFolder from.
149
+ * @param pathOrFolder - The path or folder to retrieve the folder from.
102
150
  * @param isCaseInsensitive - Specifies whether to perform a case-insensitive search. Default is `false`.
103
- * @returns The TFolder object if found, otherwise null.
151
+ * @returns The folder if found, otherwise `null`.
104
152
  */
105
153
  export declare function getFolderOrNull(app: App, pathOrFolder: null | PathOrFolder, isCaseInsensitive?: boolean): null | TFolder;
106
154
  /**
107
- * Retrieves an array of TFile objects representing the markdown files within a specified folder or path.
155
+ * Retrieves an array of files representing the markdown files within a specified folder or path.
108
156
  *
109
157
  * @param app - The Obsidian App instance.
110
158
  * @param pathOrFolder - The path or folder to retrieve the markdown files from.
111
159
  * @param isRecursive - Optional. Specifies whether to recursively search for markdown files within subfolders. Default is false.
112
- * @returns An array of TFile objects representing the markdown files.
160
+ * @returns An array of files representing the markdown files.
113
161
  */
114
162
  export declare function getMarkdownFiles(app: App, pathOrFolder: PathOrFolder, isRecursive?: boolean): TFile[];
115
163
  /**
116
- * Retrieves the TFile object for the given path or creates a new one if it does not exist.
164
+ * Retrieves the file for the given path or creates a new one if it does not exist.
117
165
  *
118
166
  * @param app - The Obsidian App instance.
119
167
  * @param path - The path of the file to retrieve or create.
120
- * @returns The TFile object representing the file
168
+ * @returns The file representing the file
121
169
  */
122
170
  export declare function getOrCreateFile(app: App, path: string): Promise<TFile>;
123
171
  /**
124
- * Retrieves the TFolder object for the given path or creates a new one if it does not exist.
172
+ * Retrieves the folder for the given path or creates a new one if it does not exist.
125
173
  *
126
174
  * @param app - The Obsidian App instance.
127
175
  * @param path - The path of the folder to retrieve or create.
128
- * @returns The TFolder object representing the folder.
176
+ * @returns The folder representing the folder.
129
177
  */
130
178
  export declare function getOrCreateFolder(app: App, path: string): Promise<TFolder>;
131
179
  /**
@@ -137,10 +185,10 @@ export declare function getOrCreateFolder(app: App, path: string): Promise<TFold
137
185
  */
138
186
  export declare function getPath(app: App, pathOrFile: PathOrAbstractFile): string;
139
187
  /**
140
- * Checks if the given file is an instance of TAbstractFile.
188
+ * Checks if the given file is an instance of abstract file.
141
189
  *
142
190
  * @param file - The file to check.
143
- * @returns A boolean indicating whether the file is an instance of TAbstractFile.
191
+ * @returns A boolean indicating whether the file is an instance of abstract file.
144
192
  */
145
193
  export declare function isAbstractFile(file: unknown): file is TAbstractFile;
146
194
  /**
@@ -160,10 +208,10 @@ export declare function isBaseFile(app: App, pathOrFile: null | PathOrAbstractFi
160
208
  */
161
209
  export declare function isCanvasFile(app: App, pathOrFile: null | PathOrAbstractFile): boolean;
162
210
  /**
163
- * Checks if the given file is an instance of TFile.
211
+ * Checks if the given file is an instance of file.
164
212
  *
165
213
  * @param file - The file to check.
166
- * @returns A boolean indicating whether the file is an instance of TFile.
214
+ * @returns A boolean indicating whether the file is an instance of file.
167
215
  */
168
216
  export declare function isFile(file: unknown): file is TFile;
169
217
  /**
@@ -137,9 +137,9 @@ __export(MetadataCache_exports, {
137
137
  });
138
138
  module.exports = __toCommonJS(MetadataCache_exports);
139
139
  var import_implementations = require('obsidian-typings/implementations');
140
- var import_Async = require('../Async.cjs');
141
140
  var import_ObjectUtils = require('../ObjectUtils.cjs');
142
141
  var import_App = require('./App.cjs');
142
+ var import_AsyncWithNotice = require('./AsyncWithNotice.cjs');
143
143
  var import_FileSystem = require('./FileSystem.cjs');
144
144
  var import_Frontmatter = require('./Frontmatter.cjs');
145
145
  var import_FrontmatterLinkCacheWithOffsets = require('./FrontmatterLinkCacheWithOffsets.cjs');
@@ -192,51 +192,55 @@ async function getBacklinksForFileSafe(app, pathOrFile, retryOptions = {}) {
192
192
  return safeOverload(pathOrFile);
193
193
  }
194
194
  let backlinks = new import_implementations.CustomArrayDictImpl();
195
- await (0, import_Async.retryWithTimeout)(async (abortSignal) => {
196
- abortSignal.throwIfAborted();
197
- const file = (0, import_FileSystem.getFile)(app, pathOrFile);
198
- await ensureMetadataCacheReady(app);
199
- abortSignal.throwIfAborted();
200
- backlinks = getBacklinksForFileOrPath(app, file);
201
- for (const notePath of backlinks.keys()) {
195
+ await (0, import_AsyncWithNotice.retryWithTimeoutNotice)({
196
+ async operationFn(abortSignal) {
202
197
  abortSignal.throwIfAborted();
203
- const note = (0, import_FileSystem.getFileOrNull)(app, notePath);
204
- if (!note) {
205
- return false;
206
- }
207
- await (0, import_Vault.saveNote)(app, note);
208
- abortSignal.throwIfAborted();
209
- const content = await (0, import_Vault.readSafe)(app, note);
198
+ const file = (0, import_FileSystem.getFile)(app, pathOrFile);
199
+ await ensureMetadataCacheReady(app);
210
200
  abortSignal.throwIfAborted();
211
- if (!content) {
212
- return false;
213
- }
214
- const frontmatter = (0, import_Frontmatter.parseFrontmatter)(content);
215
- const links = backlinks.get(notePath);
216
- if (!links) {
217
- return false;
218
- }
219
- for (const link of links) {
220
- let actualLink;
221
- if ((0, import_implementations.isReferenceCache)(link)) {
222
- actualLink = content.slice(link.position.start.offset, link.position.end.offset);
223
- } else if ((0, import_implementations.isFrontmatterLinkCache)(link)) {
224
- const propertyValue = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, link.key);
225
- if (typeof propertyValue !== "string") {
226
- return false;
227
- }
228
- const linkWithOffsets = (0, import_FrontmatterLinkCacheWithOffsets.toFrontmatterLinkCacheWithOffsets)(link);
229
- actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
230
- } else {
231
- return true;
201
+ backlinks = getBacklinksForFileOrPath(app, file);
202
+ for (const notePath of backlinks.keys()) {
203
+ abortSignal.throwIfAborted();
204
+ const note = (0, import_FileSystem.getFileOrNull)(app, notePath);
205
+ if (!note) {
206
+ return false;
207
+ }
208
+ await (0, import_Vault.saveNote)(app, note);
209
+ abortSignal.throwIfAborted();
210
+ const content = await (0, import_Vault.readSafe)(app, note);
211
+ abortSignal.throwIfAborted();
212
+ if (!content) {
213
+ return false;
232
214
  }
233
- if (actualLink !== link.original) {
215
+ const frontmatter = (0, import_Frontmatter.parseFrontmatter)(content);
216
+ const links = backlinks.get(notePath);
217
+ if (!links) {
234
218
  return false;
235
219
  }
220
+ for (const link of links) {
221
+ let actualLink;
222
+ if ((0, import_implementations.isReferenceCache)(link)) {
223
+ actualLink = content.slice(link.position.start.offset, link.position.end.offset);
224
+ } else if ((0, import_implementations.isFrontmatterLinkCache)(link)) {
225
+ const propertyValue = (0, import_ObjectUtils.getNestedPropertyValue)(frontmatter, link.key);
226
+ if (typeof propertyValue !== "string") {
227
+ return false;
228
+ }
229
+ const linkWithOffsets = (0, import_FrontmatterLinkCacheWithOffsets.toFrontmatterLinkCacheWithOffsets)(link);
230
+ actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);
231
+ } else {
232
+ return true;
233
+ }
234
+ if (actualLink !== link.original) {
235
+ return false;
236
+ }
237
+ }
236
238
  }
237
- }
238
- return true;
239
- }, retryOptions);
239
+ return true;
240
+ },
241
+ operationName: `Get backlinks for ${(0, import_FileSystem.getPath)(app, pathOrFile)}`,
242
+ retryOptions
243
+ });
240
244
  return backlinks;
241
245
  }
242
246
  async function getCacheSafe(app, fileOrPath) {
@@ -360,4 +364,4 @@ function getRegisteredFilesCounts(app) {
360
364
  unregisterFileCacheForNonExistingFile,
361
365
  unregisterFiles
362
366
  });
363
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MetadataCache.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the metadata cache in Obsidian.\n */\n\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TAbstractFile\n} from 'obsidian';\nimport type { CustomArrayDict } from 'obsidian-typings';\n\nimport {\n  CustomArrayDictImpl,\n  isFrontmatterLinkCache,\n  isReferenceCache,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\n\nimport { retryWithTimeout } from '../Async.ts';\nimport { getNestedPropertyValue } from '../ObjectUtils.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  isFile\n} from './FileSystem.ts';\nimport { parseFrontmatter } from './Frontmatter.ts';\nimport {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport { sortReferences } from './Reference.ts';\nimport {\n  readSafe,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Wrapper for the getBacklinksForFile method that provides a safe overload.\n */\nexport interface GetBacklinksForFileSafeWrapper {\n  /**\n   * Retrieves the backlinks for a file safely.\n   *\n   * @param pathOrFile - The path or file object.\n   * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n   */\n  safe(pathOrFile: PathOrFile): Promise<CustomArrayDict<Reference>>;\n}\n\n/**\n * Ensures that the metadata cache is ready for all files.\n *\n * @param app - The Obsidian app instance.\n * @returns A {@link Promise} that resolves when the metadata cache is ready.\n */\nexport async function ensureMetadataCacheReady(app: App): Promise<void> {\n  await new Promise((resolve) => {\n    app.metadataCache.onCleanCache(resolve);\n  });\n}\n\n/**\n * Retrieves all links from the provided cache.\n *\n * @param cache - The cached metadata.\n * @returns An array of reference caches representing the links.\n */\nexport function getAllLinks(cache: CachedMetadata): Reference[] {\n  let links: Reference[] = [];\n\n  if (cache.links) {\n    links.push(...cache.links);\n  }\n\n  if (cache.embeds) {\n    links.push(...cache.embeds);\n  }\n\n  if (cache.frontmatterLinks) {\n    links.push(...cache.frontmatterLinks);\n  }\n\n  sortReferences(links);\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  links = links.filter((link, index) => {\n    if (index === 0) {\n      return true;\n    }\n\n    const previousLink = links[index - 1];\n    if (!previousLink) {\n      return true;\n    }\n\n    if (isReferenceCache(link) && isReferenceCache(previousLink)) {\n      return link.position.start.offset !== previousLink.position.start.offset;\n    }\n\n    if (isFrontmatterLinkCache(link) && isFrontmatterLinkCache(previousLink)) {\n      const linkStartOffset = isFrontmatterLinkCacheWithOffsets(link) ? link.startOffset : 0;\n      const previousLinkStartOffset = isFrontmatterLinkCacheWithOffsets(previousLink) ? previousLink.startOffset : 0;\n      return link.key !== previousLink.key || isFrontmatterLinkCacheWithOffsets(link) !== isFrontmatterLinkCacheWithOffsets(previousLink)\n        || linkStartOffset !== previousLinkStartOffset;\n    }\n\n    return true;\n  });\n\n  return links;\n}\n\n/**\n * Retrieves the backlinks for a file or path.\n * NOTE: The file may be non-existent.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @returns The backlinks for the file.\n */\nexport function getBacklinksForFileOrPath(app: App, pathOrFile: PathOrFile): CustomArrayDict<Reference> {\n  const file = getFile(app, pathOrFile, true);\n  return tempRegisterFilesAndRun(app, [file], () => app.metadataCache.getBacklinksForFile(file));\n}\n\n/**\n * Retrieves the backlinks for a file safely.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @param retryOptions - Optional retry options.\n * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n */\nexport async function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile, retryOptions: RetryOptions = {}): Promise<CustomArrayDict<Reference>> {\n  const safeOverload = (app.metadataCache.getBacklinksForFile as Partial<GetBacklinksForFileSafeWrapper>).safe;\n  if (safeOverload) {\n    return safeOverload(pathOrFile);\n  }\n  let backlinks: CustomArrayDict<Reference> = new CustomArrayDictImpl<Reference>();\n  await retryWithTimeout(async (abortSignal) => {\n    abortSignal.throwIfAborted();\n    const file = getFile(app, pathOrFile);\n    await ensureMetadataCacheReady(app);\n    abortSignal.throwIfAborted();\n    backlinks = getBacklinksForFileOrPath(app, file);\n    for (const notePath of backlinks.keys()) {\n      abortSignal.throwIfAborted();\n      const note = getFileOrNull(app, notePath);\n      if (!note) {\n        return false;\n      }\n\n      await saveNote(app, note);\n      abortSignal.throwIfAborted();\n\n      const content = await readSafe(app, note);\n      abortSignal.throwIfAborted();\n      if (!content) {\n        return false;\n      }\n      const frontmatter = parseFrontmatter(content);\n      const links = backlinks.get(notePath);\n      if (!links) {\n        return false;\n      }\n\n      for (const link of links) {\n        let actualLink: string;\n        if (isReferenceCache(link)) {\n          actualLink = content.slice(link.position.start.offset, link.position.end.offset);\n        } else if (isFrontmatterLinkCache(link)) {\n          const propertyValue = getNestedPropertyValue(frontmatter, link.key);\n          if (typeof propertyValue !== 'string') {\n            return false;\n          }\n\n          const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);\n          actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);\n        } else {\n          return true;\n        }\n        if (actualLink !== link.original) {\n          return false;\n        }\n      }\n    }\n\n    return true;\n  }, retryOptions);\n\n  return backlinks;\n}\n\n/**\n * Retrieves the cached metadata for a given file or path.\n *\n * @param app - The Obsidian app instance.\n * @param fileOrPath - The file or path to retrieve the metadata for.\n * @returns The cached metadata for the file, or null if it doesn't exist.\n */\nexport async function getCacheSafe(app: App, fileOrPath: PathOrFile): Promise<CachedMetadata | null> {\n  const file = getFileOrNull(app, fileOrPath);\n\n  try {\n    if (!file) {\n      return null;\n    }\n\n    if (file.deleted) {\n      return app.metadataCache.getFileCache(file);\n    }\n\n    await saveNote(app, file);\n\n    const fileCacheEntry = app.metadataCache.fileCache[file.path];\n    const isUpToDate = fileCacheEntry\n      && fileCacheEntry.mtime === file.stat.mtime\n      && fileCacheEntry.size === file.stat.size\n      && app.metadataCache.metadataCache[fileCacheEntry.hash];\n    if (!isUpToDate) {\n      await app.metadataCache.computeFileMetadataAsync(file);\n      await ensureMetadataCacheReady(app);\n    }\n    return app.metadataCache.getFileCache(file);\n  } catch (error) {\n    if (!file || file.deleted) {\n      return null;\n    }\n\n    throw error;\n  }\n}\n\n/**\n * Retrieves the front matter from the metadata cache safely.\n *\n * @typeParam CustomFrontmatter - The type of custom front matter.\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to retrieve the front matter from.\n * @returns The combined front matter.\n */\nexport async function getFrontmatterSafe<CustomFrontmatter = unknown>(app: App, pathOrFile: PathOrFile): Promise<CombinedFrontmatter<CustomFrontmatter>> {\n  const cache = await getCacheSafe(app, pathOrFile);\n  return (cache?.frontmatter ?? {}) as CombinedFrontmatter<CustomFrontmatter>;\n}\n\n/**\n * Parses the metadata for a given string.\n *\n * @param app - The Obsidian app instance.\n * @param str - The string to parse the metadata for.\n * @returns The parsed metadata.\n */\nexport async function parseMetadata(app: App, str: string): Promise<CachedMetadata> {\n  const encoder = new TextEncoder();\n  const buffer = encoder.encode(str).buffer;\n  return await app.metadataCache.computeMetadataAsync(buffer) ?? {};\n}\n\n/**\n * Registers the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to register the file cache for.\n * @param cache - The file cache to register.\n */\nexport function registerFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile, cache: CachedMetadata): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n\n  app.metadataCache.fileCache[file.path] = {\n    hash: file.path,\n    mtime: 0,\n    size: 0\n  };\n\n  app.metadataCache.metadataCache[file.path] = cache;\n}\n\n/**\n * Registers files in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to register.\n */\nexport function registerFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 0;\n      count++;\n      registeredFilesCounts.set(file.path, count);\n\n      app.vault.fileMap[file.path] = file;\n\n      if (isFile(file)) {\n        app.metadataCache.uniqueFileLookup.add(file.name.toLowerCase(), file);\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\n/**\n * Temporarily registers files and runs a function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport function tempRegisterFilesAndRun<T>(app: App, files: TAbstractFile[], fn: () => T): T {\n  try {\n    registerFiles(app, files);\n    return fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Temporarily registers files and runs an async function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport async function tempRegisterFilesAndRunAsync<T>(app: App, files: TAbstractFile[], fn: () => Promise<T>): Promise<T> {\n  try {\n    registerFiles(app, files);\n    return await fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Unregisters the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to unregister the file cache for.\n */\nexport function unregisterFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.fileCache[file.path];\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.metadataCache[file.path];\n}\n\n/**\n * Unregisters files from the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to unregister.\n */\nexport function unregisterFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 1;\n      count--;\n      registeredFilesCounts.set(file.path, count);\n      if (count === 0) {\n        registeredFilesCounts.delete(file.path);\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete app.vault.fileMap[file.path];\n\n        if (isFile(file)) {\n          app.metadataCache.uniqueFileLookup.remove(file.name.toLowerCase(), file);\n        }\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\nfunction getRegisteredFilesCounts(app: App): Map<string, number> {\n  return getObsidianDevUtilsState(app, 'registeredFilesCounts', new Map<string, number>()).value;\n}\n"],
  "mappings": "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",
  "names": []
}

367
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/MetadataCache.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utility functions for working with the metadata cache in Obsidian.\n */\n\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TAbstractFile\n} from 'obsidian';\nimport type { CustomArrayDict } from 'obsidian-typings';\n\nimport {\n  CustomArrayDictImpl,\n  isFrontmatterLinkCache,\n  isReferenceCache,\n  parentFolderPath\n} from 'obsidian-typings/implementations';\n\nimport type { RetryOptions } from '../Async.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\n\nimport { getNestedPropertyValue } from '../ObjectUtils.ts';\nimport { getObsidianDevUtilsState } from './App.ts';\nimport { retryWithTimeoutNotice } from './AsyncWithNotice.ts';\nimport {\n  getFile,\n  getFileOrNull,\n  getFolder,\n  getPath,\n  isFile\n} from './FileSystem.ts';\nimport { parseFrontmatter } from './Frontmatter.ts';\nimport {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport { sortReferences } from './Reference.ts';\nimport {\n  readSafe,\n  saveNote\n} from './Vault.ts';\n\n/**\n * Wrapper for the getBacklinksForFile method that provides a safe overload.\n */\nexport interface GetBacklinksForFileSafeWrapper {\n  /**\n   * Retrieves the backlinks for a file safely.\n   *\n   * @param pathOrFile - The path or file object.\n   * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n   */\n  safe(pathOrFile: PathOrFile): Promise<CustomArrayDict<Reference>>;\n}\n\n/**\n * Ensures that the metadata cache is ready for all files.\n *\n * @param app - The Obsidian app instance.\n * @returns A {@link Promise} that resolves when the metadata cache is ready.\n */\nexport async function ensureMetadataCacheReady(app: App): Promise<void> {\n  await new Promise((resolve) => {\n    app.metadataCache.onCleanCache(resolve);\n  });\n}\n\n/**\n * Retrieves all links from the provided cache.\n *\n * @param cache - The cached metadata.\n * @returns An array of reference caches representing the links.\n */\nexport function getAllLinks(cache: CachedMetadata): Reference[] {\n  let links: Reference[] = [];\n\n  if (cache.links) {\n    links.push(...cache.links);\n  }\n\n  if (cache.embeds) {\n    links.push(...cache.embeds);\n  }\n\n  if (cache.frontmatterLinks) {\n    links.push(...cache.frontmatterLinks);\n  }\n\n  sortReferences(links);\n\n  // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n  links = links.filter((link, index) => {\n    if (index === 0) {\n      return true;\n    }\n\n    const previousLink = links[index - 1];\n    if (!previousLink) {\n      return true;\n    }\n\n    if (isReferenceCache(link) && isReferenceCache(previousLink)) {\n      return link.position.start.offset !== previousLink.position.start.offset;\n    }\n\n    if (isFrontmatterLinkCache(link) && isFrontmatterLinkCache(previousLink)) {\n      const linkStartOffset = isFrontmatterLinkCacheWithOffsets(link) ? link.startOffset : 0;\n      const previousLinkStartOffset = isFrontmatterLinkCacheWithOffsets(previousLink) ? previousLink.startOffset : 0;\n      return link.key !== previousLink.key || isFrontmatterLinkCacheWithOffsets(link) !== isFrontmatterLinkCacheWithOffsets(previousLink)\n        || linkStartOffset !== previousLinkStartOffset;\n    }\n\n    return true;\n  });\n\n  return links;\n}\n\n/**\n * Retrieves the backlinks for a file or path.\n * NOTE: The file may be non-existent.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @returns The backlinks for the file.\n */\nexport function getBacklinksForFileOrPath(app: App, pathOrFile: PathOrFile): CustomArrayDict<Reference> {\n  const file = getFile(app, pathOrFile, true);\n  return tempRegisterFilesAndRun(app, [file], () => app.metadataCache.getBacklinksForFile(file));\n}\n\n/**\n * Retrieves the backlinks for a file safely.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file object.\n * @param retryOptions - Optional retry options.\n * @returns A {@link Promise} that resolves to an array dictionary of backlinks.\n */\nexport async function getBacklinksForFileSafe(app: App, pathOrFile: PathOrFile, retryOptions: RetryOptions = {}): Promise<CustomArrayDict<Reference>> {\n  const safeOverload = (app.metadataCache.getBacklinksForFile as Partial<GetBacklinksForFileSafeWrapper>).safe;\n  if (safeOverload) {\n    return safeOverload(pathOrFile);\n  }\n  let backlinks: CustomArrayDict<Reference> = new CustomArrayDictImpl<Reference>();\n  await retryWithTimeoutNotice({\n    async operationFn(abortSignal) {\n      abortSignal.throwIfAborted();\n      const file = getFile(app, pathOrFile);\n      await ensureMetadataCacheReady(app);\n      abortSignal.throwIfAborted();\n      backlinks = getBacklinksForFileOrPath(app, file);\n      for (const notePath of backlinks.keys()) {\n        abortSignal.throwIfAborted();\n        const note = getFileOrNull(app, notePath);\n        if (!note) {\n          return false;\n        }\n\n        await saveNote(app, note);\n        abortSignal.throwIfAborted();\n\n        const content = await readSafe(app, note);\n        abortSignal.throwIfAborted();\n        if (!content) {\n          return false;\n        }\n        const frontmatter = parseFrontmatter(content);\n        const links = backlinks.get(notePath);\n        if (!links) {\n          return false;\n        }\n\n        for (const link of links) {\n          let actualLink: string;\n          if (isReferenceCache(link)) {\n            actualLink = content.slice(link.position.start.offset, link.position.end.offset);\n          } else if (isFrontmatterLinkCache(link)) {\n            const propertyValue = getNestedPropertyValue(frontmatter, link.key);\n            if (typeof propertyValue !== 'string') {\n              return false;\n            }\n\n            const linkWithOffsets = toFrontmatterLinkCacheWithOffsets(link);\n            actualLink = propertyValue.slice(linkWithOffsets.startOffset, linkWithOffsets.endOffset);\n          } else {\n            return true;\n          }\n          if (actualLink !== link.original) {\n            return false;\n          }\n        }\n      }\n\n      return true;\n    },\n    operationName: `Get backlinks for ${getPath(app, pathOrFile)}`,\n    retryOptions\n  });\n\n  return backlinks;\n}\n\n/**\n * Retrieves the cached metadata for a given file or path.\n *\n * @param app - The Obsidian app instance.\n * @param fileOrPath - The file or path to retrieve the metadata for.\n * @returns The cached metadata for the file, or null if it doesn't exist.\n */\nexport async function getCacheSafe(app: App, fileOrPath: PathOrFile): Promise<CachedMetadata | null> {\n  const file = getFileOrNull(app, fileOrPath);\n\n  try {\n    if (!file) {\n      return null;\n    }\n\n    if (file.deleted) {\n      return app.metadataCache.getFileCache(file);\n    }\n\n    await saveNote(app, file);\n\n    const fileCacheEntry = app.metadataCache.fileCache[file.path];\n    const isUpToDate = fileCacheEntry\n      && fileCacheEntry.mtime === file.stat.mtime\n      && fileCacheEntry.size === file.stat.size\n      && app.metadataCache.metadataCache[fileCacheEntry.hash];\n    if (!isUpToDate) {\n      await app.metadataCache.computeFileMetadataAsync(file);\n      await ensureMetadataCacheReady(app);\n    }\n    return app.metadataCache.getFileCache(file);\n  } catch (error) {\n    if (!file || file.deleted) {\n      return null;\n    }\n\n    throw error;\n  }\n}\n\n/**\n * Retrieves the front matter from the metadata cache safely.\n *\n * @typeParam CustomFrontmatter - The type of custom front matter.\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to retrieve the front matter from.\n * @returns The combined front matter.\n */\nexport async function getFrontmatterSafe<CustomFrontmatter = unknown>(app: App, pathOrFile: PathOrFile): Promise<CombinedFrontmatter<CustomFrontmatter>> {\n  const cache = await getCacheSafe(app, pathOrFile);\n  return (cache?.frontmatter ?? {}) as CombinedFrontmatter<CustomFrontmatter>;\n}\n\n/**\n * Parses the metadata for a given string.\n *\n * @param app - The Obsidian app instance.\n * @param str - The string to parse the metadata for.\n * @returns The parsed metadata.\n */\nexport async function parseMetadata(app: App, str: string): Promise<CachedMetadata> {\n  const encoder = new TextEncoder();\n  const buffer = encoder.encode(str).buffer;\n  return await app.metadataCache.computeMetadataAsync(buffer) ?? {};\n}\n\n/**\n * Registers the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to register the file cache for.\n * @param cache - The file cache to register.\n */\nexport function registerFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile, cache: CachedMetadata): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n\n  app.metadataCache.fileCache[file.path] = {\n    hash: file.path,\n    mtime: 0,\n    size: 0\n  };\n\n  app.metadataCache.metadataCache[file.path] = cache;\n}\n\n/**\n * Registers files in the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to register.\n */\nexport function registerFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 0;\n      count++;\n      registeredFilesCounts.set(file.path, count);\n\n      app.vault.fileMap[file.path] = file;\n\n      if (isFile(file)) {\n        app.metadataCache.uniqueFileLookup.add(file.name.toLowerCase(), file);\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\n/**\n * Temporarily registers files and runs a function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport function tempRegisterFilesAndRun<T>(app: App, files: TAbstractFile[], fn: () => T): T {\n  try {\n    registerFiles(app, files);\n    return fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Temporarily registers files and runs an async function.\n *\n * @typeParam T - The type of the result of the function.\n * @param app - The Obsidian app instance.\n * @param files - The files to temporarily register.\n * @param fn - The function to run.\n * @returns The result of the function.\n */\nexport async function tempRegisterFilesAndRunAsync<T>(app: App, files: TAbstractFile[], fn: () => Promise<T>): Promise<T> {\n  try {\n    registerFiles(app, files);\n    return await fn();\n  } finally {\n    unregisterFiles(app, files);\n  }\n}\n\n/**\n * Unregisters the file cache for a non-existing file.\n *\n * @param app - The Obsidian app instance.\n * @param pathOrFile - The path or file to unregister the file cache for.\n */\nexport function unregisterFileCacheForNonExistingFile(app: App, pathOrFile: PathOrFile): void {\n  const file = getFile(app, pathOrFile, true);\n  if (!file.deleted) {\n    throw new Error('File is existing');\n  }\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.fileCache[file.path];\n  // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n  delete app.metadataCache.metadataCache[file.path];\n}\n\n/**\n * Unregisters files from the Obsidian app.\n *\n * @param app - The Obsidian app instance.\n * @param files - The files to unregister.\n */\nexport function unregisterFiles(app: App, files: TAbstractFile[]): void {\n  const registeredFilesCounts = getRegisteredFilesCounts(app);\n\n  for (let file of files) {\n    while (file.deleted) {\n      let count = registeredFilesCounts.get(file.path) ?? 1;\n      count--;\n      registeredFilesCounts.set(file.path, count);\n      if (count === 0) {\n        registeredFilesCounts.delete(file.path);\n        // eslint-disable-next-line @typescript-eslint/no-dynamic-delete -- We have no other way to delete the property.\n        delete app.vault.fileMap[file.path];\n\n        if (isFile(file)) {\n          app.metadataCache.uniqueFileLookup.remove(file.name.toLowerCase(), file);\n        }\n      }\n\n      file = getFolder(app, parentFolderPath(file.path), true);\n    }\n  }\n}\n\nfunction getRegisteredFilesCounts(app: App): Map<string, number> {\n  return getObsidianDevUtilsState(app, 'registeredFilesCounts', new Map<string, number>()).value;\n}\n"],
  "mappings": "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",
  "names": []
}
