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.
- package/CHANGELOG.md +8 -0
- package/dist/lib/cjs/Async.cjs +53 -32
- package/dist/lib/cjs/Async.d.cts +92 -10
- package/dist/lib/cjs/Library.cjs +1 -1
- package/dist/lib/cjs/obsidian/AsyncWithNotice.cjs +167 -0
- package/dist/lib/cjs/obsidian/AsyncWithNotice.d.cts +74 -0
- package/dist/lib/cjs/obsidian/Backlink.cjs +2 -3
- package/dist/lib/cjs/obsidian/Callout.cjs +8 -2
- package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.cjs +56 -41
- package/dist/lib/cjs/obsidian/Commands/AbstractFileCommandBase.d.cts +45 -12
- package/dist/lib/cjs/obsidian/Commands/CommandBase.cjs +13 -1
- package/dist/lib/cjs/obsidian/Commands/CommandBase.d.cts +7 -0
- package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.cjs +3 -9
- package/dist/lib/cjs/obsidian/Commands/EditorCommandBase.d.cts +7 -7
- package/dist/lib/cjs/obsidian/Commands/FileCommandBase.cjs +123 -26
- package/dist/lib/cjs/obsidian/Commands/FileCommandBase.d.cts +100 -9
- package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.cjs +115 -26
- package/dist/lib/cjs/obsidian/Commands/FolderCommandBase.d.cts +94 -10
- package/dist/lib/cjs/obsidian/FileSystem.cjs +44 -1
- package/dist/lib/cjs/obsidian/FileSystem.d.cts +81 -33
- package/dist/lib/cjs/obsidian/MetadataCache.cjs +45 -41
- package/dist/lib/cjs/obsidian/Queue.cjs +34 -17
- package/dist/lib/cjs/obsidian/Queue.d.cts +62 -12
- package/dist/lib/cjs/obsidian/RenameDeleteHandler.cjs +37 -27
- package/dist/lib/cjs/obsidian/Vault.cjs +45 -66
- package/dist/lib/cjs/obsidian/Vault.d.cts +0 -8
- package/dist/lib/cjs/obsidian/index.cjs +4 -1
- package/dist/lib/cjs/obsidian/index.d.cts +1 -0
- package/dist/lib/esm/Async.d.mts +92 -10
- package/dist/lib/esm/Async.mjs +53 -32
- package/dist/lib/esm/Library.mjs +1 -1
- package/dist/lib/esm/obsidian/AsyncWithNotice.d.mts +74 -0
- package/dist/lib/esm/obsidian/AsyncWithNotice.mjs +63 -0
- package/dist/lib/esm/obsidian/Backlink.mjs +4 -4
- package/dist/lib/esm/obsidian/Callout.mjs +8 -2
- package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.d.mts +45 -12
- package/dist/lib/esm/obsidian/Commands/AbstractFileCommandBase.mjs +54 -40
- package/dist/lib/esm/obsidian/Commands/CommandBase.d.mts +7 -0
- package/dist/lib/esm/obsidian/Commands/CommandBase.mjs +13 -1
- package/dist/lib/esm/obsidian/Commands/EditorCommandBase.d.mts +7 -7
- package/dist/lib/esm/obsidian/Commands/EditorCommandBase.mjs +3 -9
- package/dist/lib/esm/obsidian/Commands/FileCommandBase.d.mts +100 -9
- package/dist/lib/esm/obsidian/Commands/FileCommandBase.mjs +127 -26
- package/dist/lib/esm/obsidian/Commands/FolderCommandBase.d.mts +94 -10
- package/dist/lib/esm/obsidian/Commands/FolderCommandBase.mjs +119 -26
- package/dist/lib/esm/obsidian/FileSystem.d.mts +81 -33
- package/dist/lib/esm/obsidian/FileSystem.mjs +38 -1
- package/dist/lib/esm/obsidian/MetadataCache.mjs +46 -41
- package/dist/lib/esm/obsidian/Queue.d.mts +62 -12
- package/dist/lib/esm/obsidian/Queue.mjs +35 -19
- package/dist/lib/esm/obsidian/RenameDeleteHandler.mjs +37 -27
- package/dist/lib/esm/obsidian/Vault.d.mts +0 -8
- package/dist/lib/esm/obsidian/Vault.mjs +46 -70
- package/dist/lib/esm/obsidian/index.d.mts +1 -0
- package/dist/lib/esm/obsidian/index.mjs +3 -1
- package/obsidian/AsyncWithNotice/package.json +6 -0
- 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
|
|
21
|
+
* A path or an abstract file.
|
|
22
22
|
*/
|
|
23
23
|
export type PathOrAbstractFile = string | TAbstractFile;
|
|
24
24
|
/**
|
|
25
|
-
* A path or a
|
|
25
|
+
* A path or a file.
|
|
26
26
|
*/
|
|
27
27
|
export type PathOrFile = string | TFile;
|
|
28
28
|
/**
|
|
29
|
-
* A path or a
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
75
|
-
* If the provided argument is already a
|
|
76
|
-
* Otherwise, the function uses the app's vault to retrieve the
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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
|
|
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,
|
|
196
|
-
abortSignal
|
|
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
|
|
204
|
-
|
|
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
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
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
|
-
|
|
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
|
-
|
|
239
|
-
|
|
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": []
}

|