obsidian-dev-utils 2.23.0 → 2.23.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +4 -0
- package/dist/lib/obsidian/Vault.cjs +3 -2
- package/package.json +1 -1
package/CHANGELOG.md
CHANGED
@@ -193,8 +193,9 @@ async function removeEmptyFolderHierarchy(app, pathOrFolder) {
|
|
193
193
|
if (folder.children.length > 0) {
|
194
194
|
return;
|
195
195
|
}
|
196
|
+
const parent = folder.parent;
|
196
197
|
await removeFolderSafe(app, folder.path);
|
197
|
-
folder =
|
198
|
+
folder = parent;
|
198
199
|
}
|
199
200
|
}
|
200
201
|
async function createTempFile(app, path) {
|
@@ -249,4 +250,4 @@ async function createTempFolder(app, path) {
|
|
249
250
|
removeFolderSafe,
|
250
251
|
safeList
|
251
252
|
});
|
252
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["var __import_meta_url = globalThis[\"import.meta.url\"] ?? (()=>require(\"node:url\").pathToFileURL(__filename))();\nvar __process = globalThis[\"process\"] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation Vault\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport {\n  App,\n  Notice,\n  TFile,\n  TFolder,\n  type ListedFiles\n} from \"obsidian\";\nimport { deepEqual } from \"../Object.ts\";\nimport {\n  retryWithTimeout,\n  type RetryOptions\n} from \"../Async.ts\";\nimport { getBacklinksForFileSafe } from \"./MetadataCache.ts\";\nimport { printError } from \"../Error.ts\";\nimport { toJson } from \"../JSON.ts\";\nimport {\n  getFile,\n  type PathOrFile\n} from \"./TFile.ts\";\nimport { getPath } from \"./TAbstractFile.ts\";\nimport {\n  getFolderOrNull,\n  type PathOrFolder\n} from \"./TFolder.ts\";\nimport {\n  resolveValue,\n  type ValueProvider\n} from \"../ValueProvider.ts\";\nimport { dirname } from \"../Path.ts\";\n\n/**\n * Represents a file change in the Vault.\n */\nexport type FileChange = {\n  startIndex: number;\n  endIndex: number;\n  oldContent: string;\n  newContent: string;\n};\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param retryOptions - Optional. Configuration options for retrying the process. If not provided, default options will be used.\n *\n * @returns A promise that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function processWithRetry(app: App, pathOrFile: PathOrFile, newContentProvider: ValueProvider<string | null, [string]>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  const DEFAULT_RETRY_OPTIONS: Partial<RetryOptions> = { timeoutInMilliseconds: 60000 };\n  const overriddenOptions: Partial<RetryOptions> = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await retryWithTimeout(async () => {\n    const oldContent = await app.vault.adapter.read(file.path);\n    const newContent = await resolveValue(newContentProvider, oldContent);\n    if (newContent === null) {\n      return false;\n    }\n    let success = true;\n    await app.vault.process(file, (content) => {\n      if (content !== oldContent) {\n        console.warn(`Content of ${file.path} has changed since it was read. Retrying...`);\n        success = false;\n        return content;\n      }\n\n      return newContent;\n    });\n\n    return success;\n  }, overriddenOptions);\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param retryOptions - Optional settings that determine how the operation should retry on failure.\n *\n * @returns A promise that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(app: App, pathOrFile: PathOrFile, changesProvider: ValueProvider<FileChange[]>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS: Partial<RetryOptions> = { timeoutInMilliseconds: 60000 };\n  const overriddenOptions: Partial<RetryOptions> = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await processWithRetry(app, pathOrFile, async (content) => {\n    let changes = await resolveValue(changesProvider);\n\n    for (const change of changes) {\n      const actualContent = content.slice(change.startIndex, change.endIndex);\n      if (actualContent !== change.oldContent) {\n        console.warn(`Content mismatch at ${change.startIndex}-${change.endIndex} in ${getPath(pathOrFile)}:\\nExpected: ${change.oldContent}\\nActual: ${actualContent}`);\n        return null;\n      }\n    }\n\n    changes.sort((a, b) => a.startIndex - b.startIndex);\n\n    // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n    changes = changes.filter((change, index) => {\n      if (index === 0) {\n        return true;\n      }\n      return !deepEqual(change, changes[index - 1]);\n    });\n\n    for (let i = 1; i < changes.length; i++) {\n      const change = changes[i]!;\n      const previousChange = changes[i - 1]!;\n      if (previousChange.endIndex > change.startIndex) {\n        console.warn(`Overlapping changes:\\n${toJson(previousChange)}\\n${toJson(change)}`);\n        return null;\n      }\n    }\n\n    let newContent = \"\";\n    let lastIndex = 0;\n\n    for (const change of changes) {\n      newContent += content.slice(lastIndex, change.startIndex);\n      newContent += change.newContent;\n      lastIndex = change.endIndex;\n    }\n\n    newContent += content.slice(lastIndex);\n    return newContent;\n  }, overriddenOptions);\n}\n\n/**\n * Removes a folder and its contents safely from the vault.\n *\n * @param app - The Obsidian application instance.\n * @param folderPath - The path of the folder to be removed.\n * @param removedNotePath - Optional. The path of the note that triggered the removal.\n * @returns A promise that resolves to a boolean indicating whether the removal was successful.\n */\nexport async function removeFolderSafe(app: App, folderPath: string, removedNotePath?: string): Promise<boolean> {\n  const folder = app.vault.getFolderByPath(folderPath);\n\n  if (!folder) {\n    return false;\n  }\n\n  let canRemove = true;\n\n  for (const child of folder.children) {\n    if (child instanceof TFile) {\n      const backlinks = await getBacklinksForFileSafe(app, child);\n      if (removedNotePath) {\n        backlinks.removeKey(removedNotePath);\n      }\n      if (backlinks.count() !== 0) {\n        new Notice(`Attachment ${child.path} is still used by other notes. It will not be deleted.`);\n        canRemove = false;\n      } else {\n        try {\n          await app.vault.delete(child);\n        } catch (e) {\n          if (await app.vault.adapter.exists(child.path)) {\n            printError(new Error(`Failed to delete ${child.path}`, { cause: e }));\n            canRemove = false;\n          }\n        }\n      }\n    } else if (child instanceof TFolder) {\n      canRemove &&= await removeFolderSafe(app, child.path, removedNotePath);\n    }\n  }\n\n  if (canRemove) {\n    try {\n      await app.vault.delete(folder, true);\n    } catch (e) {\n      if (await app.vault.adapter.exists(folder.path)) {\n        printError(new Error(`Failed to delete ${folder.path}`, { cause: e }));\n        canRemove = false;\n      }\n    }\n  }\n\n  return canRemove;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.adapter.mkdir(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.adapter.exists(path)) {\n      throw e;\n    }\n\n    return true;\n  }\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param path - The path to list files and folders from.\n * @returns A promise that resolves to a `ListedFiles` object containing the listed files and folders.\n */\nexport async function safeList(app: App, path: string): Promise<ListedFiles> {\n  const EMPTY = { files: [], folders: [] };\n  if (!(await app.vault.exists(path))) {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.adapter.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Removes empty folder hierarchy starting from the given folder.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The folder to start removing empty hierarchy from.\n * @returns A promise that resolves when the empty hierarchy is removed.\n */\nexport async function removeEmptyFolderHierarchy(app: App, pathOrFolder: PathOrFolder | null): Promise<void> {\n  let folder = getFolderOrNull(app, pathOrFolder);\n\n  while (folder) {\n    if (folder.children.length > 0) {\n      return;\n    }\n    await removeFolderSafe(app, folder.path);\n    folder = folder.parent;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = app.vault.getFileByPath(path);\n  if (file) {\n    return async () => {\n    };\n  }\n\n  const folderCleanup = await createTempFolder(app, dirname(path));\n\n  try {\n    await app.vault.create(path, \"\");\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  file = app.vault.getFileByPath(path)!;\n\n  return async () => {\n    if (!file.deleted) {\n      await app.vault.delete(file, true);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns - A promise that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = app.vault.getFolderByPath(path);\n  if (folder) {\n    return async () => {\n    };\n  }\n\n  const dirPath = dirname(path);\n  await createTempFolder(app, dirPath);\n\n  const folderCleanup = await createTempFolder(app, dirname(path));\n\n  await createFolderSafe(app, path);\n\n  folder = app.vault.getFolderByPath(path)!;\n\n  return async () => {\n    if (!folder.deleted) {\n      await app.vault.delete(folder, true);\n    }\n    await folderCleanup();\n  };\n}\n"],
  "mappings": "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",
  "names": []
}

|
253
|
+
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../src/obsidian/Vault.ts"],
  "sourcesContent": ["var __import_meta_url = globalThis[\"import.meta.url\"] ?? (()=>require(\"node:url\").pathToFileURL(__filename))();\nvar __process = globalThis[\"process\"] ?? {\n  \"cwd\": ()=>\"/\",\n  \"env\": {},\n  \"platform\": \"android\"\n};\n/**\n * @packageDocumentation Vault\n * This module provides utility functions for working with the Obsidian Vault.\n */\n\nimport {\n  App,\n  Notice,\n  TFile,\n  TFolder,\n  type ListedFiles\n} from \"obsidian\";\nimport { deepEqual } from \"../Object.ts\";\nimport {\n  retryWithTimeout,\n  type RetryOptions\n} from \"../Async.ts\";\nimport { getBacklinksForFileSafe } from \"./MetadataCache.ts\";\nimport { printError } from \"../Error.ts\";\nimport { toJson } from \"../JSON.ts\";\nimport {\n  getFile,\n  type PathOrFile\n} from \"./TFile.ts\";\nimport { getPath } from \"./TAbstractFile.ts\";\nimport {\n  getFolderOrNull,\n  type PathOrFolder\n} from \"./TFolder.ts\";\nimport {\n  resolveValue,\n  type ValueProvider\n} from \"../ValueProvider.ts\";\nimport { dirname } from \"../Path.ts\";\n\n/**\n * Represents a file change in the Vault.\n */\nexport type FileChange = {\n  startIndex: number;\n  endIndex: number;\n  oldContent: string;\n  newContent: string;\n};\n\n/**\n * Retrieves an array of Markdown files from the app's vault and sorts them alphabetically by their file path.\n *\n * @param app - The Obsidian app instance.\n * @returns An array of Markdown files sorted by file path.\n */\nexport function getMarkdownFilesSorted(app: App): TFile[] {\n  return app.vault.getMarkdownFiles().sort((a, b) => a.path.localeCompare(b.path));\n}\n\n/**\n * Processes a file with retry logic, updating its content based on a provided value or function.\n *\n * @param app - The application instance, typically used for accessing the vault.\n * @param pathOrFile - The path or file to be processed. It can be a string representing the path or a file object.\n * @param newContentProvider - A value provider that returns the new content based on the old content of the file.\n * It can be a string or a function that takes the old content as an argument and returns the new content.\n * If function is provided, it should return `null` if the process should be retried.\n * @param retryOptions - Optional. Configuration options for retrying the process. If not provided, default options will be used.\n *\n * @returns A promise that resolves once the process is complete.\n *\n * @throws Will throw an error if the process fails after the specified number of retries or timeout.\n */\nexport async function processWithRetry(app: App, pathOrFile: PathOrFile, newContentProvider: ValueProvider<string | null, [string]>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const file = getFile(app, pathOrFile);\n  const DEFAULT_RETRY_OPTIONS: Partial<RetryOptions> = { timeoutInMilliseconds: 60000 };\n  const overriddenOptions: Partial<RetryOptions> = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await retryWithTimeout(async () => {\n    const oldContent = await app.vault.adapter.read(file.path);\n    const newContent = await resolveValue(newContentProvider, oldContent);\n    if (newContent === null) {\n      return false;\n    }\n    let success = true;\n    await app.vault.process(file, (content) => {\n      if (content !== oldContent) {\n        console.warn(`Content of ${file.path} has changed since it was read. Retrying...`);\n        success = false;\n        return content;\n      }\n\n      return newContent;\n    });\n\n    return success;\n  }, overriddenOptions);\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param retryOptions - Optional settings that determine how the operation should retry on failure.\n *\n * @returns A promise that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(app: App, pathOrFile: PathOrFile, changesProvider: ValueProvider<FileChange[]>, retryOptions: Partial<RetryOptions> = {}): Promise<void> {\n  const DEFAULT_RETRY_OPTIONS: Partial<RetryOptions> = { timeoutInMilliseconds: 60000 };\n  const overriddenOptions: Partial<RetryOptions> = { ...DEFAULT_RETRY_OPTIONS, ...retryOptions };\n  await processWithRetry(app, pathOrFile, async (content) => {\n    let changes = await resolveValue(changesProvider);\n\n    for (const change of changes) {\n      const actualContent = content.slice(change.startIndex, change.endIndex);\n      if (actualContent !== change.oldContent) {\n        console.warn(`Content mismatch at ${change.startIndex}-${change.endIndex} in ${getPath(pathOrFile)}:\\nExpected: ${change.oldContent}\\nActual: ${actualContent}`);\n        return null;\n      }\n    }\n\n    changes.sort((a, b) => a.startIndex - b.startIndex);\n\n    // BUG: https://forum.obsidian.md/t/bug-duplicated-links-in-metadatacache-inside-footnotes/85551\n    changes = changes.filter((change, index) => {\n      if (index === 0) {\n        return true;\n      }\n      return !deepEqual(change, changes[index - 1]);\n    });\n\n    for (let i = 1; i < changes.length; i++) {\n      const change = changes[i]!;\n      const previousChange = changes[i - 1]!;\n      if (previousChange.endIndex > change.startIndex) {\n        console.warn(`Overlapping changes:\\n${toJson(previousChange)}\\n${toJson(change)}`);\n        return null;\n      }\n    }\n\n    let newContent = \"\";\n    let lastIndex = 0;\n\n    for (const change of changes) {\n      newContent += content.slice(lastIndex, change.startIndex);\n      newContent += change.newContent;\n      lastIndex = change.endIndex;\n    }\n\n    newContent += content.slice(lastIndex);\n    return newContent;\n  }, overriddenOptions);\n}\n\n/**\n * Removes a folder and its contents safely from the vault.\n *\n * @param app - The Obsidian application instance.\n * @param folderPath - The path of the folder to be removed.\n * @param removedNotePath - Optional. The path of the note that triggered the removal.\n * @returns A promise that resolves to a boolean indicating whether the removal was successful.\n */\nexport async function removeFolderSafe(app: App, folderPath: string, removedNotePath?: string): Promise<boolean> {\n  const folder = app.vault.getFolderByPath(folderPath);\n\n  if (!folder) {\n    return false;\n  }\n\n  let canRemove = true;\n\n  for (const child of folder.children) {\n    if (child instanceof TFile) {\n      const backlinks = await getBacklinksForFileSafe(app, child);\n      if (removedNotePath) {\n        backlinks.removeKey(removedNotePath);\n      }\n      if (backlinks.count() !== 0) {\n        new Notice(`Attachment ${child.path} is still used by other notes. It will not be deleted.`);\n        canRemove = false;\n      } else {\n        try {\n          await app.vault.delete(child);\n        } catch (e) {\n          if (await app.vault.adapter.exists(child.path)) {\n            printError(new Error(`Failed to delete ${child.path}`, { cause: e }));\n            canRemove = false;\n          }\n        }\n      }\n    } else if (child instanceof TFolder) {\n      canRemove &&= await removeFolderSafe(app, child.path, removedNotePath);\n    }\n  }\n\n  if (canRemove) {\n    try {\n      await app.vault.delete(folder, true);\n    } catch (e) {\n      if (await app.vault.adapter.exists(folder.path)) {\n        printError(new Error(`Failed to delete ${folder.path}`, { cause: e }));\n        canRemove = false;\n      }\n    }\n  }\n\n  return canRemove;\n}\n\n/**\n * Creates a folder safely in the specified path.\n *\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns A promise that resolves to a boolean indicating whether the folder was created.\n * @throws If an error occurs while creating the folder and it still doesn't exist.\n */\nexport async function createFolderSafe(app: App, path: string): Promise<boolean> {\n  if (await app.vault.adapter.exists(path)) {\n    return false;\n  }\n\n  try {\n    await app.vault.adapter.mkdir(path);\n    return true;\n  } catch (e) {\n    if (!await app.vault.adapter.exists(path)) {\n      throw e;\n    }\n\n    return true;\n  }\n}\n\n/**\n * Safely lists the files and folders at the specified path in the vault.\n *\n * @param app - The Obsidian application instance.\n * @param path - The path to list files and folders from.\n * @returns A promise that resolves to a `ListedFiles` object containing the listed files and folders.\n */\nexport async function safeList(app: App, path: string): Promise<ListedFiles> {\n  const EMPTY = { files: [], folders: [] };\n  if (!(await app.vault.exists(path))) {\n    return EMPTY;\n  }\n\n  try {\n    return await app.vault.adapter.list(path);\n  } catch (e) {\n    if (await app.vault.adapter.exists(path)) {\n      throw e;\n    }\n    return EMPTY;\n  }\n}\n\n/**\n * Removes empty folder hierarchy starting from the given folder.\n *\n * @param app - The application instance.\n * @param pathOrFolder - The folder to start removing empty hierarchy from.\n * @returns A promise that resolves when the empty hierarchy is removed.\n */\nexport async function removeEmptyFolderHierarchy(app: App, pathOrFolder: PathOrFolder | null): Promise<void> {\n  let folder = getFolderOrNull(app, pathOrFolder);\n\n  while (folder) {\n    if (folder.children.length > 0) {\n      return;\n    }\n    const parent = folder.parent;\n    await removeFolderSafe(app, folder.path);\n    folder = parent;\n  }\n}\n\n/**\n * Creates a temporary file in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the file to create.\n * @returns A promise that resolves to a function that can be called to delete the temporary file and all its created parents.\n */\nexport async function createTempFile(app: App, path: string): Promise<() => Promise<void>> {\n  let file = app.vault.getFileByPath(path);\n  if (file) {\n    return async () => {\n    };\n  }\n\n  const folderCleanup = await createTempFolder(app, dirname(path));\n\n  try {\n    await app.vault.create(path, \"\");\n  } catch (e) {\n    if (!await app.vault.exists(path)) {\n      throw e;\n    }\n  }\n\n  file = app.vault.getFileByPath(path)!;\n\n  return async () => {\n    if (!file.deleted) {\n      await app.vault.delete(file, true);\n    }\n    await folderCleanup();\n  };\n}\n\n/**\n * Creates a temporary folder in the vault with parent folders if needed.\n * @param app - The application instance.\n * @param path - The path of the folder to create.\n * @returns - A promise that resolves to a function that can be called to delete the temporary folder and all its created parents.\n */\nexport async function createTempFolder(app: App, path: string): Promise<() => Promise<void>> {\n  let folder = app.vault.getFolderByPath(path);\n  if (folder) {\n    return async () => {\n    };\n  }\n\n  const dirPath = dirname(path);\n  await createTempFolder(app, dirPath);\n\n  const folderCleanup = await createTempFolder(app, dirname(path));\n\n  await createFolderSafe(app, path);\n\n  folder = app.vault.getFolderByPath(path)!;\n\n  return async () => {\n    if (!folder.deleted) {\n      await app.vault.delete(folder, true);\n    }\n    await folderCleanup();\n  };\n}\n"],
  "mappings": "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",
  "names": []
}

|
package/package.json
CHANGED