obsidian-dev-utils 40.16.0 → 41.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.
@@ -46,7 +46,7 @@ async function addGitTag(newVersion) {
46
46
  }
47
47
  async function addUpdatedFilesToGit(newVersion) {
48
48
  await execFromRoot(["git", "add", "--all"], { isQuiet: true });
49
- await execFromRoot(`git commit -m ${newVersion} --allow-empty`, { isQuiet: true });
49
+ await execFromRoot(["git", "commit", "-m", `chore: release ${newVersion}`, "--allow-empty"], { isQuiet: true });
50
50
  }
51
51
  async function checkGitHubCliInstalled() {
52
52
  try {
@@ -358,4 +358,4 @@ export {
358
358
  updateVersionInFiles,
359
359
  validate
360
360
  };
361
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/ScriptUtils/version.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides functions for managing version updates in a project.\n * It includes tasks such as validating version update types, checking the state\n * of Git and GitHub CLI, updating version numbers in files, and performing\n * Git operations such as tagging and pushing.\n */\n\nimport type { PackageLockJson } from './Npm.ts';\n\nimport { getLibDebugger } from '../Debug.ts';\nimport { throwExpression } from '../Error.ts';\nimport { ObsidianPluginRepoPaths } from '../obsidian/Plugin/ObsidianPluginRepoPaths.ts';\nimport { join } from '../Path.ts';\nimport { replaceAll } from '../String.ts';\nimport { readdirPosix } from './Fs.ts';\nimport { editJson } from './JSON.ts';\nimport {\n  cp,\n  createInterface,\n  existsSync,\n  readFile,\n  rm,\n  writeFile\n} from './NodeModules.ts';\nimport {\n  editNpmShrinkWrapJson,\n  editPackageJson,\n  editPackageLockJson,\n  readPackageJson\n} from './Npm.ts';\nimport { npmRun } from './NpmRun.ts';\nimport { ObsidianDevUtilsRepoPaths } from './ObsidianDevUtilsRepoPaths.ts';\nimport {\n  execFromRoot,\n  resolvePathFromRootSafe\n} from './Root.ts';\n\n/**\n * Enum representing different types of version updates.\n */\nexport enum VersionUpdateType {\n  Beta = 'beta',\n  Invalid = 'invalid',\n  Major = 'major',\n  Manual = 'manual',\n  Minor = 'minor',\n  Patch = 'patch'\n}\n\n/**\n * Type representing the manifest file format for Obsidian plugins.\n */\nexport interface Manifest {\n  /**\n   * A minimum Obsidian version required for the plugin.\n   */\n  minAppVersion: string;\n\n  /**\n   * A version of the plugin.\n   */\n  version: string;\n}\n\n/**\n * Type representing the structure of Obsidian releases JSON.\n */\nexport interface ObsidianReleasesJson {\n  /**\n   * A name of the Obsidian release.\n   */\n  name: string;\n}\n\n/**\n * Creates a Git tag for the new version.\n *\n * @param newVersion - The new version number to use for the tag.\n * @returns A {@link Promise} that resolves when the tag has been created.\n */\nexport async function addGitTag(newVersion: string): Promise<void> {\n  await execFromRoot(`git tag -a ${newVersion} -m ${newVersion} --force`, { isQuiet: true });\n}\n\n/**\n * Adds updated files to the Git staging area and commits them with the new version message.\n *\n * @param newVersion - The new version number used as the commit message.\n * @returns A {@link Promise} that resolves when the files have been added and committed.\n */\nexport async function addUpdatedFilesToGit(newVersion: string): Promise<void> {\n  await execFromRoot(['git', 'add', '--all'], { isQuiet: true });\n  await execFromRoot(`git commit -m ${newVersion} --allow-empty`, { isQuiet: true });\n}\n\n/**\n * Checks if the GitHub CLI is installed on the system.\n *\n * Throws an error if the GitHub CLI is not installed.\n *\n * @throws Error if the GitHub CLI is not installed.\n */\nexport async function checkGitHubCliInstalled(): Promise<void> {\n  try {\n    await execFromRoot('gh --version', { isQuiet: true });\n  } catch {\n    throw new Error('GitHub CLI is not installed. Please install it from https://cli.github.com/');\n  }\n}\n\n/**\n * Checks if Git is installed on the system.\n *\n * Throws an error if Git is not installed.\n *\n * @throws Error if Git is not installed.\n */\nexport async function checkGitInstalled(): Promise<void> {\n  try {\n    await execFromRoot('git --version', { isQuiet: true });\n  } catch {\n    throw new Error('Git is not installed. Please install it from https://git-scm.com/');\n  }\n}\n\n/**\n * Checks if the Git repository is clean, meaning there are no uncommitted changes.\n *\n * Throws an error if the Git repository is not clean.\n *\n * @throws Error if the Git repository is not clean.\n */\nexport async function checkGitRepoClean(): Promise<void> {\n  try {\n    const stdout = await execFromRoot('git status --porcelain --untracked-files=all', { isQuiet: true });\n    if (stdout) {\n      throw new Error();\n    }\n  } catch {\n    throw new Error('Git repository is not clean. Please commit or stash your changes before releasing a new version.');\n  }\n}\n\n/**\n * Copies the updated manifest file to the distribution build folder.\n *\n * @returns A {@link Promise} that resolves when the copy operation is complete.\n */\nexport async function copyUpdatedManifest(): Promise<void> {\n  await cp(\n    resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson),\n    resolvePathFromRootSafe(join(ObsidianPluginRepoPaths.DistBuild, ObsidianPluginRepoPaths.ManifestJson)),\n    { force: true }\n  );\n}\n\n/**\n * Generates a new version string based on the current version and the specified update type.\n *\n * @param versionUpdateType - The type of version update (major, minor, patch, beta, or manual).\n * @returns A {@link Promise} that resolves to the new version string.\n * @throws Error if the current version format is invalid.\n */\nexport async function getNewVersion(versionUpdateType: string): Promise<string> {\n  const versionType = getVersionUpdateType(versionUpdateType);\n  if (versionType === VersionUpdateType.Manual) {\n    return versionUpdateType;\n  }\n\n  const packageJson = await readPackageJson();\n  const currentVersion = packageJson.version ?? '';\n\n  const match = /^(?<Major>\\d+)\\.(?<Minor>\\d+)\\.(?<Patch>\\d+)(?:-beta\\.(?<Beta>\\d+))?$/.exec(currentVersion);\n  if (!match) {\n    throw new Error(`Invalid current version format: ${currentVersion}`);\n  }\n\n  let major = Number(match.groups?.['Major'] ?? '');\n  let minor = Number(match.groups?.['Minor'] ?? '');\n  let patch = Number(match.groups?.['Patch'] ?? '');\n  let beta = Number(match.groups?.['Beta'] ?? '');\n\n  switch (versionType) {\n    case VersionUpdateType.Beta:\n      if (beta === 0) {\n        patch++;\n      }\n      beta++;\n      break;\n    case VersionUpdateType.Major:\n      major++;\n      minor = 0;\n      patch = 0;\n      beta = 0;\n      break;\n    case VersionUpdateType.Minor:\n      minor++;\n      patch = 0;\n      beta = 0;\n      break;\n    case VersionUpdateType.Patch:\n      if (beta === 0) {\n        patch++;\n      } else {\n        beta = 0;\n      }\n      break;\n    default:\n      throw new Error(`Invalid version update type: ${versionType}`);\n  }\n\n  return `${String(major)}.${String(minor)}.${String(patch)}${beta > 0 ? `-beta.${String(beta)}` : ''}`;\n}\n\n/**\n * Retrieves the release notes for a specific version from the changelog.\n *\n * @param newVersion - The new version number for which to get the release notes.\n * @returns A {@link Promise} that resolves to the release notes for the specified version.\n */\nexport async function getReleaseNotes(newVersion: string): Promise<string> {\n  const changelogPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ChangelogMd);\n  const content = await readFile(changelogPath, 'utf-8');\n  const newVersionEscaped = replaceAll(newVersion, '.', '\\\\.');\n  const match = new RegExp(`\\n## ${newVersionEscaped}\\n\\n((.|\\n)+?)\\n\\n##`).exec(content);\n  let releaseNotes = match?.[1] ? `${match[1]}\\n\\n` : '';\n\n  const tags = (await execFromRoot('git tag --sort=-creatordate', { isQuiet: true })).split(/\\r?\\n/);\n  const previousVersion = tags[1];\n  let changesUrl: string;\n\n  const repoUrl = await execFromRoot('gh repo view --json url -q .url', { isQuiet: true });\n\n  if (previousVersion) {\n    changesUrl = `${repoUrl}/compare/${previousVersion}...${newVersion}`;\n  } else {\n    changesUrl = `${repoUrl}/commits/${newVersion}`;\n  }\n\n  releaseNotes += `**Full Changelog**: ${changesUrl}`;\n  return releaseNotes;\n}\n\n/**\n * Determines the type of version update based on the input string.\n *\n * @param versionUpdateType - The input string representing the version update type.\n * @returns The corresponding `VersionUpdateType`.\n */\nexport function getVersionUpdateType(versionUpdateType: string): VersionUpdateType {\n  const versionUpdateTypeEnum = versionUpdateType as VersionUpdateType;\n  switch (versionUpdateTypeEnum) {\n    case VersionUpdateType.Beta:\n    case VersionUpdateType.Major:\n    case VersionUpdateType.Minor:\n    case VersionUpdateType.Patch:\n      return versionUpdateTypeEnum;\n\n    default:\n      if (/^\\d+\\.\\d+\\.\\d+(?:-[\\w\\d.-]+)?$/.test(versionUpdateType)) {\n        return VersionUpdateType.Manual;\n      }\n\n      return VersionUpdateType.Invalid;\n  }\n}\n\n/**\n * Pushes commits and tags to the remote Git repository.\n *\n * @returns A {@link Promise} that resolves when the push operation is complete.\n */\nexport async function gitPush(): Promise<void> {\n  await execFromRoot('git push --follow-tags --force', { isQuiet: true });\n}\n\n/**\n * Publishes a GitHub release for the new version.\n *\n * Handles the creation of a release and uploading files for either an Obsidian plugin or another project.\n *\n * @param newVersion - The new version number for the release.\n * @param isObsidianPlugin - A boolean indicating if the project is an Obsidian plugin.\n * @returns A {@link Promise} that resolves when the release has been published.\n */\nexport async function publishGitHubRelease(newVersion: string, isObsidianPlugin: boolean): Promise<void> {\n  let filePaths: string[];\n\n  if (isObsidianPlugin) {\n    const buildFolder = resolvePathFromRootSafe(ObsidianPluginRepoPaths.DistBuild);\n    const fileNames = await readdirPosix(buildFolder);\n    filePaths = fileNames.map((fileName) => join(buildFolder, fileName));\n  } else {\n    const resultJson = await execFromRoot(['npm', 'pack', '--pack-destination', ObsidianDevUtilsRepoPaths.Dist, '--json'], { isQuiet: true });\n    const result = JSON.parse(resultJson) as [{ filename: string }];\n    filePaths = [\n      join(ObsidianDevUtilsRepoPaths.Dist, result[0].filename),\n      join(ObsidianDevUtilsRepoPaths.Dist, ObsidianDevUtilsRepoPaths.StylesCss)\n    ];\n  }\n\n  filePaths = filePaths.filter((filePath) => existsSync(resolvePathFromRootSafe(filePath)));\n\n  await execFromRoot([\n    'gh',\n    'release',\n    'create',\n    newVersion,\n    ...filePaths,\n    '--title',\n    `v${newVersion}`,\n    ...(isBeta(newVersion) ? ['--prerelease'] : []),\n    '--notes-file',\n    '-'\n  ], {\n    isQuiet: true,\n    stdin: await getReleaseNotes(newVersion)\n  });\n}\n\n/**\n * Updates the changelog file with new version information and commit messages.\n *\n * This function reads the current changelog, appends new entries for the latest version,\n * and prompts the user to review the changes.\n *\n * @param newVersion - The new version number to be added to the changelog.\n * @returns A {@link Promise} that resolves when the changelog update is complete.\n */\nexport async function updateChangelog(newVersion: string): Promise<void> {\n  const HEADER_LINES_COUNT = 2;\n  const changelogPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ChangelogMd);\n  let previousChangelogLines: string[];\n  if (existsSync(changelogPath)) {\n    const content = await readFile(changelogPath, 'utf-8');\n    previousChangelogLines = content.split('\\n').slice(HEADER_LINES_COUNT);\n    if (previousChangelogLines.at(-1) === '') {\n      previousChangelogLines.pop();\n    }\n  } else {\n    previousChangelogLines = [];\n  }\n\n  const lastTag = replaceAll(previousChangelogLines[0] ?? '', '## ', '');\n  const commitRange = lastTag ? `${lastTag}..HEAD` : 'HEAD';\n  const commitMessagesStr = await execFromRoot(`git log ${commitRange} --format=%B --first-parent -z`, { isQuiet: true });\n  const commitMessages = commitMessagesStr.split('\\0').filter(Boolean).map(toSingleLine);\n\n  let newChangeLog = `# CHANGELOG\\n\\n## ${newVersion}\\n\\n`;\n\n  for (const message of commitMessages) {\n    newChangeLog += `- ${message}\\n`;\n  }\n\n  if (previousChangelogLines.length > 0) {\n    newChangeLog += '\\n';\n    for (const line of previousChangelogLines) {\n      newChangeLog += `${line}\\n`;\n    }\n  }\n\n  await writeFile(changelogPath, newChangeLog, 'utf-8');\n\n  const codeVersion = await execFromRoot('code --version', {\n    isQuiet: true,\n    shouldIgnoreExitCode: true\n  });\n  const versionDebugger = getLibDebugger('Version');\n  if (codeVersion) {\n    versionDebugger(`Please update the ${ObsidianPluginRepoPaths.ChangelogMd} file. Close Visual Studio Code when you are done...`);\n    await execFromRoot(['code', '-w', changelogPath], {\n      isQuiet: true,\n      shouldIgnoreExitCode: true\n    });\n  } else {\n    versionDebugger('Could not find Visual Studio Code in your PATH. Using console mode instead.');\n    await createInterface(process.stdin, process.stdout).question(\n      `Please update the ${ObsidianPluginRepoPaths.ChangelogMd} file. Press Enter when you are done...`\n    );\n  }\n}\n\n/**\n * Updates the version of the project based on the specified update type.\n *\n * This function performs a series of tasks to handle version updates:\n * 1. Validates the version update type.\n * 2. Checks if Git and GitHub CLI are installed.\n * 3. Verifies that the Git repository is clean.\n * 4. Runs spellcheck and linting.\n * 5. Builds the project.\n * 6. Updates version in files and changelog.\n * 7. Adds updated files to Git, tags the commit, and pushes to the repository.\n * 8. If an Obsidian plugin, copies the updated manifest and publishes a GitHub release.\n *\n * @param versionUpdateType - The type of version update to perform (major, minor, patch, beta, or x.y.z[-beta:u]).\n * @param prepareGitHubRelease - A callback function to prepare the GitHub release.\n * @returns A {@link Promise} that resolves when the version update is complete.\n */\nexport async function updateVersion(versionUpdateType?: string, prepareGitHubRelease?: (newVersion: string) => Promise<void>): Promise<void> {\n  if (!versionUpdateType) {\n    const npmOldVersion = process.env['npm_old_version'];\n    const npmNewVersion = process.env['npm_new_version'];\n\n    if (npmOldVersion && npmNewVersion) {\n      await updateVersionInFiles(npmOldVersion);\n      await updateVersion(npmNewVersion, prepareGitHubRelease);\n      return;\n    }\n\n    throw new Error('No version update type provided');\n  }\n\n  const isObsidianPlugin = existsSync(resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson));\n\n  validate(versionUpdateType);\n  await checkGitInstalled();\n  await checkGitRepoClean();\n  await checkGitHubCliInstalled();\n  await npmRun('format:check');\n  await npmRun('spellcheck');\n  await npmRun('build');\n  await npmRun('lint');\n\n  const newVersion = await getNewVersion(versionUpdateType);\n  await updateVersionInFiles(newVersion);\n  if (isObsidianPlugin) {\n    await updateVersionInFilesForPlugin(newVersion);\n  }\n\n  await updateChangelog(newVersion);\n  await addUpdatedFilesToGit(newVersion);\n  await addGitTag(newVersion);\n  await gitPush();\n  await prepareGitHubRelease?.(newVersion);\n  await publishGitHubRelease(newVersion, isObsidianPlugin);\n}\n\n/**\n * Updates the version in various files, including `package.json`, `package-lock.json`,\n * and Obsidian plugin manifests if applicable.\n *\n * @param newVersion - The new version string to update in the files.\n * @returns A {@link Promise} that resolves when the update is complete.\n */\nexport async function updateVersionInFiles(newVersion: string): Promise<void> {\n  await editPackageJson((packageJson) => {\n    packageJson.version = newVersion;\n  });\n\n  await editPackageLockJson(update, { shouldSkipIfMissing: true });\n  await editNpmShrinkWrapJson(update, { shouldSkipIfMissing: true });\n\n  function update(packageLockJson: PackageLockJson): void {\n    packageLockJson.version = newVersion;\n    const defaultPackage = packageLockJson.packages?.[''];\n    if (defaultPackage) {\n      defaultPackage.version = newVersion;\n    }\n  }\n}\n\n/**\n * Validates the version update type to ensure it is either a recognized type\n * or a valid manual version string.\n *\n * @param versionUpdateType - The version update type to validate.\n * @throws Error if the version update type is invalid.\n */\nexport function validate(versionUpdateType: string): void {\n  if (getVersionUpdateType(versionUpdateType) === VersionUpdateType.Invalid) {\n    throw new Error('Invalid version update type. Please use \\'major\\', \\'minor\\', \\'patch\\', or \\'x.y.z[-suffix]\\' format.');\n  }\n}\n\n/**\n * Fetches the latest version of Obsidian from the GitHub releases API.\n *\n * @returns A {@link Promise} that resolves to the latest version of Obsidian.\n */\nasync function getLatestObsidianVersion(): Promise<string> {\n  const response = await fetch('https://api.github.com/repos/obsidianmd/obsidian-releases/releases/latest');\n  const obsidianReleasesJson = await response.json() as Partial<ObsidianReleasesJson>;\n  return obsidianReleasesJson.name ?? throwExpression(new Error('Could not find the name of the latest Obsidian release'));\n}\n\nfunction isBeta(version: string): boolean {\n  return version.includes(VersionUpdateType.Beta);\n}\n\nfunction toSingleLine(str: string): string {\n  const lines = str.split(/\\r?\\n/).filter(Boolean);\n  return lines.join(' ');\n}\n\nasync function updateVersionInFilesForPlugin(newVersion: string): Promise<void> {\n  const manifestBetaJsonPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestBetaJson);\n  if (isBeta(newVersion)) {\n    await cp(\n      resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson),\n      manifestBetaJsonPath,\n      { force: true }\n    );\n    await editJson<Manifest>(ObsidianPluginRepoPaths.ManifestBetaJson, (manifest) => {\n      manifest.version = newVersion;\n    });\n  } else {\n    const latestObsidianVersion = await getLatestObsidianVersion();\n\n    await editJson<Manifest>(ObsidianPluginRepoPaths.ManifestJson, (manifest) => {\n      manifest.minAppVersion = latestObsidianVersion;\n      manifest.version = newVersion;\n    });\n\n    await editJson<Record<string, string>>(ObsidianPluginRepoPaths.VersionsJson, (versions) => {\n      versions[newVersion] = latestObsidianVersion;\n    });\n\n    if (existsSync(manifestBetaJsonPath)) {\n      await rm(manifestBetaJsonPath);\n    }\n  }\n\n  await copyUpdatedManifest();\n}\n"],
  "mappings": ";;;;;;;AAWA,SAAS,sBAAsB;AAC/B,SAAS,uBAAuB;AAChC,SAAS,+BAA+B;AACxC,SAAS,YAAY;AACrB,SAAS,kBAAkB;AAC3B,SAAS,oBAAoB;AAC7B,SAAS,gBAAgB;AACzB;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,cAAc;AACvB,SAAS,iCAAiC;AAC1C;AAAA,EACE;AAAA,EACA;AAAA,OACK;AAKA,IAAK,oBAAL,kBAAKA,uBAAL;AACL,EAAAA,mBAAA,UAAO;AACP,EAAAA,mBAAA,aAAU;AACV,EAAAA,mBAAA,WAAQ;AACR,EAAAA,mBAAA,YAAS;AACT,EAAAA,mBAAA,WAAQ;AACR,EAAAA,mBAAA,WAAQ;AANE,SAAAA;AAAA,GAAA;AAwCZ,eAAsB,UAAU,YAAmC;AACjE,QAAM,aAAa,cAAc,UAAU,OAAO,UAAU,YAAY,EAAE,SAAS,KAAK,CAAC;AAC3F;AAQA,eAAsB,qBAAqB,YAAmC;AAC5E,QAAM,aAAa,CAAC,OAAO,OAAO,OAAO,GAAG,EAAE,SAAS,KAAK,CAAC;AAC7D,QAAM,aAAa,iBAAiB,UAAU,kBAAkB,EAAE,SAAS,KAAK,CAAC;AACnF;AASA,eAAsB,0BAAyC;AAC7D,MAAI;AACF,UAAM,aAAa,gBAAgB,EAAE,SAAS,KAAK,CAAC;AAAA,EACtD,QAAQ;AACN,UAAM,IAAI,MAAM,6EAA6E;AAAA,EAC/F;AACF;AASA,eAAsB,oBAAmC;AACvD,MAAI;AACF,UAAM,aAAa,iBAAiB,EAAE,SAAS,KAAK,CAAC;AAAA,EACvD,QAAQ;AACN,UAAM,IAAI,MAAM,mEAAmE;AAAA,EACrF;AACF;AASA,eAAsB,oBAAmC;AACvD,MAAI;AACF,UAAM,SAAS,MAAM,aAAa,gDAAgD,EAAE,SAAS,KAAK,CAAC;AACnG,QAAI,QAAQ;AACV,YAAM,IAAI,MAAM;AAAA,IAClB;AAAA,EACF,QAAQ;AACN,UAAM,IAAI,MAAM,kGAAkG;AAAA,EACpH;AACF;AAOA,eAAsB,sBAAqC;AACzD,QAAM;AAAA,IACJ,wBAAwB,wBAAwB,YAAY;AAAA,IAC5D,wBAAwB,KAAK,wBAAwB,WAAW,wBAAwB,YAAY,CAAC;AAAA,IACrG,EAAE,OAAO,KAAK;AAAA,EAChB;AACF;AASA,eAAsB,cAAc,mBAA4C;AAC9E,QAAM,cAAc,qBAAqB,iBAAiB;AAC1D,MAAI,gBAAgB,uBAA0B;AAC5C,WAAO;AAAA,EACT;AAEA,QAAM,cAAc,MAAM,gBAAgB;AAC1C,QAAM,iBAAiB,YAAY,WAAW;AAE9C,QAAM,QAAQ,wEAAwE,KAAK,cAAc;AACzG,MAAI,CAAC,OAAO;AACV,UAAM,IAAI,MAAM,mCAAmC,cAAc,EAAE;AAAA,EACrE;AAEA,MAAI,QAAQ,OAAO,MAAM,SAAS,OAAO,KAAK,EAAE;AAChD,MAAI,QAAQ,OAAO,MAAM,SAAS,OAAO,KAAK,EAAE;AAChD,MAAI,QAAQ,OAAO,MAAM,SAAS,OAAO,KAAK,EAAE;AAChD,MAAI,OAAO,OAAO,MAAM,SAAS,MAAM,KAAK,EAAE;AAE9C,UAAQ,aAAa;AAAA,IACnB,KAAK;AACH,UAAI,SAAS,GAAG;AACd;AAAA,MACF;AACA;AACA;AAAA,IACF,KAAK;AACH;AACA,cAAQ;AACR,cAAQ;AACR,aAAO;AACP;AAAA,IACF,KAAK;AACH;AACA,cAAQ;AACR,aAAO;AACP;AAAA,IACF,KAAK;AACH,UAAI,SAAS,GAAG;AACd;AAAA,MACF,OAAO;AACL,eAAO;AAAA,MACT;AACA;AAAA,IACF;AACE,YAAM,IAAI,MAAM,gCAAgC,WAAW,EAAE;AAAA,EACjE;AAEA,SAAO,GAAG,OAAO,KAAK,CAAC,IAAI,OAAO,KAAK,CAAC,IAAI,OAAO,KAAK,CAAC,GAAG,OAAO,IAAI,SAAS,OAAO,IAAI,CAAC,KAAK,EAAE;AACrG;AAQA,eAAsB,gBAAgB,YAAqC;AACzE,QAAM,gBAAgB,wBAAwB,wBAAwB,WAAW;AACjF,QAAM,UAAU,MAAM,SAAS,eAAe,OAAO;AACrD,QAAM,oBAAoB,WAAW,YAAY,KAAK,KAAK;AAC3D,QAAM,QAAQ,IAAI,OAAO;AAAA,KAAQ,iBAAiB;AAAA;AAAA;AAAA;AAAA;AAAA,GAAsB,EAAE,KAAK,OAAO;AACtF,MAAI,eAAe,QAAQ,CAAC,IAAI,GAAG,MAAM,CAAC,CAAC;AAAA;AAAA,IAAS;AAEpD,QAAM,QAAQ,MAAM,aAAa,+BAA+B,EAAE,SAAS,KAAK,CAAC,GAAG,MAAM,OAAO;AACjG,QAAM,kBAAkB,KAAK,CAAC;AAC9B,MAAI;AAEJ,QAAM,UAAU,MAAM,aAAa,mCAAmC,EAAE,SAAS,KAAK,CAAC;AAEvF,MAAI,iBAAiB;AACnB,iBAAa,GAAG,OAAO,YAAY,eAAe,MAAM,UAAU;AAAA,EACpE,OAAO;AACL,iBAAa,GAAG,OAAO,YAAY,UAAU;AAAA,EAC/C;AAEA,kBAAgB,uBAAuB,UAAU;AACjD,SAAO;AACT;AAQO,SAAS,qBAAqB,mBAA8C;AACjF,QAAM,wBAAwB;AAC9B,UAAQ,uBAAuB;AAAA,IAC7B,KAAK;AAAA,IACL,KAAK;AAAA,IACL,KAAK;AAAA,IACL,KAAK;AACH,aAAO;AAAA,IAET;AACE,UAAI,iCAAiC,KAAK,iBAAiB,GAAG;AAC5D,eAAO;AAAA,MACT;AAEA,aAAO;AAAA,EACX;AACF;AAOA,eAAsB,UAAyB;AAC7C,QAAM,aAAa,kCAAkC,EAAE,SAAS,KAAK,CAAC;AACxE;AAWA,eAAsB,qBAAqB,YAAoB,kBAA0C;AACvG,MAAI;AAEJ,MAAI,kBAAkB;AACpB,UAAM,cAAc,wBAAwB,wBAAwB,SAAS;AAC7E,UAAM,YAAY,MAAM,aAAa,WAAW;AAChD,gBAAY,UAAU,IAAI,CAAC,aAAa,KAAK,aAAa,QAAQ,CAAC;AAAA,EACrE,OAAO;AACL,UAAM,aAAa,MAAM,aAAa,CAAC,OAAO,QAAQ,sBAAsB,0BAA0B,MAAM,QAAQ,GAAG,EAAE,SAAS,KAAK,CAAC;AACxI,UAAM,SAAS,KAAK,MAAM,UAAU;AACpC,gBAAY;AAAA,MACV,KAAK,0BAA0B,MAAM,OAAO,CAAC,EAAE,QAAQ;AAAA,MACvD,KAAK,0BAA0B,MAAM,0BAA0B,SAAS;AAAA,IAC1E;AAAA,EACF;AAEA,cAAY,UAAU,OAAO,CAAC,aAAa,WAAW,wBAAwB,QAAQ,CAAC,CAAC;AAExF,QAAM,aAAa;AAAA,IACjB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,GAAG;AAAA,IACH;AAAA,IACA,IAAI,UAAU;AAAA,IACd,GAAI,OAAO,UAAU,IAAI,CAAC,cAAc,IAAI,CAAC;AAAA,IAC7C;AAAA,IACA;AAAA,EACF,GAAG;AAAA,IACD,SAAS;AAAA,IACT,OAAO,MAAM,gBAAgB,UAAU;AAAA,EACzC,CAAC;AACH;AAWA,eAAsB,gBAAgB,YAAmC;AACvE,QAAM,qBAAqB;AAC3B,QAAM,gBAAgB,wBAAwB,wBAAwB,WAAW;AACjF,MAAI;AACJ,MAAI,WAAW,aAAa,GAAG;AAC7B,UAAM,UAAU,MAAM,SAAS,eAAe,OAAO;AACrD,6BAAyB,QAAQ,MAAM,IAAI,EAAE,MAAM,kBAAkB;AACrE,QAAI,uBAAuB,GAAG,EAAE,MAAM,IAAI;AACxC,6BAAuB,IAAI;AAAA,IAC7B;AAAA,EACF,OAAO;AACL,6BAAyB,CAAC;AAAA,EAC5B;AAEA,QAAM,UAAU,WAAW,uBAAuB,CAAC,KAAK,IAAI,OAAO,EAAE;AACrE,QAAM,cAAc,UAAU,GAAG,OAAO,WAAW;AACnD,QAAM,oBAAoB,MAAM,aAAa,WAAW,WAAW,kCAAkC,EAAE,SAAS,KAAK,CAAC;AACtH,QAAM,iBAAiB,kBAAkB,MAAM,IAAI,EAAE,OAAO,OAAO,EAAE,IAAI,YAAY;AAErF,MAAI,eAAe;AAAA;AAAA,KAAqB,UAAU;AAAA;AAAA;AAElD,aAAW,WAAW,gBAAgB;AACpC,oBAAgB,KAAK,OAAO;AAAA;AAAA,EAC9B;AAEA,MAAI,uBAAuB,SAAS,GAAG;AACrC,oBAAgB;AAChB,eAAW,QAAQ,wBAAwB;AACzC,sBAAgB,GAAG,IAAI;AAAA;AAAA,IACzB;AAAA,EACF;AAEA,QAAM,UAAU,eAAe,cAAc,OAAO;AAEpD,QAAM,cAAc,MAAM,aAAa,kBAAkB;AAAA,IACvD,SAAS;AAAA,IACT,sBAAsB;AAAA,EACxB,CAAC;AACD,QAAM,kBAAkB,eAAe,SAAS;AAChD,MAAI,aAAa;AACf,oBAAgB,qBAAqB,wBAAwB,WAAW,sDAAsD;AAC9H,UAAM,aAAa,CAAC,QAAQ,MAAM,aAAa,GAAG;AAAA,MAChD,SAAS;AAAA,MACT,sBAAsB;AAAA,IACxB,CAAC;AAAA,EACH,OAAO;AACL,oBAAgB,6EAA6E;AAC7F,UAAM,gBAAgB,QAAQ,OAAO,QAAQ,MAAM,EAAE;AAAA,MACnD,qBAAqB,wBAAwB,WAAW;AAAA,IAC1D;AAAA,EACF;AACF;AAmBA,eAAsB,cAAc,mBAA4B,sBAA6E;AAC3I,MAAI,CAAC,mBAAmB;AACtB,UAAM,gBAAgB,QAAQ,IAAI,iBAAiB;AACnD,UAAM,gBAAgB,QAAQ,IAAI,iBAAiB;AAEnD,QAAI,iBAAiB,eAAe;AAClC,YAAM,qBAAqB,aAAa;AACxC,YAAM,cAAc,eAAe,oBAAoB;AACvD;AAAA,IACF;AAEA,UAAM,IAAI,MAAM,iCAAiC;AAAA,EACnD;AAEA,QAAM,mBAAmB,WAAW,wBAAwB,wBAAwB,YAAY,CAAC;AAEjG,WAAS,iBAAiB;AAC1B,QAAM,kBAAkB;AACxB,QAAM,kBAAkB;AACxB,QAAM,wBAAwB;AAC9B,QAAM,OAAO,cAAc;AAC3B,QAAM,OAAO,YAAY;AACzB,QAAM,OAAO,OAAO;AACpB,QAAM,OAAO,MAAM;AAEnB,QAAM,aAAa,MAAM,cAAc,iBAAiB;AACxD,QAAM,qBAAqB,UAAU;AACrC,MAAI,kBAAkB;AACpB,UAAM,8BAA8B,UAAU;AAAA,EAChD;AAEA,QAAM,gBAAgB,UAAU;AAChC,QAAM,qBAAqB,UAAU;AACrC,QAAM,UAAU,UAAU;AAC1B,QAAM,QAAQ;AACd,QAAM,uBAAuB,UAAU;AACvC,QAAM,qBAAqB,YAAY,gBAAgB;AACzD;AASA,eAAsB,qBAAqB,YAAmC;AAC5E,QAAM,gBAAgB,CAAC,gBAAgB;AACrC,gBAAY,UAAU;AAAA,EACxB,CAAC;AAED,QAAM,oBAAoB,QAAQ,EAAE,qBAAqB,KAAK,CAAC;AAC/D,QAAM,sBAAsB,QAAQ,EAAE,qBAAqB,KAAK,CAAC;AAEjE,WAAS,OAAO,iBAAwC;AACtD,oBAAgB,UAAU;AAC1B,UAAM,iBAAiB,gBAAgB,WAAW,EAAE;AACpD,QAAI,gBAAgB;AAClB,qBAAe,UAAU;AAAA,IAC3B;AAAA,EACF;AACF;AASO,SAAS,SAAS,mBAAiC;AACxD,MAAI,qBAAqB,iBAAiB,MAAM,yBAA2B;AACzE,UAAM,IAAI,MAAM,gGAAwG;AAAA,EAC1H;AACF;AAOA,eAAe,2BAA4C;AACzD,QAAM,WAAW,MAAM,MAAM,2EAA2E;AACxG,QAAM,uBAAuB,MAAM,SAAS,KAAK;AACjD,SAAO,qBAAqB,QAAQ,gBAAgB,IAAI,MAAM,wDAAwD,CAAC;AACzH;AAEA,SAAS,OAAO,SAA0B;AACxC,SAAO,QAAQ,SAAS,iBAAsB;AAChD;AAEA,SAAS,aAAa,KAAqB;AACzC,QAAM,QAAQ,IAAI,MAAM,OAAO,EAAE,OAAO,OAAO;AAC/C,SAAO,MAAM,KAAK,GAAG;AACvB;AAEA,eAAe,8BAA8B,YAAmC;AAC9E,QAAM,uBAAuB,wBAAwB,wBAAwB,gBAAgB;AAC7F,MAAI,OAAO,UAAU,GAAG;AACtB,UAAM;AAAA,MACJ,wBAAwB,wBAAwB,YAAY;AAAA,MAC5D;AAAA,MACA,EAAE,OAAO,KAAK;AAAA,IAChB;AACA,UAAM,SAAmB,wBAAwB,kBAAkB,CAAC,aAAa;AAC/E,eAAS,UAAU;AAAA,IACrB,CAAC;AAAA,EACH,OAAO;AACL,UAAM,wBAAwB,MAAM,yBAAyB;AAE7D,UAAM,SAAmB,wBAAwB,cAAc,CAAC,aAAa;AAC3E,eAAS,gBAAgB;AACzB,eAAS,UAAU;AAAA,IACrB,CAAC;AAED,UAAM,SAAiC,wBAAwB,cAAc,CAAC,aAAa;AACzF,eAAS,UAAU,IAAI;AAAA,IACzB,CAAC;AAED,QAAI,WAAW,oBAAoB,GAAG;AACpC,YAAM,GAAG,oBAAoB;AAAA,IAC/B;AAAA,EACF;AAEA,QAAM,oBAAoB;AAC5B;",
  "names": ["VersionUpdateType"]
}

361
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/ScriptUtils/version.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides functions for managing version updates in a project.\n * It includes tasks such as validating version update types, checking the state\n * of Git and GitHub CLI, updating version numbers in files, and performing\n * Git operations such as tagging and pushing.\n */\n\nimport type { PackageLockJson } from './Npm.ts';\n\nimport { getLibDebugger } from '../Debug.ts';\nimport { throwExpression } from '../Error.ts';\nimport { ObsidianPluginRepoPaths } from '../obsidian/Plugin/ObsidianPluginRepoPaths.ts';\nimport { join } from '../Path.ts';\nimport { replaceAll } from '../String.ts';\nimport { readdirPosix } from './Fs.ts';\nimport { editJson } from './JSON.ts';\nimport {\n  cp,\n  createInterface,\n  existsSync,\n  readFile,\n  rm,\n  writeFile\n} from './NodeModules.ts';\nimport {\n  editNpmShrinkWrapJson,\n  editPackageJson,\n  editPackageLockJson,\n  readPackageJson\n} from './Npm.ts';\nimport { npmRun } from './NpmRun.ts';\nimport { ObsidianDevUtilsRepoPaths } from './ObsidianDevUtilsRepoPaths.ts';\nimport {\n  execFromRoot,\n  resolvePathFromRootSafe\n} from './Root.ts';\n\n/**\n * Enum representing different types of version updates.\n */\nexport enum VersionUpdateType {\n  Beta = 'beta',\n  Invalid = 'invalid',\n  Major = 'major',\n  Manual = 'manual',\n  Minor = 'minor',\n  Patch = 'patch'\n}\n\n/**\n * Type representing the manifest file format for Obsidian plugins.\n */\nexport interface Manifest {\n  /**\n   * A minimum Obsidian version required for the plugin.\n   */\n  minAppVersion: string;\n\n  /**\n   * A version of the plugin.\n   */\n  version: string;\n}\n\n/**\n * Type representing the structure of Obsidian releases JSON.\n */\nexport interface ObsidianReleasesJson {\n  /**\n   * A name of the Obsidian release.\n   */\n  name: string;\n}\n\n/**\n * Creates a Git tag for the new version.\n *\n * @param newVersion - The new version number to use for the tag.\n * @returns A {@link Promise} that resolves when the tag has been created.\n */\nexport async function addGitTag(newVersion: string): Promise<void> {\n  await execFromRoot(`git tag -a ${newVersion} -m ${newVersion} --force`, { isQuiet: true });\n}\n\n/**\n * Adds updated files to the Git staging area and commits them with the new version message.\n *\n * @param newVersion - The new version number used as the commit message.\n * @returns A {@link Promise} that resolves when the files have been added and committed.\n */\nexport async function addUpdatedFilesToGit(newVersion: string): Promise<void> {\n  await execFromRoot(['git', 'add', '--all'], { isQuiet: true });\n  await execFromRoot(['git', 'commit', '-m', `chore: release ${newVersion}`, '--allow-empty'], { isQuiet: true });\n}\n\n/**\n * Checks if the GitHub CLI is installed on the system.\n *\n * Throws an error if the GitHub CLI is not installed.\n *\n * @throws Error if the GitHub CLI is not installed.\n */\nexport async function checkGitHubCliInstalled(): Promise<void> {\n  try {\n    await execFromRoot('gh --version', { isQuiet: true });\n  } catch {\n    throw new Error('GitHub CLI is not installed. Please install it from https://cli.github.com/');\n  }\n}\n\n/**\n * Checks if Git is installed on the system.\n *\n * Throws an error if Git is not installed.\n *\n * @throws Error if Git is not installed.\n */\nexport async function checkGitInstalled(): Promise<void> {\n  try {\n    await execFromRoot('git --version', { isQuiet: true });\n  } catch {\n    throw new Error('Git is not installed. Please install it from https://git-scm.com/');\n  }\n}\n\n/**\n * Checks if the Git repository is clean, meaning there are no uncommitted changes.\n *\n * Throws an error if the Git repository is not clean.\n *\n * @throws Error if the Git repository is not clean.\n */\nexport async function checkGitRepoClean(): Promise<void> {\n  try {\n    const stdout = await execFromRoot('git status --porcelain --untracked-files=all', { isQuiet: true });\n    if (stdout) {\n      throw new Error();\n    }\n  } catch {\n    throw new Error('Git repository is not clean. Please commit or stash your changes before releasing a new version.');\n  }\n}\n\n/**\n * Copies the updated manifest file to the distribution build folder.\n *\n * @returns A {@link Promise} that resolves when the copy operation is complete.\n */\nexport async function copyUpdatedManifest(): Promise<void> {\n  await cp(\n    resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson),\n    resolvePathFromRootSafe(join(ObsidianPluginRepoPaths.DistBuild, ObsidianPluginRepoPaths.ManifestJson)),\n    { force: true }\n  );\n}\n\n/**\n * Generates a new version string based on the current version and the specified update type.\n *\n * @param versionUpdateType - The type of version update (major, minor, patch, beta, or manual).\n * @returns A {@link Promise} that resolves to the new version string.\n * @throws Error if the current version format is invalid.\n */\nexport async function getNewVersion(versionUpdateType: string): Promise<string> {\n  const versionType = getVersionUpdateType(versionUpdateType);\n  if (versionType === VersionUpdateType.Manual) {\n    return versionUpdateType;\n  }\n\n  const packageJson = await readPackageJson();\n  const currentVersion = packageJson.version ?? '';\n\n  const match = /^(?<Major>\\d+)\\.(?<Minor>\\d+)\\.(?<Patch>\\d+)(?:-beta\\.(?<Beta>\\d+))?$/.exec(currentVersion);\n  if (!match) {\n    throw new Error(`Invalid current version format: ${currentVersion}`);\n  }\n\n  let major = Number(match.groups?.['Major'] ?? '');\n  let minor = Number(match.groups?.['Minor'] ?? '');\n  let patch = Number(match.groups?.['Patch'] ?? '');\n  let beta = Number(match.groups?.['Beta'] ?? '');\n\n  switch (versionType) {\n    case VersionUpdateType.Beta:\n      if (beta === 0) {\n        patch++;\n      }\n      beta++;\n      break;\n    case VersionUpdateType.Major:\n      major++;\n      minor = 0;\n      patch = 0;\n      beta = 0;\n      break;\n    case VersionUpdateType.Minor:\n      minor++;\n      patch = 0;\n      beta = 0;\n      break;\n    case VersionUpdateType.Patch:\n      if (beta === 0) {\n        patch++;\n      } else {\n        beta = 0;\n      }\n      break;\n    default:\n      throw new Error(`Invalid version update type: ${versionType}`);\n  }\n\n  return `${String(major)}.${String(minor)}.${String(patch)}${beta > 0 ? `-beta.${String(beta)}` : ''}`;\n}\n\n/**\n * Retrieves the release notes for a specific version from the changelog.\n *\n * @param newVersion - The new version number for which to get the release notes.\n * @returns A {@link Promise} that resolves to the release notes for the specified version.\n */\nexport async function getReleaseNotes(newVersion: string): Promise<string> {\n  const changelogPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ChangelogMd);\n  const content = await readFile(changelogPath, 'utf-8');\n  const newVersionEscaped = replaceAll(newVersion, '.', '\\\\.');\n  const match = new RegExp(`\\n## ${newVersionEscaped}\\n\\n((.|\\n)+?)\\n\\n##`).exec(content);\n  let releaseNotes = match?.[1] ? `${match[1]}\\n\\n` : '';\n\n  const tags = (await execFromRoot('git tag --sort=-creatordate', { isQuiet: true })).split(/\\r?\\n/);\n  const previousVersion = tags[1];\n  let changesUrl: string;\n\n  const repoUrl = await execFromRoot('gh repo view --json url -q .url', { isQuiet: true });\n\n  if (previousVersion) {\n    changesUrl = `${repoUrl}/compare/${previousVersion}...${newVersion}`;\n  } else {\n    changesUrl = `${repoUrl}/commits/${newVersion}`;\n  }\n\n  releaseNotes += `**Full Changelog**: ${changesUrl}`;\n  return releaseNotes;\n}\n\n/**\n * Determines the type of version update based on the input string.\n *\n * @param versionUpdateType - The input string representing the version update type.\n * @returns The corresponding `VersionUpdateType`.\n */\nexport function getVersionUpdateType(versionUpdateType: string): VersionUpdateType {\n  const versionUpdateTypeEnum = versionUpdateType as VersionUpdateType;\n  switch (versionUpdateTypeEnum) {\n    case VersionUpdateType.Beta:\n    case VersionUpdateType.Major:\n    case VersionUpdateType.Minor:\n    case VersionUpdateType.Patch:\n      return versionUpdateTypeEnum;\n\n    default:\n      if (/^\\d+\\.\\d+\\.\\d+(?:-[\\w\\d.-]+)?$/.test(versionUpdateType)) {\n        return VersionUpdateType.Manual;\n      }\n\n      return VersionUpdateType.Invalid;\n  }\n}\n\n/**\n * Pushes commits and tags to the remote Git repository.\n *\n * @returns A {@link Promise} that resolves when the push operation is complete.\n */\nexport async function gitPush(): Promise<void> {\n  await execFromRoot('git push --follow-tags --force', { isQuiet: true });\n}\n\n/**\n * Publishes a GitHub release for the new version.\n *\n * Handles the creation of a release and uploading files for either an Obsidian plugin or another project.\n *\n * @param newVersion - The new version number for the release.\n * @param isObsidianPlugin - A boolean indicating if the project is an Obsidian plugin.\n * @returns A {@link Promise} that resolves when the release has been published.\n */\nexport async function publishGitHubRelease(newVersion: string, isObsidianPlugin: boolean): Promise<void> {\n  let filePaths: string[];\n\n  if (isObsidianPlugin) {\n    const buildFolder = resolvePathFromRootSafe(ObsidianPluginRepoPaths.DistBuild);\n    const fileNames = await readdirPosix(buildFolder);\n    filePaths = fileNames.map((fileName) => join(buildFolder, fileName));\n  } else {\n    const resultJson = await execFromRoot(['npm', 'pack', '--pack-destination', ObsidianDevUtilsRepoPaths.Dist, '--json'], { isQuiet: true });\n    const result = JSON.parse(resultJson) as [{ filename: string }];\n    filePaths = [\n      join(ObsidianDevUtilsRepoPaths.Dist, result[0].filename),\n      join(ObsidianDevUtilsRepoPaths.Dist, ObsidianDevUtilsRepoPaths.StylesCss)\n    ];\n  }\n\n  filePaths = filePaths.filter((filePath) => existsSync(resolvePathFromRootSafe(filePath)));\n\n  await execFromRoot([\n    'gh',\n    'release',\n    'create',\n    newVersion,\n    ...filePaths,\n    '--title',\n    `v${newVersion}`,\n    ...(isBeta(newVersion) ? ['--prerelease'] : []),\n    '--notes-file',\n    '-'\n  ], {\n    isQuiet: true,\n    stdin: await getReleaseNotes(newVersion)\n  });\n}\n\n/**\n * Updates the changelog file with new version information and commit messages.\n *\n * This function reads the current changelog, appends new entries for the latest version,\n * and prompts the user to review the changes.\n *\n * @param newVersion - The new version number to be added to the changelog.\n * @returns A {@link Promise} that resolves when the changelog update is complete.\n */\nexport async function updateChangelog(newVersion: string): Promise<void> {\n  const HEADER_LINES_COUNT = 2;\n  const changelogPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ChangelogMd);\n  let previousChangelogLines: string[];\n  if (existsSync(changelogPath)) {\n    const content = await readFile(changelogPath, 'utf-8');\n    previousChangelogLines = content.split('\\n').slice(HEADER_LINES_COUNT);\n    if (previousChangelogLines.at(-1) === '') {\n      previousChangelogLines.pop();\n    }\n  } else {\n    previousChangelogLines = [];\n  }\n\n  const lastTag = replaceAll(previousChangelogLines[0] ?? '', '## ', '');\n  const commitRange = lastTag ? `${lastTag}..HEAD` : 'HEAD';\n  const commitMessagesStr = await execFromRoot(`git log ${commitRange} --format=%B --first-parent -z`, { isQuiet: true });\n  const commitMessages = commitMessagesStr.split('\\0').filter(Boolean).map(toSingleLine);\n\n  let newChangeLog = `# CHANGELOG\\n\\n## ${newVersion}\\n\\n`;\n\n  for (const message of commitMessages) {\n    newChangeLog += `- ${message}\\n`;\n  }\n\n  if (previousChangelogLines.length > 0) {\n    newChangeLog += '\\n';\n    for (const line of previousChangelogLines) {\n      newChangeLog += `${line}\\n`;\n    }\n  }\n\n  await writeFile(changelogPath, newChangeLog, 'utf-8');\n\n  const codeVersion = await execFromRoot('code --version', {\n    isQuiet: true,\n    shouldIgnoreExitCode: true\n  });\n  const versionDebugger = getLibDebugger('Version');\n  if (codeVersion) {\n    versionDebugger(`Please update the ${ObsidianPluginRepoPaths.ChangelogMd} file. Close Visual Studio Code when you are done...`);\n    await execFromRoot(['code', '-w', changelogPath], {\n      isQuiet: true,\n      shouldIgnoreExitCode: true\n    });\n  } else {\n    versionDebugger('Could not find Visual Studio Code in your PATH. Using console mode instead.');\n    await createInterface(process.stdin, process.stdout).question(\n      `Please update the ${ObsidianPluginRepoPaths.ChangelogMd} file. Press Enter when you are done...`\n    );\n  }\n}\n\n/**\n * Updates the version of the project based on the specified update type.\n *\n * This function performs a series of tasks to handle version updates:\n * 1. Validates the version update type.\n * 2. Checks if Git and GitHub CLI are installed.\n * 3. Verifies that the Git repository is clean.\n * 4. Runs spellcheck and linting.\n * 5. Builds the project.\n * 6. Updates version in files and changelog.\n * 7. Adds updated files to Git, tags the commit, and pushes to the repository.\n * 8. If an Obsidian plugin, copies the updated manifest and publishes a GitHub release.\n *\n * @param versionUpdateType - The type of version update to perform (major, minor, patch, beta, or x.y.z[-beta:u]).\n * @param prepareGitHubRelease - A callback function to prepare the GitHub release.\n * @returns A {@link Promise} that resolves when the version update is complete.\n */\nexport async function updateVersion(versionUpdateType?: string, prepareGitHubRelease?: (newVersion: string) => Promise<void>): Promise<void> {\n  if (!versionUpdateType) {\n    const npmOldVersion = process.env['npm_old_version'];\n    const npmNewVersion = process.env['npm_new_version'];\n\n    if (npmOldVersion && npmNewVersion) {\n      await updateVersionInFiles(npmOldVersion);\n      await updateVersion(npmNewVersion, prepareGitHubRelease);\n      return;\n    }\n\n    throw new Error('No version update type provided');\n  }\n\n  const isObsidianPlugin = existsSync(resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson));\n\n  validate(versionUpdateType);\n  await checkGitInstalled();\n  await checkGitRepoClean();\n  await checkGitHubCliInstalled();\n  await npmRun('format:check');\n  await npmRun('spellcheck');\n  await npmRun('build');\n  await npmRun('lint');\n\n  const newVersion = await getNewVersion(versionUpdateType);\n  await updateVersionInFiles(newVersion);\n  if (isObsidianPlugin) {\n    await updateVersionInFilesForPlugin(newVersion);\n  }\n\n  await updateChangelog(newVersion);\n  await addUpdatedFilesToGit(newVersion);\n  await addGitTag(newVersion);\n  await gitPush();\n  await prepareGitHubRelease?.(newVersion);\n  await publishGitHubRelease(newVersion, isObsidianPlugin);\n}\n\n/**\n * Updates the version in various files, including `package.json`, `package-lock.json`,\n * and Obsidian plugin manifests if applicable.\n *\n * @param newVersion - The new version string to update in the files.\n * @returns A {@link Promise} that resolves when the update is complete.\n */\nexport async function updateVersionInFiles(newVersion: string): Promise<void> {\n  await editPackageJson((packageJson) => {\n    packageJson.version = newVersion;\n  });\n\n  await editPackageLockJson(update, { shouldSkipIfMissing: true });\n  await editNpmShrinkWrapJson(update, { shouldSkipIfMissing: true });\n\n  function update(packageLockJson: PackageLockJson): void {\n    packageLockJson.version = newVersion;\n    const defaultPackage = packageLockJson.packages?.[''];\n    if (defaultPackage) {\n      defaultPackage.version = newVersion;\n    }\n  }\n}\n\n/**\n * Validates the version update type to ensure it is either a recognized type\n * or a valid manual version string.\n *\n * @param versionUpdateType - The version update type to validate.\n * @throws Error if the version update type is invalid.\n */\nexport function validate(versionUpdateType: string): void {\n  if (getVersionUpdateType(versionUpdateType) === VersionUpdateType.Invalid) {\n    throw new Error('Invalid version update type. Please use \\'major\\', \\'minor\\', \\'patch\\', or \\'x.y.z[-suffix]\\' format.');\n  }\n}\n\n/**\n * Fetches the latest version of Obsidian from the GitHub releases API.\n *\n * @returns A {@link Promise} that resolves to the latest version of Obsidian.\n */\nasync function getLatestObsidianVersion(): Promise<string> {\n  const response = await fetch('https://api.github.com/repos/obsidianmd/obsidian-releases/releases/latest');\n  const obsidianReleasesJson = await response.json() as Partial<ObsidianReleasesJson>;\n  return obsidianReleasesJson.name ?? throwExpression(new Error('Could not find the name of the latest Obsidian release'));\n}\n\nfunction isBeta(version: string): boolean {\n  return version.includes(VersionUpdateType.Beta);\n}\n\nfunction toSingleLine(str: string): string {\n  const lines = str.split(/\\r?\\n/).filter(Boolean);\n  return lines.join(' ');\n}\n\nasync function updateVersionInFilesForPlugin(newVersion: string): Promise<void> {\n  const manifestBetaJsonPath = resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestBetaJson);\n  if (isBeta(newVersion)) {\n    await cp(\n      resolvePathFromRootSafe(ObsidianPluginRepoPaths.ManifestJson),\n      manifestBetaJsonPath,\n      { force: true }\n    );\n    await editJson<Manifest>(ObsidianPluginRepoPaths.ManifestBetaJson, (manifest) => {\n      manifest.version = newVersion;\n    });\n  } else {\n    const latestObsidianVersion = await getLatestObsidianVersion();\n\n    await editJson<Manifest>(ObsidianPluginRepoPaths.ManifestJson, (manifest) => {\n      manifest.minAppVersion = latestObsidianVersion;\n      manifest.version = newVersion;\n    });\n\n    await editJson<Record<string, string>>(ObsidianPluginRepoPaths.VersionsJson, (versions) => {\n      versions[newVersion] = latestObsidianVersion;\n    });\n\n    if (existsSync(manifestBetaJsonPath)) {\n      await rm(manifestBetaJsonPath);\n    }\n  }\n\n  await copyUpdatedManifest();\n}\n"],
  "mappings": "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",
  "names": ["VersionUpdateType"]
}

@@ -91,17 +91,6 @@ export interface ConvertLinkOptions {
91
91
  */
92
92
  shouldUpdateFileNameAlias?: boolean;
93
93
  }
94
- /**
95
- * Wrapper for default options for {@link GenerateMarkdownLinkOptions}.
96
- */
97
- export interface GenerateMarkdownLinkDefaultOptionsWrapper {
98
- /**
99
- * A default options for generating markdown links.
100
- *
101
- * @returns A default options for generating markdown links.
102
- */
103
- defaultOptionsFn(): Partial<GenerateMarkdownLinkOptions>;
104
- }
105
94
  /**
106
95
  * Options for {@link generateMarkdownLink}.
107
96
  */
@@ -164,12 +164,10 @@ function fixFrontmatterMarkdownLinks(cache) {
164
164
  }
165
165
  function generateMarkdownLink(options) {
166
166
  const { app } = options;
167
- const configurableDefaultOptionsFn = app.fileManager.generateMarkdownLink.defaultOptionsFn ?? (() => ({}));
168
- const configurableDefaultOptions = configurableDefaultOptionsFn();
169
167
  const DEFAULT_OPTIONS = {
170
168
  isEmptyEmbedAliasAllowed: true
171
169
  };
172
- options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };
170
+ options = { ...DEFAULT_OPTIONS, ...options };
173
171
  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);
174
172
  return tempRegisterFilesAndRun(app, [targetFile], () => generateMarkdownLinkImpl(options));
175
173
  }
@@ -693,4 +691,4 @@ export {
693
691
  updateLinksInContent,
694
692
  updateLinksInFile
695
693
  };
696
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Link.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n */\n\nimport type { Link } from 'mdast';\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TFile\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\nimport type { Node } from 'unist';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\nimport { visit } from 'unist-util-visit';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { MaybeReturn } from '../Type.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll,\n  trimEnd\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport {\n  applyContentChanges,\n  applyFileChanges,\n  isCanvasChange,\n  isContentChange\n} from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  MARKDOWN_FILE_EXTENSION\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  parseMetadata,\n  tempRegisterFilesAndRun\n} from './MetadataCache.ts';\nimport {\n  getNewLinkFormat,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport {\n  isCanvasFileNodeReference,\n  referenceToFileChange\n} from './Reference.ts';\n\nconst ESCAPED_WIKILINK_DIVIDER = '\\\\|';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Regular expression for unescaped pipes.\n */\nconst UNESCAPED_WIKILINK_DIVIDER_REGEXP = /(?<!\\\\)\\|/g;\n\nconst WIKILINK_DIVIDER = '|';\n\n/**\n * A style of the link path.\n */\nexport enum LinkPathStyle {\n  /**\n   * Use the absolute path in the vault.\n   *\n   * @example `[[path/from/the/vault/root/link]]`\n   */\n  AbsolutePathInVault = 'AbsolutePathInVault',\n\n  /**\n   * Use the default link path style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Use the relative path to the note.\n   *\n   * @example `[[../../relative/path/to/link]]`\n   */\n  RelativePathToTheNote = 'RelativePathToTheNote',\n\n  /**\n   * Use the shortest path when possible.\n   *\n   * @example `[[shortest-path-to-link]]`\n   */\n  ShortestPathWhenPossible = 'ShortestPathWhenPossible'\n}\n\n/**\n * A style of the link.\n */\nexport enum LinkStyle {\n  /**\n   * Force the link to be in markdown format.\n   *\n   * @example `[alias](path/to/link.md)`\n   */\n  Markdown = 'Markdown',\n\n  /**\n   * Use the default link style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Preserve the existing link style.\n   */\n  PreserveExisting = 'PreserveExisting',\n\n  /**\n   * Force the link to be in wikilink format.\n   *\n   * @example `[[path/to/link]]`\n   * @example `[[path/to/link|alias]]`\n   */\n  Wikilink = 'Wikilink'\n}\n\nenum FinalLinkPathStyle {\n  AbsolutePathInVault = 'AbsolutePathInVault',\n  RelativePathToTheNote = 'RelativePathToTheNote',\n  ShortestPathWhenPossible = 'ShortestPathWhenPossible'\n}\n\n/**\n * Options for {@link convertLink}.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Wrapper for default options for {@link GenerateMarkdownLinkOptions}.\n */\nexport interface GenerateMarkdownLinkDefaultOptionsWrapper {\n  /**\n   * A default options for generating markdown links.\n   *\n   * @returns A default options for generating markdown links.\n   */\n  defaultOptionsFn(): Partial<GenerateMarkdownLinkOptions>;\n}\n\n/**\n * Options for {@link generateMarkdownLink}.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * An alias for the link.\n   *\n   * @example `[[alias|link]]`\n   * @example `[alias](link.md)`\n   */\n  alias?: string;\n\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   *\n   * If `true`: `![[link]]`.\n   *\n   * If `false`: `[[link]]`.\n   */\n  isEmbed?: boolean;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   *\n   * Applicable only if the result link style is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `![](foo.png)`.\n   *\n   * If `false`: `![foo](foo.png)`.\n   */\n  isEmptyEmbedAliasAllowed?: boolean;\n\n  /**\n   * Whether to allow non-existing files. Defaults to `false`.\n   *\n   * If `false` and {@link targetPathOrFile} is a non-existing file, an error will be thrown.\n   */\n  isNonExistingFileAllowed?: boolean;\n\n  /**\n   * Whether to allow a single subpath. Defaults to `true`.\n   *\n   * Applicable only if {@link targetPathOrFile} and {@link sourcePathOrFile} are the same file.\n   *\n   * If `true`: `[[#subpath]]`.\n   *\n   * If `false`: `[[source#subpath]]`\n   */\n  isSingleSubpathAllowed?: boolean;\n\n  /**\n   * A style of the link path.\n   */\n  linkPathStyle?: LinkPathStyle;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * An original link text.\n   *\n   * If provided, it will be used to infer the values of\n   *\n   * - {@link isEmbed}\n   * - {@link linkStyle}\n   * - {@link shouldUseAngleBrackets}\n   * - {@link shouldUseLeadingDotForRelativePaths}\n   * - {@link shouldUseLeadingSlashForAbsolutePaths}\n   *\n   * These inferred values will be overridden by corresponding settings if specified.\n   */\n  originalLink?: string;\n\n  /**\n   * Whether to escape the alias. Defaults to `false`.\n   *\n   * Applicable only if the result link style is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `[\\*\\*alias\\*\\*](link.md)`.\n   *\n   * If `false`: `[**alias**](link.md)`.\n   */\n  shouldEscapeAlias?: boolean;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Defaults to `false`.\n   *\n   * Applicable only if {@link isEmptyEmbedAliasAllowed} is `false`.\n   *\n   * If `true`: `[foo.png](foo.png)`.\n   *\n   * If `false`: `[foo](foo.png)`.\n   */\n  shouldIncludeAttachmentExtensionToEmbedAlias?: boolean;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`.\n   *\n   * Applicable only if {@link linkStyle} is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `[alias](<path with spaces.md>)`.\n   *\n   * If `false`: `[alias](path%20with%20spaces.md)`.\n   */\n  shouldUseAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`.\n   *\n   * Applicable only if {@link linkPathStyle} is {@link LinkPathStyle.RelativePathToTheNote}.\n   *\n   * If `true`: `[[./relative/path/to/target]]`\n   *\n   * If `false`: `[[relative/path/to/target]]`\n   */\n  shouldUseLeadingDotForRelativePaths?: boolean;\n\n  /**\n   * Indicates if the link should use a leading slash. Defaults to `false`.\n   *\n   * Applicable only if {@link linkPathStyle} is {@link LinkPathStyle.AbsolutePathInVault}.\n   *\n   * If `true`: `[[/absolute/path/to/target]]`\n   *\n   * If `false`: `[[absolute/path/to/target]]`\n   */\n  shouldUseLeadingSlashForAbsolutePaths?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * A subpath of the link.\n   *\n   * Should be empty or start with `#`.\n   *\n   * @example `[[link-without-subpath]]`\n   * @example `[[link-with-subpath#subpath]]`\n   * @example `[[link-with-subpath#subpath#nested-subpath]]`\n   */\n  subpath?: string;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * A result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * An alias of the link.\n   */\n  alias?: string;\n\n  /**\n   * An encoded URL of the link.\n   */\n  encodedUrl?: string;\n\n  /**\n   * An end offset of the link in the original text.\n   */\n  endOffset: number;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * A raw link text.\n   */\n  raw: string;\n\n  /**\n   * A start offset of the link in the original text.\n   */\n  startOffset: number;\n\n  /**\n   * A title of the link.\n   */\n  title?: string;\n\n  /**\n   * An URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for {@link shouldResetAlias}.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old target path of the link.\n   */\n  oldTargetPath: PathOrFile;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * A link path.\n   */\n  linkPath: string;\n\n  /**\n   * A subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for {@link updateLink}.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * Whether to force markdown links.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * A file associated with the link.\n   */\n  newTargetPathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldTargetPathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Options for {@link updateLinksInFile}.\n */\nexport interface UpdateLinksInFileOptions extends ProcessOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A file to update the links in.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface LinkConfig {\n  isEmbed: boolean;\n  isSingleSubpathAllowed: boolean;\n  isWikilink: boolean;\n  linkPathStyle: FinalLinkPathStyle;\n  shouldUseAngleBrackets: boolean;\n  shouldUseLeadingDotForRelativePaths: boolean;\n  shouldUseLeadingSlashForAbsolutePaths: boolean;\n}\n\ninterface TablePosition {\n  end: number;\n  start: number;\n}\n\n/**\n * Options for {@link updateLinksInContent}.\n */\ninterface UpdateLinksInContentOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A content to update the links in.\n   */\n  content: string;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A new source path or file.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source path or file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface WikiLinkNode extends Node {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);\n  if (!targetFile) {\n    return options.link.original;\n  }\n\n  return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n    app: options.app,\n    link: options.link,\n    linkStyle: options.linkStyle,\n    newSourcePathOrFile: options.newSourcePathOrFile,\n    newTargetPathOrFile: targetFile,\n    oldSourcePathOrFile: options.oldSourcePathOrFile,\n    shouldUpdateFileNameAlias: options.shouldUpdateFileNameAlias\n  }));\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editBacklinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, processOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, processOptions);\n  }\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async (abortSignal, content) => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    abortSignal.throwIfAborted();\n    const file = getFile(app, pathOrFile);\n    const cachedContent = await app.vault.cachedRead(file);\n    abortSignal.throwIfAborted();\n    if (content !== cachedContent) {\n      return null;\n    }\n\n    return await getFileChanges(cache, isCanvasFile(app, pathOrFile), linkConverter, abortSignal);\n  }, processOptions);\n}\n\n/**\n * Edits the links in a content string.\n *\n * @param app - The Obsidian application instance.\n * @param content - The content to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns The promise that resolves to the updated content.\n */\nexport async function editLinksInContent(\n  app: App,\n  content: string,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const newContent = await applyContentChanges(abortSignal, content, '', async () => {\n    const cache = await parseMetadata(app, content);\n    abortSignal.throwIfAborted();\n    const changes = await getFileChanges(cache, false, linkConverter, abortSignal);\n    abortSignal.throwIfAborted();\n    return changes;\n  });\n  abortSignal.throwIfAborted();\n\n  if (newContent === null) {\n    throw new Error('Failed to update links in content');\n  }\n\n  return newContent;\n}\n\n/**\n * Encodes a URL.\n *\n * @param url - The URL to encode.\n * @returns The encoded URL.\n */\nexport function encodeUrl(url: string): string {\n  return replaceAll(url, SPECIAL_LINK_SYMBOLS_REGEXP, ({ substring: specialLinkSymbol }) => encodeURIComponent(specialLinkSymbol));\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param sourcePathOrFile - The source path or file.\n * @param shouldAllowNonExistingFile - Whether to allow non-existing files. Defaults to `false`.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, sourcePathOrFile: PathOrFile, shouldAllowNonExistingFile = false): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  const sourcePath = getPath(app, sourcePathOrFile);\n  const file = app.metadataCache.getFirstLinkpathDest(linkPath, sourcePath);\n  if (file) {\n    return file;\n  }\n\n  if (!shouldAllowNonExistingFile) {\n    return null;\n  }\n\n  if (linkPath.startsWith('/')) {\n    return getFile(app, linkPath, true);\n  }\n\n  const fullLinkPath = join(dirname(sourcePath), `./${linkPath}`);\n\n  if (fullLinkPath.startsWith('../')) {\n    return null;\n  }\n\n  return getFile(app, fullLinkPath, true);\n}\n\n/**\n * Fixes the frontmatter markdown links in the provided metadata cache.\n *\n * @param cache - The metadata cache to fix the frontmatter markdown links in.\n * @returns Whether the frontmatter markdown links were fixed.\n */\nexport function fixFrontmatterMarkdownLinks(cache: CachedMetadata): boolean {\n  return _fixFrontmatterMarkdownLinks(cache.frontmatter, '', cache);\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const configurableDefaultOptionsFn = (app.fileManager.generateMarkdownLink as Partial<GenerateMarkdownLinkDefaultOptionsWrapper>).defaultOptionsFn\n    ?? ((): Partial<GenerateMarkdownLinkOptions> => ({}));\n  const configurableDefaultOptions = configurableDefaultOptionsFn();\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    isEmptyEmbedAliasAllowed: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...configurableDefaultOptions, ...options };\n\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n\n  return tempRegisterFilesAndRun(app, [targetFile], () => generateMarkdownLinkImpl(options));\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  const links = parseLinks(str);\n  return links[0]?.raw === str ? links[0] : null;\n}\n\n/**\n * Parses all links in a string.\n *\n * @param str - The string to parse the links in.\n * @returns The parsed links.\n */\nexport function parseLinks(str: string): ParseLinkResult[] {\n  const embedSymbolOffsets = new Set<number>();\n\n  const EMBED_LINK_PREFIX = '![';\n  const NO_EMBED_LINK_PREFIX = ' [';\n\n  const noEmbedStr = replaceAll(str, EMBED_LINK_PREFIX, (args) => {\n    embedSymbolOffsets.add(args.offset);\n    return NO_EMBED_LINK_PREFIX;\n  });\n\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(noEmbedStr);\n\n  const links: ParseLinkResult[] = [];\n  const textLinks: ParseLinkResult[] = [];\n\n  visit(root, (node: Node) => {\n    let link: ParseLinkResult;\n    switch (node.type) {\n      case 'link':\n        link = parseLinkNode(node as Link, str);\n        break;\n      case 'wikiLink':\n        link = parseWikilinkNode(node as WikiLinkNode, str);\n        break;\n      default:\n        return;\n    }\n\n    if (embedSymbolOffsets.has(link.startOffset - 1)) {\n      link.isEmbed = true;\n      link.startOffset--;\n      link.raw = `!${link.raw}`;\n    }\n    links.push(link);\n  });\n\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  let textStartOffset = 0;\n\n  for (const link of links) {\n    extractTextLinks(str, textStartOffset, link.startOffset - 1, textLinks);\n    textStartOffset = link.endOffset + 1;\n  }\n\n  extractTextLinks(str, textStartOffset, str.length - 1, textLinks);\n\n  links.push(...textLinks);\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  return links;\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  if (!displayText) {\n    return true;\n  }\n\n  const targetFile = getFile(app, targetPathOrFile, true);\n  const newSourcePath = getPath(app, newSourcePathOrFile);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const newSourceFolder = dirname(newSourcePath);\n  const oldSourceFolder = dirname(oldSourcePath);\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [targetFile.path, oldTargetPath]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(app, pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(newSourceFolder, path));\n    aliasesToReset.add(relative(oldSourceFolder, path));\n  }\n\n  for (const sourcePath of [oldSourcePath, newSourcePath]) {\n    aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));\n  }\n\n  const cleanDisplayText = replaceAll(normalizePath(displayText.split(' > ')[0] ?? ''), /^\\.\\//g, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const folder = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(folder, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link has a leading slash, possibly embed:\n * `[[/link]]`, `[title](/link)`, `[title](</link>)`,\n * `![[/link]]`, `![title](/link)`, `![title](</link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading slash\n */\nexport function testLeadingSlash(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('/') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    link,\n    linkStyle,\n    newSourcePathOrFile,\n    newTargetPathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPathOrFile,\n    shouldUpdateFileNameAlias\n  } = options;\n  if (!newTargetPathOrFile) {\n    return link.original;\n  }\n  const newTargetFile = getFile(app, newTargetPathOrFile, true);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const oldTargetPath = getPath(app, oldTargetPathOrFile ?? newTargetPathOrFile);\n  const isWikilink = shouldUseWikilinkStyle(app, link.original, linkStyle);\n\n  const { subpath } = splitSubpath(link.link);\n  let shouldKeepAlias = !shouldUpdateFileNameAlias;\n\n  if (isCanvasFile(app, newSourcePathOrFile)) {\n    if (isCanvasFileNodeReference(link)) {\n      return newTargetFile.path + subpath;\n    }\n  }\n\n  const parseLinkResult = parseLink(link.original);\n  let alias: string | undefined;\n\n  if (isWikilink && parseLinkResult?.alias) {\n    alias = parseLinkResult.alias;\n    shouldKeepAlias = true;\n  }\n\n  alias ??= shouldResetAlias(normalizeOptionalProperties<ShouldResetAliasOptions>({\n      app,\n      displayText: link.displayText,\n      isWikilink,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      oldTargetPath,\n      targetPathOrFile: newTargetFile\n    }))\n    ? undefined\n    : parseLinkResult?.alias;\n\n  if (!shouldKeepAlias) {\n    if (alias === basename(oldTargetPath, extname(oldTargetPath))) {\n      alias = newTargetFile.basename;\n    } else if (alias === basename(oldTargetPath)) {\n      alias = newTargetFile.name;\n    }\n  }\n\n  const newLink = generateMarkdownLink(normalizeOptionalProperties<GenerateMarkdownLinkOptions>({\n    alias,\n    app,\n    isSingleSubpathAllowed: oldSourcePath === oldTargetPath && !!parseLinkResult?.alias,\n    linkStyle,\n    originalLink: link.original,\n    sourcePathOrFile: newSourcePathOrFile,\n    subpath,\n    targetPathOrFile: newTargetFile\n  }));\n  return newLink;\n}\n\n/**\n * Updates the links in a content string based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves to the content with updated links.\n */\nexport async function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<string> {\n  const {\n    app,\n    content,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  return await editLinksInContent(app, content, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  });\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  if (isCanvasFile(app, newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n    return;\n  }\n\n  await editLinks(app, newSourcePathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  }, options);\n}\n\nfunction _fixFrontmatterMarkdownLinks(value: unknown, key: string, cache: CachedMetadata): boolean {\n  if (typeof value === 'string') {\n    const parseLinkResult = parseLink(value);\n    if (!parseLinkResult || parseLinkResult.isWikilink || parseLinkResult.isExternal) {\n      return false;\n    }\n\n    cache.frontmatterLinks ??= [];\n    let link = cache.frontmatterLinks.find((frontmatterLink) => frontmatterLink.key === key);\n\n    if (!link) {\n      link = {\n        key,\n        link: '',\n        original: ''\n      };\n      cache.frontmatterLinks.push(link);\n    }\n\n    link.link = parseLinkResult.url;\n    link.original = value;\n    if (parseLinkResult.alias !== undefined) {\n      link.displayText = parseLinkResult.alias;\n    }\n\n    return true;\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return false;\n  }\n\n  let hasFrontmatterLinks = false;\n\n  for (const [childKey, childValue] of Object.entries(value as GenericObject)) {\n    const hasChildFrontmatterLinks = _fixFrontmatterMarkdownLinks(childValue, key ? `${key}.${childKey}` : childKey, cache);\n    hasFrontmatterLinks ||= hasChildFrontmatterLinks;\n  }\n\n  return hasFrontmatterLinks;\n}\n\nfunction extractTextLinks(str: string, startOffset: number, endOffset: number, textLinks: ParseLinkResult[]): void {\n  if (startOffset > endOffset) {\n    return;\n  }\n\n  const textPart = str.slice(startOffset, endOffset + 1);\n  replaceAll(textPart, /(?<Url>\\S+)/g, (args, url) => {\n    if (!isUrl(url)) {\n      return;\n    }\n\n    textLinks.push({\n      encodedUrl: encodeUrl(url),\n      endOffset: startOffset + args.offset + url.length,\n      hasAngleBrackets: false,\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      raw: url,\n      startOffset: startOffset + args.offset,\n      url\n    });\n  });\n}\n\nfunction generateLinkText(app: App, targetFile: TFile, sourcePath: string, subpath: string, config: LinkConfig): string {\n  if (sourcePath === '/') {\n    sourcePath = '';\n  }\n\n  let linkText: string;\n\n  if (targetFile.path === sourcePath && subpath && config.isSingleSubpathAllowed) {\n    linkText = '';\n  } else {\n    switch (config.linkPathStyle) {\n      case FinalLinkPathStyle.AbsolutePathInVault:\n        linkText = targetFile.path;\n        if (config.shouldUseLeadingSlashForAbsolutePaths && !linkText.startsWith('/')) {\n          linkText = `/${linkText}`;\n        }\n        break;\n      case FinalLinkPathStyle.RelativePathToTheNote:\n        linkText = relative(dirname(sourcePath), targetFile.path);\n        if (config.shouldUseLeadingDotForRelativePaths && !linkText.startsWith('.')) {\n          linkText = `./${linkText}`;\n        }\n        break;\n      case FinalLinkPathStyle.ShortestPathWhenPossible: {\n        const shortestName = isMarkdownFile(app, targetFile) ? targetFile.basename : targetFile.name;\n        const matchedFiles = app.metadataCache.getLinkpathDest(shortestName, sourcePath);\n        linkText = matchedFiles.length === 1 && matchedFiles[0] === targetFile ? targetFile.name : targetFile.path;\n        break;\n      }\n      default:\n        throw new Error(`Invalid link path style: ${config.linkPathStyle as string}.`);\n    }\n  }\n\n  linkText = config.isWikilink ? trimEnd(linkText, `.${MARKDOWN_FILE_EXTENSION}`) : linkText;\n  linkText += subpath;\n\n  return linkText;\n}\n\nfunction generateMarkdownLinkImpl(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n  const sourcePath = getPath(app, options.sourcePathOrFile);\n  const subpath = options.subpath ?? '';\n\n  const linkConfig = getLinkConfig(options, targetFile);\n  const linkText = generateLinkText(app, targetFile, sourcePath, subpath, linkConfig);\n\n  return linkConfig.isWikilink\n    ? generateWikiLink(linkText, options.alias, linkConfig.isEmbed)\n    : generateMarkdownStyleLink(linkText, targetFile, options, linkConfig);\n}\n\nfunction generateMarkdownStyleLink(linkText: string, targetFile: TFile, options: GenerateMarkdownLinkOptions, config: LinkConfig): string {\n  const { app } = options;\n  const embedPrefix = config.isEmbed ? '!' : '';\n\n  const processedLinkText = config.shouldUseAngleBrackets\n    ? `<${linkText}>`\n    : encodeUrl(linkText);\n\n  let alias = options.alias ?? '';\n  if (!alias && (!config.isEmbed || !options.isEmptyEmbedAliasAllowed)) {\n    alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || isMarkdownFile(app, targetFile)\n      ? targetFile.basename\n      : targetFile.name;\n  }\n\n  const escapedAlias = options.shouldEscapeAlias ? replaceAll(alias, SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&') : alias;\n  return `${embedPrefix}[${escapedAlias}](${processedLinkText})`;\n}\n\nfunction generateWikiLink(linkText: string, alias: string | undefined, isEmbed: boolean): string {\n  const embedPrefix = isEmbed ? '!' : '';\n  const normalizedAlias = alias ?? '';\n\n  if (normalizedAlias && normalizedAlias.toLowerCase() === linkText.toLowerCase()) {\n    return `${embedPrefix}[[${normalizedAlias}]]`;\n  }\n\n  const aliasPart = normalizedAlias ? `|${normalizedAlias}` : '';\n  return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n}\n\nasync function getFileChanges(\n  cache: CachedMetadata | null,\n  isCanvasFileCache: boolean,\n  linkConverter: (link: Reference, abortSignal: AbortSignal) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<FileChange[]> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  if (!cache) {\n    return [];\n  }\n\n  const changes: FileChange[] = [];\n\n  const tablePositions: TablePosition[] = (cache.sections ?? []).filter((section) => section.type === 'table').map((section) => ({\n    end: section.position.end.offset,\n    start: section.position.start.offset\n  }));\n\n  for (const link of getAllLinks(cache)) {\n    abortSignal.throwIfAborted();\n    const newContent = await linkConverter(link, abortSignal);\n    abortSignal.throwIfAborted();\n    if (newContent === undefined) {\n      continue;\n    }\n\n    const fileChange = referenceToFileChange(link, newContent);\n\n    if (isCanvasFileCache) {\n      if (isCanvasChange(fileChange)) {\n        changes.push(fileChange);\n      } else {\n        const message = 'Unsupported file change';\n        console.error(message, fileChange);\n        throw new Error(message);\n      }\n    } else {\n      if (shouldEscapeWikilinkDivider(fileChange, tablePositions)) {\n        fileChange.newContent = fileChange.newContent.replaceAll(UNESCAPED_WIKILINK_DIVIDER_REGEXP, ESCAPED_WIKILINK_DIVIDER);\n      }\n\n      changes.push(fileChange);\n    }\n  }\n  return changes;\n}\n\nfunction getFinalLinkPathStyle(app: App, linkPathStyle?: LinkPathStyle): FinalLinkPathStyle {\n  switch (linkPathStyle ?? LinkPathStyle.ObsidianSettingsDefault) {\n    case LinkPathStyle.AbsolutePathInVault:\n      return FinalLinkPathStyle.AbsolutePathInVault;\n    case LinkPathStyle.RelativePathToTheNote:\n      return FinalLinkPathStyle.RelativePathToTheNote;\n    case LinkPathStyle.ShortestPathWhenPossible:\n      return FinalLinkPathStyle.ShortestPathWhenPossible;\n    case LinkPathStyle.ObsidianSettingsDefault: {\n      const newLinkFormat = getNewLinkFormat(app);\n      switch (newLinkFormat) {\n        case 'absolute':\n          return FinalLinkPathStyle.AbsolutePathInVault;\n        case 'relative':\n          return FinalLinkPathStyle.RelativePathToTheNote;\n        case 'shortest':\n          return FinalLinkPathStyle.ShortestPathWhenPossible;\n        default:\n          throw new Error(`Invalid link format: ${newLinkFormat as string}.`);\n      }\n    }\n    default:\n      throw new Error(`Invalid link path style: ${linkPathStyle as string}.`);\n  }\n}\n\nfunction getLinkConfig(options: GenerateMarkdownLinkOptions, targetFile: TFile): LinkConfig {\n  const { app } = options;\n  return {\n    isEmbed: options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(app, targetFile),\n    isSingleSubpathAllowed: options.isSingleSubpathAllowed ?? true,\n    isWikilink: shouldUseWikilinkStyle(app, options.originalLink, options.linkStyle),\n    linkPathStyle: getFinalLinkPathStyle(app, options.linkPathStyle),\n    shouldUseAngleBrackets: options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingDotForRelativePaths: options.shouldUseLeadingDotForRelativePaths\n      ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingSlashForAbsolutePaths: options.shouldUseLeadingSlashForAbsolutePaths\n      ?? (options.originalLink ? testLeadingSlash(options.originalLink) : undefined) ?? false\n  };\n}\n\nfunction getRawLink(node: Node, str: string): string {\n  return str.slice(node.position?.start.offset ?? 0, node.position?.end.offset ?? 0);\n}\n\nfunction parseLinkNode(node: Link, str: string): ParseLinkResult {\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const raw = getRawLink(node, str);\n  const aliasNodeStartOffset = node.children[0]?.position?.start.offset ?? 1;\n  const aliasNodeEndOffset = node.children.at(-1)?.position?.end.offset ?? 1;\n  const rawUrl = str.slice(aliasNodeEndOffset + LINK_ALIAS_SUFFIX.length, (node.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n  const hasAngleBrackets = raw.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n  const isExternal = isUrl(node.url);\n  let url = node.url;\n  if (!isExternal && !hasAngleBrackets) {\n    try {\n      url = decodeURIComponent(url);\n    } catch (error) {\n      console.error(`Failed to decode URL ${url}`, error);\n    }\n  }\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: aliasNodeStartOffset < aliasNodeEndOffset ? str.slice(aliasNodeStartOffset, aliasNodeEndOffset) : undefined,\n    encodedUrl: isExternal ? encodeUrl(url) : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    hasAngleBrackets,\n    isEmbed: false,\n    isExternal,\n    isWikilink: false,\n    raw,\n    startOffset: node.position?.start.offset ?? 0,\n    title: node.title ?? undefined,\n    url\n  });\n}\n\nfunction parseWikilinkNode(node: WikiLinkNode, str: string): ParseLinkResult {\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: str.includes(WIKILINK_DIVIDER) ? node.data.alias : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    isEmbed: false,\n    isExternal: false,\n    isWikilink: true,\n    raw: getRawLink(node, str),\n    startOffset: node.position?.start.offset ?? 0,\n    url: node.value\n  });\n}\n\nfunction shouldEscapeWikilinkDivider(fileChange: FileChange, tablePositions: TablePosition[]): boolean {\n  if (!isContentChange(fileChange)) {\n    return false;\n  }\n\n  if (!UNESCAPED_WIKILINK_DIVIDER_REGEXP.test(fileChange.newContent)) {\n    return false;\n  }\n\n  return tablePositions.some((tablePosition) =>\n    tablePosition.start <= fileChange.reference.position.start.offset && fileChange.reference.position.end.offset <= tablePosition.end\n  );\n}\n\nfunction shouldUseWikilinkStyle(app: App, originalLink?: string, linkStyle?: LinkStyle): boolean {\n  switch (linkStyle ?? LinkStyle.PreserveExisting) {\n    case LinkStyle.Markdown:\n      return false;\n    case LinkStyle.ObsidianSettingsDefault:\n      return shouldUseWikilinks(app);\n    case LinkStyle.PreserveExisting:\n      return originalLink === undefined ? shouldUseWikilinks(app) : testWikilink(originalLink);\n    case LinkStyle.Wikilink:\n      return true;\n    default:\n      throw new Error(`Invalid link style: ${linkStyle as string}.`);\n  }\n}\n"],
  "mappings": "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",
  "names": ["LinkPathStyle", "LinkStyle"]
}

694
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/Link.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * This module provides utilities for handling and updating links within Obsidian vaults. It includes\n * functions to split paths, update links in files, and generate markdown links with various options.\n */\n\nimport type { Link } from 'mdast';\nimport type {\n  App,\n  CachedMetadata,\n  Reference,\n  TFile\n} from 'obsidian';\nimport type { Promisable } from 'type-fest';\nimport type { Node } from 'unist';\n\nimport {\n  normalizePath,\n  parseLinktext\n} from 'obsidian';\nimport { InternalPluginName } from 'obsidian-typings/implementations';\nimport { remark } from 'remark';\nimport remarkParse from 'remark-parse';\nimport { wikiLinkPlugin } from 'remark-wiki-link';\nimport { visit } from 'unist-util-visit';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { MaybeReturn } from '../Type.ts';\nimport type { FileChange } from './FileChange.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { abortSignalNever } from '../AbortController.ts';\nimport {\n  normalizeOptionalProperties,\n  toJson\n} from '../ObjectUtils.ts';\nimport {\n  basename,\n  dirname,\n  extname,\n  join,\n  relative\n} from '../Path.ts';\nimport {\n  normalize,\n  replaceAll,\n  trimEnd\n} from '../String.ts';\nimport { isUrl } from '../url.ts';\nimport {\n  applyContentChanges,\n  applyFileChanges,\n  isCanvasChange,\n  isContentChange\n} from './FileChange.ts';\nimport {\n  getFile,\n  getPath,\n  isCanvasFile,\n  isMarkdownFile,\n  MARKDOWN_FILE_EXTENSION\n} from './FileSystem.ts';\nimport {\n  getAllLinks,\n  getBacklinksForFileSafe,\n  getCacheSafe,\n  parseMetadata,\n  tempRegisterFilesAndRun\n} from './MetadataCache.ts';\nimport {\n  getNewLinkFormat,\n  shouldUseWikilinks\n} from './ObsidianSettings.ts';\nimport {\n  isCanvasFileNodeReference,\n  referenceToFileChange\n} from './Reference.ts';\n\nconst ESCAPED_WIKILINK_DIVIDER = '\\\\|';\n\n/**\n * Regular expression for special link symbols.\n */\n// eslint-disable-next-line no-control-regex\nconst SPECIAL_LINK_SYMBOLS_REGEXP = /[\\\\\\x00\\x08\\x0B\\x0C\\x0E-\\x1F ]/g;\n\n/**\n * Regular expression for special markdown link symbols.\n */\nconst SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX = /[\\\\[\\]<>_*~=`$]/g;\n\n/**\n * Regular expression for unescaped pipes.\n */\nconst UNESCAPED_WIKILINK_DIVIDER_REGEXP = /(?<!\\\\)\\|/g;\n\nconst WIKILINK_DIVIDER = '|';\n\n/**\n * A style of the link path.\n */\nexport enum LinkPathStyle {\n  /**\n   * Use the absolute path in the vault.\n   *\n   * @example `[[path/from/the/vault/root/link]]`\n   */\n  AbsolutePathInVault = 'AbsolutePathInVault',\n\n  /**\n   * Use the default link path style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Use the relative path to the note.\n   *\n   * @example `[[../../relative/path/to/link]]`\n   */\n  RelativePathToTheNote = 'RelativePathToTheNote',\n\n  /**\n   * Use the shortest path when possible.\n   *\n   * @example `[[shortest-path-to-link]]`\n   */\n  ShortestPathWhenPossible = 'ShortestPathWhenPossible'\n}\n\n/**\n * A style of the link.\n */\nexport enum LinkStyle {\n  /**\n   * Force the link to be in markdown format.\n   *\n   * @example `[alias](path/to/link.md)`\n   */\n  Markdown = 'Markdown',\n\n  /**\n   * Use the default link style defined in Obsidian settings.\n   */\n  ObsidianSettingsDefault = 'ObsidianSettingsDefault',\n\n  /**\n   * Preserve the existing link style.\n   */\n  PreserveExisting = 'PreserveExisting',\n\n  /**\n   * Force the link to be in wikilink format.\n   *\n   * @example `[[path/to/link]]`\n   * @example `[[path/to/link|alias]]`\n   */\n  Wikilink = 'Wikilink'\n}\n\nenum FinalLinkPathStyle {\n  AbsolutePathInVault = 'AbsolutePathInVault',\n  RelativePathToTheNote = 'RelativePathToTheNote',\n  ShortestPathWhenPossible = 'ShortestPathWhenPossible'\n}\n\n/**\n * Options for {@link convertLink}.\n */\nexport interface ConvertLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Options for {@link generateMarkdownLink}.\n */\nexport interface GenerateMarkdownLinkOptions {\n  /**\n   * An alias for the link.\n   *\n   * @example `[[alias|link]]`\n   * @example `[alias](link.md)`\n   */\n  alias?: string;\n\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * Indicates if the link should be embedded. If not provided, it will be inferred based on the file type.\n   *\n   * If `true`: `![[link]]`.\n   *\n   * If `false`: `[[link]]`.\n   */\n  isEmbed?: boolean;\n\n  /**\n   * Whether to allow an empty alias for embeds. Defaults to `true`.\n   *\n   * Applicable only if the result link style is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `![](foo.png)`.\n   *\n   * If `false`: `![foo](foo.png)`.\n   */\n  isEmptyEmbedAliasAllowed?: boolean;\n\n  /**\n   * Whether to allow non-existing files. Defaults to `false`.\n   *\n   * If `false` and {@link targetPathOrFile} is a non-existing file, an error will be thrown.\n   */\n  isNonExistingFileAllowed?: boolean;\n\n  /**\n   * Whether to allow a single subpath. Defaults to `true`.\n   *\n   * Applicable only if {@link targetPathOrFile} and {@link sourcePathOrFile} are the same file.\n   *\n   * If `true`: `[[#subpath]]`.\n   *\n   * If `false`: `[[source#subpath]]`\n   */\n  isSingleSubpathAllowed?: boolean;\n\n  /**\n   * A style of the link path.\n   */\n  linkPathStyle?: LinkPathStyle;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * An original link text.\n   *\n   * If provided, it will be used to infer the values of\n   *\n   * - {@link isEmbed}\n   * - {@link linkStyle}\n   * - {@link shouldUseAngleBrackets}\n   * - {@link shouldUseLeadingDotForRelativePaths}\n   * - {@link shouldUseLeadingSlashForAbsolutePaths}\n   *\n   * These inferred values will be overridden by corresponding settings if specified.\n   */\n  originalLink?: string;\n\n  /**\n   * Whether to escape the alias. Defaults to `false`.\n   *\n   * Applicable only if the result link style is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `[\\*\\*alias\\*\\*](link.md)`.\n   *\n   * If `false`: `[**alias**](link.md)`.\n   */\n  shouldEscapeAlias?: boolean;\n\n  /**\n   * Whether to include the attachment extension in the embed alias. Defaults to `false`.\n   *\n   * Applicable only if {@link isEmptyEmbedAliasAllowed} is `false`.\n   *\n   * If `true`: `[foo.png](foo.png)`.\n   *\n   * If `false`: `[foo](foo.png)`.\n   */\n  shouldIncludeAttachmentExtensionToEmbedAlias?: boolean;\n\n  /**\n   * Indicates if the link should use angle brackets. Defaults to `false`.\n   *\n   * Applicable only if {@link linkStyle} is {@link LinkStyle.Markdown}.\n   *\n   * If `true`: `[alias](<path with spaces.md>)`.\n   *\n   * If `false`: `[alias](path%20with%20spaces.md)`.\n   */\n  shouldUseAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link should use a leading dot. Defaults to `false`.\n   *\n   * Applicable only if {@link linkPathStyle} is {@link LinkPathStyle.RelativePathToTheNote}.\n   *\n   * If `true`: `[[./relative/path/to/target]]`\n   *\n   * If `false`: `[[relative/path/to/target]]`\n   */\n  shouldUseLeadingDotForRelativePaths?: boolean;\n\n  /**\n   * Indicates if the link should use a leading slash. Defaults to `false`.\n   *\n   * Applicable only if {@link linkPathStyle} is {@link LinkPathStyle.AbsolutePathInVault}.\n   *\n   * If `true`: `[[/absolute/path/to/target]]`\n   *\n   * If `false`: `[[absolute/path/to/target]]`\n   */\n  shouldUseLeadingSlashForAbsolutePaths?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  sourcePathOrFile: PathOrFile;\n\n  /**\n   * A subpath of the link.\n   *\n   * Should be empty or start with `#`.\n   *\n   * @example `[[link-without-subpath]]`\n   * @example `[[link-with-subpath#subpath]]`\n   * @example `[[link-with-subpath#subpath#nested-subpath]]`\n   */\n  subpath?: string;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * A result of parsing a link.\n */\nexport interface ParseLinkResult {\n  /**\n   * An alias of the link.\n   */\n  alias?: string;\n\n  /**\n   * An encoded URL of the link.\n   */\n  encodedUrl?: string;\n\n  /**\n   * An end offset of the link in the original text.\n   */\n  endOffset: number;\n\n  /**\n   * Indicates if the link has angle brackets.\n   */\n  hasAngleBrackets?: boolean;\n\n  /**\n   * Indicates if the link is an embed link.\n   */\n  isEmbed: boolean;\n\n  /**\n   * Indicates if the link is external.\n   */\n  isExternal: boolean;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink: boolean;\n\n  /**\n   * A raw link text.\n   */\n  raw: string;\n\n  /**\n   * A start offset of the link in the original text.\n   */\n  startOffset: number;\n\n  /**\n   * A title of the link.\n   */\n  title?: string;\n\n  /**\n   * An URL of the link.\n   */\n  url: string;\n}\n\n/**\n * Options for {@link shouldResetAlias}.\n */\nexport interface ShouldResetAliasOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A display text of the link.\n   */\n  displayText: string | undefined;\n\n  /**\n   * Indicates if the link is a wikilink.\n   */\n  isWikilink?: boolean;\n\n  /**\n   * A source path of the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old target path of the link.\n   */\n  oldTargetPath: PathOrFile;\n\n  /**\n   * A target path or file.\n   */\n  targetPathOrFile: PathOrFile;\n}\n\n/**\n * Splits a link into its link path and subpath.\n */\nexport interface SplitSubpathResult {\n  /**\n   * A link path.\n   */\n  linkPath: string;\n\n  /**\n   * A subpath.\n   */\n  subpath: string;\n}\n\n/**\n * Options for {@link updateLink}.\n */\nexport interface UpdateLinkOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A reference for the link.\n   */\n  link: Reference;\n\n  /**\n   * Whether to force markdown links.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A source file containing the link.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * A file associated with the link.\n   */\n  newTargetPathOrFile: PathOrFile;\n\n  /**\n   * An old source file containing the link.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldTargetPathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\n/**\n * Options for {@link updateLinksInFile}.\n */\nexport interface UpdateLinksInFileOptions extends ProcessOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A file to update the links in.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old path of the file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias. Defaults to `true`.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface LinkConfig {\n  isEmbed: boolean;\n  isSingleSubpathAllowed: boolean;\n  isWikilink: boolean;\n  linkPathStyle: FinalLinkPathStyle;\n  shouldUseAngleBrackets: boolean;\n  shouldUseLeadingDotForRelativePaths: boolean;\n  shouldUseLeadingSlashForAbsolutePaths: boolean;\n}\n\ninterface TablePosition {\n  end: number;\n  start: number;\n}\n\n/**\n * Options for {@link updateLinksInContent}.\n */\ninterface UpdateLinksInContentOptions {\n  /**\n   * An Obsidian app instance.\n   */\n  app: App;\n\n  /**\n   * A content to update the links in.\n   */\n  content: string;\n\n  /**\n   * A style of the link.\n   */\n  linkStyle?: LinkStyle;\n\n  /**\n   * A new source path or file.\n   */\n  newSourcePathOrFile: PathOrFile;\n\n  /**\n   * An old source path or file.\n   */\n  oldSourcePathOrFile?: PathOrFile;\n\n  /**\n   * Whether to update only embedded links.\n   */\n  shouldUpdateEmbedOnlyLinks?: boolean;\n\n  /**\n   * Whether to update file name alias.\n   */\n  shouldUpdateFileNameAlias?: boolean;\n}\n\ninterface WikiLinkNode extends Node {\n  data: {\n    alias: string;\n  };\n  value: string;\n}\n\n/**\n * Converts a link to a new path.\n *\n * @param options - The options for converting the link.\n * @returns The converted link.\n */\nexport function convertLink(options: ConvertLinkOptions): string {\n  const targetFile = extractLinkFile(options.app, options.link, options.oldSourcePathOrFile ?? options.newSourcePathOrFile);\n  if (!targetFile) {\n    return options.link.original;\n  }\n\n  return updateLink(normalizeOptionalProperties<UpdateLinkOptions>({\n    app: options.app,\n    link: options.link,\n    linkStyle: options.linkStyle,\n    newSourcePathOrFile: options.newSourcePathOrFile,\n    newTargetPathOrFile: targetFile,\n    oldSourcePathOrFile: options.oldSourcePathOrFile,\n    shouldUpdateFileNameAlias: options.shouldUpdateFileNameAlias\n  }));\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editBacklinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  const backlinks = await getBacklinksForFileSafe(app, pathOrFile, processOptions);\n  for (const backlinkNotePath of backlinks.keys()) {\n    const currentLinks = backlinks.get(backlinkNotePath) ?? [];\n    const linkJsons = new Set<string>(currentLinks.map((link) => toJson(link)));\n    await editLinks(app, backlinkNotePath, (link) => {\n      const linkJson = toJson(link);\n      if (!linkJsons.has(linkJson)) {\n        return;\n      }\n\n      return linkConverter(link);\n    }, processOptions);\n  }\n}\n\n/**\n * Edits the backlinks for a file or path.\n *\n * @param app - The Obsidian application instance.\n * @param pathOrFile - The path or file to edit the backlinks for.\n * @param linkConverter - The function that converts each link.\n * @param processOptions - Optional options for retrying the operation.\n * @returns A {@link Promise} that resolves when the backlinks have been edited.\n */\nexport async function editLinks(\n  app: App,\n  pathOrFile: PathOrFile,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  processOptions: ProcessOptions = {}\n): Promise<void> {\n  await applyFileChanges(app, pathOrFile, async (abortSignal, content) => {\n    const cache = await getCacheSafe(app, pathOrFile);\n    abortSignal.throwIfAborted();\n    const file = getFile(app, pathOrFile);\n    const cachedContent = await app.vault.cachedRead(file);\n    abortSignal.throwIfAborted();\n    if (content !== cachedContent) {\n      return null;\n    }\n\n    return await getFileChanges(cache, isCanvasFile(app, pathOrFile), linkConverter, abortSignal);\n  }, processOptions);\n}\n\n/**\n * Edits the links in a content string.\n *\n * @param app - The Obsidian application instance.\n * @param content - The content to edit the links in.\n * @param linkConverter - The function that converts each link.\n * @param abortSignal - The abort signal to control the execution of the function.\n * @returns The promise that resolves to the updated content.\n */\nexport async function editLinksInContent(\n  app: App,\n  content: string,\n  linkConverter: (link: Reference) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<string> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  const newContent = await applyContentChanges(abortSignal, content, '', async () => {\n    const cache = await parseMetadata(app, content);\n    abortSignal.throwIfAborted();\n    const changes = await getFileChanges(cache, false, linkConverter, abortSignal);\n    abortSignal.throwIfAborted();\n    return changes;\n  });\n  abortSignal.throwIfAborted();\n\n  if (newContent === null) {\n    throw new Error('Failed to update links in content');\n  }\n\n  return newContent;\n}\n\n/**\n * Encodes a URL.\n *\n * @param url - The URL to encode.\n * @returns The encoded URL.\n */\nexport function encodeUrl(url: string): string {\n  return replaceAll(url, SPECIAL_LINK_SYMBOLS_REGEXP, ({ substring: specialLinkSymbol }) => encodeURIComponent(specialLinkSymbol));\n}\n\n/**\n * Extracts the file associated with a link.\n *\n * @param app - The Obsidian application instance.\n * @param link - The reference cache for the link.\n * @param sourcePathOrFile - The source path or file.\n * @param shouldAllowNonExistingFile - Whether to allow non-existing files. Defaults to `false`.\n * @returns The file associated with the link, or null if not found.\n */\nexport function extractLinkFile(app: App, link: Reference, sourcePathOrFile: PathOrFile, shouldAllowNonExistingFile = false): null | TFile {\n  const { linkPath } = splitSubpath(link.link);\n  const sourcePath = getPath(app, sourcePathOrFile);\n  const file = app.metadataCache.getFirstLinkpathDest(linkPath, sourcePath);\n  if (file) {\n    return file;\n  }\n\n  if (!shouldAllowNonExistingFile) {\n    return null;\n  }\n\n  if (linkPath.startsWith('/')) {\n    return getFile(app, linkPath, true);\n  }\n\n  const fullLinkPath = join(dirname(sourcePath), `./${linkPath}`);\n\n  if (fullLinkPath.startsWith('../')) {\n    return null;\n  }\n\n  return getFile(app, fullLinkPath, true);\n}\n\n/**\n * Fixes the frontmatter markdown links in the provided metadata cache.\n *\n * @param cache - The metadata cache to fix the frontmatter markdown links in.\n * @returns Whether the frontmatter markdown links were fixed.\n */\nexport function fixFrontmatterMarkdownLinks(cache: CachedMetadata): boolean {\n  return _fixFrontmatterMarkdownLinks(cache.frontmatter, '', cache);\n}\n\n/**\n * Generates a markdown link based on the provided parameters.\n *\n * @param options - The options for generating the markdown link.\n * @returns The generated markdown link.\n */\nexport function generateMarkdownLink(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n\n  const DEFAULT_OPTIONS: Partial<GenerateMarkdownLinkOptions> = {\n    isEmptyEmbedAliasAllowed: true\n  };\n\n  options = { ...DEFAULT_OPTIONS, ...options };\n\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n\n  return tempRegisterFilesAndRun(app, [targetFile], () => generateMarkdownLinkImpl(options));\n}\n\n/**\n * Parses a link into its components.\n *\n * @param str - The link to parse.\n * @returns The parsed link.\n */\nexport function parseLink(str: string): null | ParseLinkResult {\n  const links = parseLinks(str);\n  return links[0]?.raw === str ? links[0] : null;\n}\n\n/**\n * Parses all links in a string.\n *\n * @param str - The string to parse the links in.\n * @returns The parsed links.\n */\nexport function parseLinks(str: string): ParseLinkResult[] {\n  const embedSymbolOffsets = new Set<number>();\n\n  const EMBED_LINK_PREFIX = '![';\n  const NO_EMBED_LINK_PREFIX = ' [';\n\n  const noEmbedStr = replaceAll(str, EMBED_LINK_PREFIX, (args) => {\n    embedSymbolOffsets.add(args.offset);\n    return NO_EMBED_LINK_PREFIX;\n  });\n\n  const processor = remark().use(remarkParse).use(wikiLinkPlugin, { aliasDivider: WIKILINK_DIVIDER });\n  const root = processor.parse(noEmbedStr);\n\n  const links: ParseLinkResult[] = [];\n  const textLinks: ParseLinkResult[] = [];\n\n  visit(root, (node: Node) => {\n    let link: ParseLinkResult;\n    switch (node.type) {\n      case 'link':\n        link = parseLinkNode(node as Link, str);\n        break;\n      case 'wikiLink':\n        link = parseWikilinkNode(node as WikiLinkNode, str);\n        break;\n      default:\n        return;\n    }\n\n    if (embedSymbolOffsets.has(link.startOffset - 1)) {\n      link.isEmbed = true;\n      link.startOffset--;\n      link.raw = `!${link.raw}`;\n    }\n    links.push(link);\n  });\n\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  let textStartOffset = 0;\n\n  for (const link of links) {\n    extractTextLinks(str, textStartOffset, link.startOffset - 1, textLinks);\n    textStartOffset = link.endOffset + 1;\n  }\n\n  extractTextLinks(str, textStartOffset, str.length - 1, textLinks);\n\n  links.push(...textLinks);\n  links.sort((a, b) => a.startOffset - b.startOffset);\n\n  return links;\n}\n\n/**\n * Determines if the alias of a link should be reset.\n *\n * @param options - The options for determining if the alias should be reset.\n * @returns Whether the alias should be reset.\n */\nexport function shouldResetAlias(options: ShouldResetAliasOptions): boolean {\n  const {\n    app,\n    displayText,\n    isWikilink,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPath,\n    targetPathOrFile\n  } = options;\n  if (isWikilink === false) {\n    return false;\n  }\n\n  if (!displayText) {\n    return true;\n  }\n\n  const targetFile = getFile(app, targetPathOrFile, true);\n  const newSourcePath = getPath(app, newSourcePathOrFile);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const newSourceFolder = dirname(newSourcePath);\n  const oldSourceFolder = dirname(oldSourcePath);\n  const aliasesToReset = new Set<string>();\n\n  for (const pathOrFile of [targetFile.path, oldTargetPath]) {\n    if (!pathOrFile) {\n      continue;\n    }\n\n    const path = getPath(app, pathOrFile);\n    aliasesToReset.add(path);\n    aliasesToReset.add(basename(path));\n    aliasesToReset.add(relative(newSourceFolder, path));\n    aliasesToReset.add(relative(oldSourceFolder, path));\n  }\n\n  for (const sourcePath of [oldSourcePath, newSourcePath]) {\n    aliasesToReset.add(app.metadataCache.fileToLinktext(targetFile, sourcePath, false));\n  }\n\n  const cleanDisplayText = replaceAll(normalizePath(displayText.split(' > ')[0] ?? ''), /^\\.\\//g, '').toLowerCase();\n\n  for (const alias of aliasesToReset) {\n    if (alias.toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n\n    const folder = dirname(alias);\n    const base = basename(alias, extname(alias));\n    if (join(folder, base).toLowerCase() === cleanDisplayText) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Splits a link into its link path and subpath.\n *\n * @param link - The link to split.\n * @returns An object containing the link path and subpath.\n */\nexport function splitSubpath(link: string): SplitSubpathResult {\n  const parsed = parseLinktext(normalize(link));\n  return {\n    linkPath: parsed.path,\n    subpath: parsed.subpath\n  };\n}\n\n/**\n * Tests whether a link uses angle brackets, possibly embed:\n * `[title](<link>)`, `![title](<link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link uses angle brackets\n */\nexport function testAngleBrackets(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.hasAngleBrackets ?? false;\n}\n\n/**\n * Tests whether a link is an embed link:\n * `![[link]]`, `![title](link)`.\n *\n * @param link - Link to test\n * @returns Whether the link is an embed link\n */\nexport function testEmbed(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isEmbed ?? false;\n}\n\n/**\n * Tests whether a link has a leading dot, possibly embed:\n * `[[./link]]`, `[title](./link)`, `[title](<./link>)`,\n * `![[./link]]`, `![title](./link)`, `![title](<./link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading dot\n */\nexport function testLeadingDot(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('./') ?? false;\n}\n\n/**\n * Tests whether a link has a leading slash, possibly embed:\n * `[[/link]]`, `[title](/link)`, `[title](</link>)`,\n * `![[/link]]`, `![title](/link)`, `![title](</link>)`.\n *\n * @param link - Link to test\n * @returns Whether the link has a leading slash\n */\nexport function testLeadingSlash(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.url.startsWith('/') ?? false;\n}\n\n/**\n * Tests whether a link is a wikilink, possibly embed:\n * `[[link]]`, `![[link]]`.\n *\n * @param link - Link to test\n * @returns Whether the link is a wikilink\n */\nexport function testWikilink(link: string): boolean {\n  const parseLinkResult = parseLink(link);\n  return parseLinkResult?.isWikilink ?? false;\n}\n\n/**\n * Updates a link based on the provided parameters.\n *\n * @param options - The options for updating the link.\n * @returns The updated link.\n */\nexport function updateLink(options: UpdateLinkOptions): string {\n  const {\n    app,\n    link,\n    linkStyle,\n    newSourcePathOrFile,\n    newTargetPathOrFile,\n    oldSourcePathOrFile,\n    oldTargetPathOrFile,\n    shouldUpdateFileNameAlias\n  } = options;\n  if (!newTargetPathOrFile) {\n    return link.original;\n  }\n  const newTargetFile = getFile(app, newTargetPathOrFile, true);\n  const oldSourcePath = getPath(app, oldSourcePathOrFile ?? newSourcePathOrFile);\n  const oldTargetPath = getPath(app, oldTargetPathOrFile ?? newTargetPathOrFile);\n  const isWikilink = shouldUseWikilinkStyle(app, link.original, linkStyle);\n\n  const { subpath } = splitSubpath(link.link);\n  let shouldKeepAlias = !shouldUpdateFileNameAlias;\n\n  if (isCanvasFile(app, newSourcePathOrFile)) {\n    if (isCanvasFileNodeReference(link)) {\n      return newTargetFile.path + subpath;\n    }\n  }\n\n  const parseLinkResult = parseLink(link.original);\n  let alias: string | undefined;\n\n  if (isWikilink && parseLinkResult?.alias) {\n    alias = parseLinkResult.alias;\n    shouldKeepAlias = true;\n  }\n\n  alias ??= shouldResetAlias(normalizeOptionalProperties<ShouldResetAliasOptions>({\n      app,\n      displayText: link.displayText,\n      isWikilink,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      oldTargetPath,\n      targetPathOrFile: newTargetFile\n    }))\n    ? undefined\n    : parseLinkResult?.alias;\n\n  if (!shouldKeepAlias) {\n    if (alias === basename(oldTargetPath, extname(oldTargetPath))) {\n      alias = newTargetFile.basename;\n    } else if (alias === basename(oldTargetPath)) {\n      alias = newTargetFile.name;\n    }\n  }\n\n  const newLink = generateMarkdownLink(normalizeOptionalProperties<GenerateMarkdownLinkOptions>({\n    alias,\n    app,\n    isSingleSubpathAllowed: oldSourcePath === oldTargetPath && !!parseLinkResult?.alias,\n    linkStyle,\n    originalLink: link.original,\n    sourcePathOrFile: newSourcePathOrFile,\n    subpath,\n    targetPathOrFile: newTargetFile\n  }));\n  return newLink;\n}\n\n/**\n * Updates the links in a content string based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves to the content with updated links.\n */\nexport async function updateLinksInContent(options: UpdateLinksInContentOptions): Promise<string> {\n  const {\n    app,\n    content,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  return await editLinksInContent(app, content, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  });\n}\n\n/**\n * Updates the links in a file based on the provided parameters.\n *\n * @param options - The options for updating the links.\n * @returns A {@link Promise} that resolves when the links are updated.\n */\nexport async function updateLinksInFile(options: UpdateLinksInFileOptions): Promise<void> {\n  const {\n    app,\n    linkStyle,\n    newSourcePathOrFile,\n    oldSourcePathOrFile,\n    shouldUpdateEmbedOnlyLinks,\n    shouldUpdateFileNameAlias\n  } = options;\n\n  if (isCanvasFile(app, newSourcePathOrFile) && !app.internalPlugins.getEnabledPluginById(InternalPluginName.Canvas)) {\n    return;\n  }\n\n  await editLinks(app, newSourcePathOrFile, (link) => {\n    const isEmbedLink = testEmbed(link.original);\n    if (shouldUpdateEmbedOnlyLinks !== undefined && shouldUpdateEmbedOnlyLinks !== isEmbedLink) {\n      return;\n    }\n    return convertLink(normalizeOptionalProperties<ConvertLinkOptions>({\n      app,\n      link,\n      linkStyle,\n      newSourcePathOrFile,\n      oldSourcePathOrFile,\n      shouldUpdateFileNameAlias\n    }));\n  }, options);\n}\n\nfunction _fixFrontmatterMarkdownLinks(value: unknown, key: string, cache: CachedMetadata): boolean {\n  if (typeof value === 'string') {\n    const parseLinkResult = parseLink(value);\n    if (!parseLinkResult || parseLinkResult.isWikilink || parseLinkResult.isExternal) {\n      return false;\n    }\n\n    cache.frontmatterLinks ??= [];\n    let link = cache.frontmatterLinks.find((frontmatterLink) => frontmatterLink.key === key);\n\n    if (!link) {\n      link = {\n        key,\n        link: '',\n        original: ''\n      };\n      cache.frontmatterLinks.push(link);\n    }\n\n    link.link = parseLinkResult.url;\n    link.original = value;\n    if (parseLinkResult.alias !== undefined) {\n      link.displayText = parseLinkResult.alias;\n    }\n\n    return true;\n  }\n\n  if (typeof value !== 'object' || value === null) {\n    return false;\n  }\n\n  let hasFrontmatterLinks = false;\n\n  for (const [childKey, childValue] of Object.entries(value as GenericObject)) {\n    const hasChildFrontmatterLinks = _fixFrontmatterMarkdownLinks(childValue, key ? `${key}.${childKey}` : childKey, cache);\n    hasFrontmatterLinks ||= hasChildFrontmatterLinks;\n  }\n\n  return hasFrontmatterLinks;\n}\n\nfunction extractTextLinks(str: string, startOffset: number, endOffset: number, textLinks: ParseLinkResult[]): void {\n  if (startOffset > endOffset) {\n    return;\n  }\n\n  const textPart = str.slice(startOffset, endOffset + 1);\n  replaceAll(textPart, /(?<Url>\\S+)/g, (args, url) => {\n    if (!isUrl(url)) {\n      return;\n    }\n\n    textLinks.push({\n      encodedUrl: encodeUrl(url),\n      endOffset: startOffset + args.offset + url.length,\n      hasAngleBrackets: false,\n      isEmbed: false,\n      isExternal: true,\n      isWikilink: false,\n      raw: url,\n      startOffset: startOffset + args.offset,\n      url\n    });\n  });\n}\n\nfunction generateLinkText(app: App, targetFile: TFile, sourcePath: string, subpath: string, config: LinkConfig): string {\n  if (sourcePath === '/') {\n    sourcePath = '';\n  }\n\n  let linkText: string;\n\n  if (targetFile.path === sourcePath && subpath && config.isSingleSubpathAllowed) {\n    linkText = '';\n  } else {\n    switch (config.linkPathStyle) {\n      case FinalLinkPathStyle.AbsolutePathInVault:\n        linkText = targetFile.path;\n        if (config.shouldUseLeadingSlashForAbsolutePaths && !linkText.startsWith('/')) {\n          linkText = `/${linkText}`;\n        }\n        break;\n      case FinalLinkPathStyle.RelativePathToTheNote:\n        linkText = relative(dirname(sourcePath), targetFile.path);\n        if (config.shouldUseLeadingDotForRelativePaths && !linkText.startsWith('.')) {\n          linkText = `./${linkText}`;\n        }\n        break;\n      case FinalLinkPathStyle.ShortestPathWhenPossible: {\n        const shortestName = isMarkdownFile(app, targetFile) ? targetFile.basename : targetFile.name;\n        const matchedFiles = app.metadataCache.getLinkpathDest(shortestName, sourcePath);\n        linkText = matchedFiles.length === 1 && matchedFiles[0] === targetFile ? targetFile.name : targetFile.path;\n        break;\n      }\n      default:\n        throw new Error(`Invalid link path style: ${config.linkPathStyle as string}.`);\n    }\n  }\n\n  linkText = config.isWikilink ? trimEnd(linkText, `.${MARKDOWN_FILE_EXTENSION}`) : linkText;\n  linkText += subpath;\n\n  return linkText;\n}\n\nfunction generateMarkdownLinkImpl(options: GenerateMarkdownLinkOptions): string {\n  const { app } = options;\n  const targetFile = getFile(app, options.targetPathOrFile, options.isNonExistingFileAllowed);\n  const sourcePath = getPath(app, options.sourcePathOrFile);\n  const subpath = options.subpath ?? '';\n\n  const linkConfig = getLinkConfig(options, targetFile);\n  const linkText = generateLinkText(app, targetFile, sourcePath, subpath, linkConfig);\n\n  return linkConfig.isWikilink\n    ? generateWikiLink(linkText, options.alias, linkConfig.isEmbed)\n    : generateMarkdownStyleLink(linkText, targetFile, options, linkConfig);\n}\n\nfunction generateMarkdownStyleLink(linkText: string, targetFile: TFile, options: GenerateMarkdownLinkOptions, config: LinkConfig): string {\n  const { app } = options;\n  const embedPrefix = config.isEmbed ? '!' : '';\n\n  const processedLinkText = config.shouldUseAngleBrackets\n    ? `<${linkText}>`\n    : encodeUrl(linkText);\n\n  let alias = options.alias ?? '';\n  if (!alias && (!config.isEmbed || !options.isEmptyEmbedAliasAllowed)) {\n    alias = !options.shouldIncludeAttachmentExtensionToEmbedAlias || isMarkdownFile(app, targetFile)\n      ? targetFile.basename\n      : targetFile.name;\n  }\n\n  const escapedAlias = options.shouldEscapeAlias ? replaceAll(alias, SPECIAL_MARKDOWN_LINK_SYMBOLS_REGEX, '\\\\$&') : alias;\n  return `${embedPrefix}[${escapedAlias}](${processedLinkText})`;\n}\n\nfunction generateWikiLink(linkText: string, alias: string | undefined, isEmbed: boolean): string {\n  const embedPrefix = isEmbed ? '!' : '';\n  const normalizedAlias = alias ?? '';\n\n  if (normalizedAlias && normalizedAlias.toLowerCase() === linkText.toLowerCase()) {\n    return `${embedPrefix}[[${normalizedAlias}]]`;\n  }\n\n  const aliasPart = normalizedAlias ? `|${normalizedAlias}` : '';\n  return `${embedPrefix}[[${linkText}${aliasPart}]]`;\n}\n\nasync function getFileChanges(\n  cache: CachedMetadata | null,\n  isCanvasFileCache: boolean,\n  linkConverter: (link: Reference, abortSignal: AbortSignal) => Promisable<MaybeReturn<string>>,\n  abortSignal?: AbortSignal\n): Promise<FileChange[]> {\n  abortSignal ??= abortSignalNever();\n  abortSignal.throwIfAborted();\n\n  if (!cache) {\n    return [];\n  }\n\n  const changes: FileChange[] = [];\n\n  const tablePositions: TablePosition[] = (cache.sections ?? []).filter((section) => section.type === 'table').map((section) => ({\n    end: section.position.end.offset,\n    start: section.position.start.offset\n  }));\n\n  for (const link of getAllLinks(cache)) {\n    abortSignal.throwIfAborted();\n    const newContent = await linkConverter(link, abortSignal);\n    abortSignal.throwIfAborted();\n    if (newContent === undefined) {\n      continue;\n    }\n\n    const fileChange = referenceToFileChange(link, newContent);\n\n    if (isCanvasFileCache) {\n      if (isCanvasChange(fileChange)) {\n        changes.push(fileChange);\n      } else {\n        const message = 'Unsupported file change';\n        console.error(message, fileChange);\n        throw new Error(message);\n      }\n    } else {\n      if (shouldEscapeWikilinkDivider(fileChange, tablePositions)) {\n        fileChange.newContent = fileChange.newContent.replaceAll(UNESCAPED_WIKILINK_DIVIDER_REGEXP, ESCAPED_WIKILINK_DIVIDER);\n      }\n\n      changes.push(fileChange);\n    }\n  }\n  return changes;\n}\n\nfunction getFinalLinkPathStyle(app: App, linkPathStyle?: LinkPathStyle): FinalLinkPathStyle {\n  switch (linkPathStyle ?? LinkPathStyle.ObsidianSettingsDefault) {\n    case LinkPathStyle.AbsolutePathInVault:\n      return FinalLinkPathStyle.AbsolutePathInVault;\n    case LinkPathStyle.RelativePathToTheNote:\n      return FinalLinkPathStyle.RelativePathToTheNote;\n    case LinkPathStyle.ShortestPathWhenPossible:\n      return FinalLinkPathStyle.ShortestPathWhenPossible;\n    case LinkPathStyle.ObsidianSettingsDefault: {\n      const newLinkFormat = getNewLinkFormat(app);\n      switch (newLinkFormat) {\n        case 'absolute':\n          return FinalLinkPathStyle.AbsolutePathInVault;\n        case 'relative':\n          return FinalLinkPathStyle.RelativePathToTheNote;\n        case 'shortest':\n          return FinalLinkPathStyle.ShortestPathWhenPossible;\n        default:\n          throw new Error(`Invalid link format: ${newLinkFormat as string}.`);\n      }\n    }\n    default:\n      throw new Error(`Invalid link path style: ${linkPathStyle as string}.`);\n  }\n}\n\nfunction getLinkConfig(options: GenerateMarkdownLinkOptions, targetFile: TFile): LinkConfig {\n  const { app } = options;\n  return {\n    isEmbed: options.isEmbed ?? (options.originalLink ? testEmbed(options.originalLink) : undefined) ?? !isMarkdownFile(app, targetFile),\n    isSingleSubpathAllowed: options.isSingleSubpathAllowed ?? true,\n    isWikilink: shouldUseWikilinkStyle(app, options.originalLink, options.linkStyle),\n    linkPathStyle: getFinalLinkPathStyle(app, options.linkPathStyle),\n    shouldUseAngleBrackets: options.shouldUseAngleBrackets ?? (options.originalLink ? testAngleBrackets(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingDotForRelativePaths: options.shouldUseLeadingDotForRelativePaths\n      ?? (options.originalLink ? testLeadingDot(options.originalLink) : undefined) ?? false,\n    shouldUseLeadingSlashForAbsolutePaths: options.shouldUseLeadingSlashForAbsolutePaths\n      ?? (options.originalLink ? testLeadingSlash(options.originalLink) : undefined) ?? false\n  };\n}\n\nfunction getRawLink(node: Node, str: string): string {\n  return str.slice(node.position?.start.offset ?? 0, node.position?.end.offset ?? 0);\n}\n\nfunction parseLinkNode(node: Link, str: string): ParseLinkResult {\n  const OPEN_ANGLE_BRACKET = '<';\n  const LINK_ALIAS_SUFFIX = '](';\n  const LINK_SUFFIX = ')';\n  const raw = getRawLink(node, str);\n  const aliasNodeStartOffset = node.children[0]?.position?.start.offset ?? 1;\n  const aliasNodeEndOffset = node.children.at(-1)?.position?.end.offset ?? 1;\n  const rawUrl = str.slice(aliasNodeEndOffset + LINK_ALIAS_SUFFIX.length, (node.position?.end.offset ?? 0) - LINK_SUFFIX.length);\n  const hasAngleBrackets = raw.startsWith(OPEN_ANGLE_BRACKET) || rawUrl.startsWith(OPEN_ANGLE_BRACKET);\n  const isExternal = isUrl(node.url);\n  let url = node.url;\n  if (!isExternal && !hasAngleBrackets) {\n    try {\n      url = decodeURIComponent(url);\n    } catch (error) {\n      console.error(`Failed to decode URL ${url}`, error);\n    }\n  }\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: aliasNodeStartOffset < aliasNodeEndOffset ? str.slice(aliasNodeStartOffset, aliasNodeEndOffset) : undefined,\n    encodedUrl: isExternal ? encodeUrl(url) : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    hasAngleBrackets,\n    isEmbed: false,\n    isExternal,\n    isWikilink: false,\n    raw,\n    startOffset: node.position?.start.offset ?? 0,\n    title: node.title ?? undefined,\n    url\n  });\n}\n\nfunction parseWikilinkNode(node: WikiLinkNode, str: string): ParseLinkResult {\n  return normalizeOptionalProperties<ParseLinkResult>({\n    alias: str.includes(WIKILINK_DIVIDER) ? node.data.alias : undefined,\n    endOffset: node.position?.end.offset ?? 0,\n    isEmbed: false,\n    isExternal: false,\n    isWikilink: true,\n    raw: getRawLink(node, str),\n    startOffset: node.position?.start.offset ?? 0,\n    url: node.value\n  });\n}\n\nfunction shouldEscapeWikilinkDivider(fileChange: FileChange, tablePositions: TablePosition[]): boolean {\n  if (!isContentChange(fileChange)) {\n    return false;\n  }\n\n  if (!UNESCAPED_WIKILINK_DIVIDER_REGEXP.test(fileChange.newContent)) {\n    return false;\n  }\n\n  return tablePositions.some((tablePosition) =>\n    tablePosition.start <= fileChange.reference.position.start.offset && fileChange.reference.position.end.offset <= tablePosition.end\n  );\n}\n\nfunction shouldUseWikilinkStyle(app: App, originalLink?: string, linkStyle?: LinkStyle): boolean {\n  switch (linkStyle ?? LinkStyle.PreserveExisting) {\n    case LinkStyle.Markdown:\n      return false;\n    case LinkStyle.ObsidianSettingsDefault:\n      return shouldUseWikilinks(app);\n    case LinkStyle.PreserveExisting:\n      return originalLink === undefined ? shouldUseWikilinks(app) : testWikilink(originalLink);\n    case LinkStyle.Wikilink:\n      return true;\n    default:\n      throw new Error(`Invalid link style: ${linkStyle as string}.`);\n  }\n}\n"],
  "mappings": "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",
  "names": ["LinkPathStyle", "LinkStyle"]
}

package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "obsidian-dev-utils",
3
- "version": "40.16.0",
3
+ "version": "41.0.0",
4
4
  "description": "This is the collection of useful functions that you can use for your Obsidian plugin development",
5
5
  "keywords": [
6
6
  "obsidian"
@@ -474,6 +474,7 @@
474
474
  "build:styles": "sass src/styles/main.scss dist/styles.css --embed-sources --embed-source-map",
475
475
  "build:static": "tsx scripts/cli.ts build:static",
476
476
  "build:types": "tsx scripts/build-types.ts",
477
+ "commit": "cz",
477
478
  "format": "tsx scripts/cli.ts format",
478
479
  "format:check": "tsx scripts/cli.ts format:check",
479
480
  "lint": "tsx scripts/cli.ts lint",