UNPKG

obsidian-dev-utils

Version:

This is the collection of useful functions that you can use for your Obsidian plugin development

361 lines (358 loc) 47.1 kB
/* THIS IS A GENERATED/BUNDLED FILE BY ESBUILD if you want to view the source, please visit the github repository of this plugin */ (function initEsm(){if(globalThis.process){return}const browserProcess={browser:true,cwd(){return"/"},env:{},platform:"android"};globalThis.process=browserProcess})(); import { isFrontmatterLinkCache, isReferenceCache } from "obsidian-typings/implementations"; import { getLibDebugger } from "../Debug.mjs"; import { printError } from "../Error.mjs"; import { deepEqual, getNestedPropertyValue, setNestedPropertyValue } from "../ObjectUtils.mjs"; import { resolveValue } from "../ValueProvider.mjs"; import { getPath, isCanvasFile } from "./FileSystem.mjs"; import { parseFrontmatter, setFrontmatter } from "./Frontmatter.mjs"; import { isFrontmatterLinkCacheWithOffsets, toFrontmatterLinkCacheWithOffsets } from "./FrontmatterLinkCacheWithOffsets.mjs"; import { isCanvasReference, referenceToFileChange } from "./Reference.mjs"; import { process } from "./Vault.mjs"; async function applyContentChanges(abortSignal, content, path, changesProvider, shouldRetryOnInvalidChanges = true) { abortSignal.throwIfAborted(); let changes = await resolveValue(changesProvider, abortSignal, content); abortSignal.throwIfAborted(); if (changes === null) { return null; } const { frontmatter, hasFrontmatterError } = parseFrontmatterSafely(content, path); if (!validateChanges(changes, content, frontmatter, path)) { return shouldRetryOnInvalidChanges ? null : content; } changes = sortAndFilterChanges(changes); if (!validateChangeOverlaps(changes)) { return null; } const { frontmatterChanged, newContent } = applyContentChangesToText(changes, content, hasFrontmatterError, path); await applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChanged, path); abortSignal.throwIfAborted(); return buildFinalContent(newContent, frontmatter, frontmatterChanged); } async function applyFileChanges(app, pathOrFile, changesProvider, processOptions = {}, shouldRetryOnInvalidChanges = true) { await process(app, pathOrFile, async (abortSignal, content) => { if (isCanvasFile(app, pathOrFile)) { return await applyCanvasChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges); } return await applyContentChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges); }, processOptions); } function isCanvasChange(change) { return isCanvasReference(change.reference); } function isCanvasFileNodeChange(change) { return isCanvasChange(change) && change.reference.type === "file"; } function isCanvasTextNodeChange(change) { return isCanvasChange(change) && change.reference.type === "text"; } function isContentChange(fileChange) { return isReferenceCache(fileChange.reference); } function isFrontmatterChange(fileChange) { return isFrontmatterLinkCache(fileChange.reference); } function isFrontmatterChangeWithOffsets(fileChange) { return isFrontmatterLinkCacheWithOffsets(fileChange.reference); } function toFrontmatterChangeWithOffsets(fileChange) { if (isFrontmatterChangeWithOffsets(fileChange)) { return fileChange; } return { ...fileChange, reference: toFrontmatterLinkCacheWithOffsets(fileChange.reference) }; } async function applyCanvasChanges(abortSignal, content, path, changesProvider, shouldRetryOnInvalidChanges = true) { const changes = await resolveValue(changesProvider, abortSignal, content); abortSignal.throwIfAborted(); if (changes === null) { return null; } const canvasData = parseJsonSafe(content); const canvasTextChanges = /* @__PURE__ */ new Map(); for (const change of changes) { if (!isCanvasChange(change)) { const message = "Only canvas changes are supported for canvas files"; console.error(message, { change, path }); throw new Error(message); } const node = canvasData.nodes[change.reference.nodeIndex]; if (!node) { const message = "Node not found"; console.error(message, { nodeIndex: change.reference.nodeIndex, path }); throw new Error(message); } if (isCanvasFileNodeChange(change)) { if (node.file !== change.oldContent) { getLibDebugger("FileChange:applyCanvasChanges")("Content mismatch", { actualContent: node.file, expectedContent: change.oldContent, nodeIndex: change.reference.nodeIndex, path, type: "file" }); return null; } node.file = change.newContent; } else if (isCanvasTextNodeChange(change)) { let canvasTextChangesForNode = canvasTextChanges.get(change.reference.nodeIndex); if (!canvasTextChangesForNode) { canvasTextChangesForNode = []; canvasTextChanges.set(change.reference.nodeIndex, canvasTextChangesForNode); } canvasTextChangesForNode.push(change); } } for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) { const node = canvasData.nodes[nodeIndex]; if (!node) { const message = "Node not found"; console.error(message, { nodeIndex, path }); throw new Error(message); } if (typeof node.text !== "string") { const message = "Node text is not a string"; console.error(message, { nodeIndex, path }); throw new Error(message); } const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent)); node.text = await applyContentChanges( abortSignal, node.text, `${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`, contentChanges, shouldRetryOnInvalidChanges ); } return JSON.stringify(canvasData, null, " "); } function applyContentChangesToText(changes, content, hasFrontmatterError, path) { let newContent = ""; let lastIndex = 0; const frontmatterChangesWithOffsetMap = /* @__PURE__ */ new Map(); for (const change of changes) { if (isContentChange(change)) { newContent += content.slice(lastIndex, change.reference.position.start.offset); newContent += change.newContent; lastIndex = change.reference.position.end.offset; } else if (isFrontmatterChange(change)) { if (hasFrontmatterError) { console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, { change }); } else { let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.key); if (!frontmatterChangesWithOffsets) { frontmatterChangesWithOffsets = []; frontmatterChangesWithOffsetMap.set(change.reference.key, frontmatterChangesWithOffsets); } frontmatterChangesWithOffsets.push(toFrontmatterChangeWithOffsets(change)); } } } newContent += content.slice(lastIndex); return { frontmatterChanged: frontmatterChangesWithOffsetMap, newContent }; } async function applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChangesWithOffsetMap, path) { for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) { const propertyValue = getNestedPropertyValue(frontmatter, key); if (typeof propertyValue !== "string") { return; } const contentChanges = frontmatterChangesWithOffsets.map((change) => ({ newContent: change.newContent, oldContent: change.oldContent, reference: { link: "", original: "", position: { end: { col: change.reference.endOffset, line: 0, offset: change.reference.endOffset }, start: { col: change.reference.startOffset, line: 0, offset: change.reference.startOffset } } } })); const newPropertyValue = await applyContentChanges(abortSignal, propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges); if (newPropertyValue === null) { return; } setNestedPropertyValue(frontmatter, key, newPropertyValue); } } function buildFinalContent(newContent, frontmatter, frontmatterChanged) { if (frontmatterChanged.size > 0) { return setFrontmatter(newContent, frontmatter); } return newContent; } function parseFrontmatterSafely(content, path) { let frontmatter = {}; let hasFrontmatterError = false; try { frontmatter = parseFrontmatter(content); } catch (error) { printError(new Error(`Frontmatter parsing failed in ${path}`, { cause: error })); hasFrontmatterError = true; } return { frontmatter, hasFrontmatterError }; } function parseJsonSafe(content) { let parsed; try { parsed = JSON.parse(content); } catch { parsed = null; } if (parsed === null || typeof parsed !== "object") { parsed = {}; } return parsed; } function sortAndFilterChanges(changes) { changes.sort((a, b) => { if (isContentChange(a) && isContentChange(b)) { return a.reference.position.start.offset - b.reference.position.start.offset; } if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) { return a.reference.key.localeCompare(b.reference.key) || a.reference.startOffset - b.reference.startOffset; } if (isFrontmatterChange(a) && isFrontmatterChange(b)) { return a.reference.key.localeCompare(b.reference.key); } return isContentChange(a) ? -1 : 1; }); return changes.filter((change, index) => { if (change.oldContent === change.newContent) { return false; } if (index === 0) { return true; } return !deepEqual(change, changes[index - 1]); }); } function validateChangeOverlaps(changes) { for (let i = 1; i < changes.length; i++) { const change = changes[i]; if (!change) { continue; } const previousChange = changes[i - 1]; if (!previousChange) { continue; } if (isContentChange(previousChange) && isContentChange(change) && previousChange.reference.position.end.offset && previousChange.reference.position.end.offset > change.reference.position.start.offset) { const message = "Overlapping changes"; console.error(message, { change, previousChange }); throw new Error(message); } } return true; } function validateChanges(changes, content, frontmatter, path) { const validateChangesDebugger = getLibDebugger("FileChange:validateChanges"); for (const change of changes) { if (isContentChange(change)) { const startOffset = change.reference.position.start.offset; const endOffset = change.reference.position.end.offset; const actualContent = content.slice(startOffset, endOffset); if (actualContent !== change.oldContent) { validateChangesDebugger("Content mismatch", { actualContent, endOffset, expectedContent: change.oldContent, path, startOffset }); return false; } } else if (isFrontmatterChangeWithOffsets(change)) { const propertyValue = getNestedPropertyValue(frontmatter, change.reference.key); if (typeof propertyValue !== "string") { validateChangesDebugger("Property value is not a string", { frontmatterKey: change.reference.key, path, propertyValue }); return false; } const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset); if (actualContent !== change.oldContent) { validateChangesDebugger("Content mismatch", { actualContent, expectedContent: change.oldContent, frontmatterKey: change.reference.key, path, startOffset: change.reference.startOffset }); return false; } } else if (isFrontmatterChange(change)) { const actualContent = getNestedPropertyValue(frontmatter, change.reference.key); if (actualContent !== change.oldContent) { validateChangesDebugger("Content mismatch", { actualContent, expectedContent: change.oldContent, frontmatterKey: change.reference.key, path }); return false; } } } return true; } export { applyContentChanges, applyFileChanges, isCanvasChange, isCanvasFileNodeChange, isCanvasTextNodeChange, isContentChange, isFrontmatterChange, isFrontmatterChangeWithOffsets, toFrontmatterChangeWithOffsets }; //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/obsidian/FileChange.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility types and functions for handling file changes in Obsidian.\n */\n\nimport type {\n  App,\n  FrontmatterLinkCache,\n  Reference,\n  ReferenceCache\n} from 'obsidian';\nimport type { CanvasData } from 'obsidian/Canvas.d.ts';\n\nimport {\n  isFrontmatterLinkCache,\n  isReferenceCache\n} from 'obsidian-typings/implementations';\n\nimport type { GenericObject } from '../ObjectUtils.ts';\nimport type { ValueProvider } from '../ValueProvider.ts';\nimport type { PathOrFile } from './FileSystem.ts';\nimport type { CombinedFrontmatter } from './Frontmatter.ts';\nimport type { FrontmatterLinkCacheWithOffsets } from './FrontmatterLinkCacheWithOffsets.ts';\nimport type {\n  CanvasFileNodeReference,\n  CanvasReference,\n  CanvasTextNodeReference\n} from './Reference.ts';\nimport type { ProcessOptions } from './Vault.ts';\n\nimport { getLibDebugger } from '../Debug.ts';\nimport { printError } from '../Error.ts';\nimport {\n  deepEqual,\n  getNestedPropertyValue,\n  setNestedPropertyValue\n} from '../ObjectUtils.ts';\nimport { resolveValue } from '../ValueProvider.ts';\nimport {\n  getPath,\n  isCanvasFile\n} from './FileSystem.ts';\nimport {\n  parseFrontmatter,\n  setFrontmatter\n} from './Frontmatter.ts';\nimport {\n  isFrontmatterLinkCacheWithOffsets,\n  toFrontmatterLinkCacheWithOffsets\n} from './FrontmatterLinkCacheWithOffsets.ts';\nimport {\n  isCanvasReference,\n  referenceToFileChange\n} from './Reference.ts';\nimport { process } from './Vault.ts';\n\n/**\n * A file change in the vault.\n */\nexport interface FileChange {\n  /**\n   * A new content to replace the old content.\n   */\n  newContent: string;\n\n  /**\n   * An old content that will be replaced.\n   */\n  oldContent: string;\n\n  /**\n   * A reference that caused the change.\n   */\n  reference: Reference;\n}\ntype CanvasChange = { reference: CanvasReference } & FileChange;\ntype CanvasFileNodeChange = { reference: CanvasFileNodeReference } & FileChange;\ntype CanvasTextNodeChange = { reference: CanvasTextNodeReference } & FileChange;\ntype ContentChange = { reference: ReferenceCache } & FileChange;\ntype FrontmatterChange = { reference: FrontmatterLinkCache } & FileChange;\ntype FrontmatterChangeWithOffsets = { reference: FrontmatterLinkCacheWithOffsets } & FileChange;\n\n/**\n * Applies a series of content changes to the specified content.\n *\n * @param abortSignal - The abort signal to control the execution of the function.\n * @param content - The content to which the changes should be applied.\n * @param path - The path to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of content changes to apply.\n * @param shouldRetryOnInvalidChanges - Whether to retry the operation if the changes are invalid.\n * @returns A {@link Promise} that resolves to the updated content or to `null` if update didn't succeed.\n */\nexport async function applyContentChanges(\n  abortSignal: AbortSignal,\n  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[] | null, [content: string]>,\n  shouldRetryOnInvalidChanges = true\n): Promise<null | string> {\n  abortSignal.throwIfAborted();\n  let changes = await resolveValue(changesProvider, abortSignal, content);\n  abortSignal.throwIfAborted();\n  if (changes === null) {\n    return null;\n  }\n\n  const { frontmatter, hasFrontmatterError } = parseFrontmatterSafely(content, path);\n\n  if (!validateChanges(changes, content, frontmatter, path)) {\n    return shouldRetryOnInvalidChanges ? null : content;\n  }\n\n  changes = sortAndFilterChanges(changes);\n\n  if (!validateChangeOverlaps(changes)) {\n    return null;\n  }\n\n  const { frontmatterChanged, newContent } = applyContentChangesToText(changes, content, hasFrontmatterError, path);\n\n  await applyFrontmatterChangesWithOffsets(abortSignal, frontmatter, frontmatterChanged, path);\n  abortSignal.throwIfAborted();\n\n  return buildFinalContent(newContent, frontmatter, frontmatterChanged);\n}\n\n/**\n * Applies a series of file changes to the specified file or path within the application.\n *\n * @param app - The application instance where the file changes will be applied.\n * @param pathOrFile - The path or file to which the changes should be applied.\n * @param changesProvider - A provider that returns an array of file changes to apply.\n * @param processOptions - Optional options for processing/retrying the operation.\n * @param shouldRetryOnInvalidChanges - Whether to retry the operation if the changes are invalid.\n *\n * @returns A {@link Promise} that resolves when the file changes have been successfully applied.\n */\nexport async function applyFileChanges(\n  app: App,\n  pathOrFile: PathOrFile,\n  changesProvider: ValueProvider<FileChange[] | null, [content: string]>,\n  processOptions: ProcessOptions = {},\n  shouldRetryOnInvalidChanges = true\n): Promise<void> {\n  await process(app, pathOrFile, async (abortSignal, content) => {\n    if (isCanvasFile(app, pathOrFile)) {\n      return await applyCanvasChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);\n    }\n\n    return await applyContentChanges(abortSignal, content, getPath(app, pathOrFile), changesProvider, shouldRetryOnInvalidChanges);\n  }, processOptions);\n}\n\n/**\n * Checks if a file change is a canvas change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas change.\n */\nexport function isCanvasChange(change: FileChange): change is CanvasChange {\n  return isCanvasReference(change.reference);\n}\n\n/**\n * Checks if a file change is a canvas file node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas file node change.\n */\nexport function isCanvasFileNodeChange(change: FileChange): change is CanvasFileNodeChange {\n  return isCanvasChange(change) && change.reference.type === 'file';\n}\n\n/**\n * Checks if a file change is a canvas text node change.\n *\n * @param change - The file change to check.\n * @returns Whether the file change is a canvas text node change.\n */\nexport function isCanvasTextNodeChange(change: FileChange): change is CanvasTextNodeChange {\n  return isCanvasChange(change) && change.reference.type === 'text';\n}\n\n/**\n * Checks if a file change is a content change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a content change.\n */\nexport function isContentChange(fileChange: FileChange): fileChange is ContentChange {\n  return isReferenceCache(fileChange.reference);\n}\n\n/**\n * Checks if a file change is a frontmatter change.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change.\n */\nexport function isFrontmatterChange(fileChange: FileChange): fileChange is FrontmatterChange {\n  return isFrontmatterLinkCache(fileChange.reference);\n}\n\n/**\n * Checks if a file change is a frontmatter change with offsets.\n *\n * @param fileChange - The file change to check.\n * @returns A boolean indicating whether the file change is a frontmatter change with offsets.\n */\nexport function isFrontmatterChangeWithOffsets(fileChange: FileChange): fileChange is FrontmatterChangeWithOffsets {\n  return isFrontmatterLinkCacheWithOffsets(fileChange.reference);\n}\n\n/**\n * Converts a frontmatter change to a frontmatter change with offsets.\n *\n * @param fileChange - The file change to convert.\n * @returns The converted file change.\n */\nexport function toFrontmatterChangeWithOffsets(fileChange: FrontmatterChange): FrontmatterChangeWithOffsets {\n  if (isFrontmatterChangeWithOffsets(fileChange)) {\n    return fileChange;\n  }\n\n  return {\n    ...fileChange,\n    reference: toFrontmatterLinkCacheWithOffsets(fileChange.reference)\n  };\n}\n\nasync function applyCanvasChanges(\n  abortSignal: AbortSignal,\n  content: string,\n  path: string,\n  changesProvider: ValueProvider<FileChange[] | null, [content: string]>,\n  shouldRetryOnInvalidChanges = true\n): Promise<null | string> {\n  const changes = await resolveValue(changesProvider, abortSignal, content);\n  abortSignal.throwIfAborted();\n  if (changes === null) {\n    return null;\n  }\n\n  const canvasData = parseJsonSafe(content) as CanvasData;\n\n  const canvasTextChanges = new Map<number, CanvasTextNodeChange[]>();\n\n  for (const change of changes) {\n    if (!isCanvasChange(change)) {\n      const message = 'Only canvas changes are supported for canvas files';\n      console.error(message, {\n        change,\n        path\n      });\n      throw new Error(message);\n    }\n\n    const node = canvasData.nodes[change.reference.nodeIndex];\n    if (!node) {\n      const message = 'Node not found';\n      console.error(message, {\n        nodeIndex: change.reference.nodeIndex,\n        path\n      });\n      throw new Error(message);\n    }\n\n    if (isCanvasFileNodeChange(change)) {\n      if (node.file !== change.oldContent) {\n        getLibDebugger('FileChange:applyCanvasChanges')('Content mismatch', {\n          actualContent: node.file as string | undefined,\n          expectedContent: change.oldContent,\n          nodeIndex: change.reference.nodeIndex,\n          path,\n          type: 'file'\n        });\n\n        return null;\n      }\n      node.file = change.newContent;\n    } else if (isCanvasTextNodeChange(change)) {\n      let canvasTextChangesForNode = canvasTextChanges.get(change.reference.nodeIndex);\n      if (!canvasTextChangesForNode) {\n        canvasTextChangesForNode = [];\n        canvasTextChanges.set(change.reference.nodeIndex, canvasTextChangesForNode);\n      }\n\n      canvasTextChangesForNode.push(change);\n    }\n  }\n\n  for (const [nodeIndex, canvasTextChangesForNode] of canvasTextChanges.entries()) {\n    const node = canvasData.nodes[nodeIndex];\n    if (!node) {\n      const message = 'Node not found';\n      console.error(message, {\n        nodeIndex,\n        path\n      });\n\n      throw new Error(message);\n    }\n\n    if (typeof node.text !== 'string') {\n      const message = 'Node text is not a string';\n      console.error(message, {\n        nodeIndex,\n        path\n      });\n\n      throw new Error(message);\n    }\n\n    const contentChanges = canvasTextChangesForNode.map((change) => referenceToFileChange(change.reference.originalReference, change.newContent));\n    node.text = await applyContentChanges(\n      abortSignal,\n      node.text,\n      `${path}.node${String(nodeIndex)}.VIRTUAL_FILE.md`,\n      contentChanges,\n      shouldRetryOnInvalidChanges\n    );\n  }\n\n  return JSON.stringify(canvasData, null, '\\t');\n}\n\nfunction applyContentChangesToText(\n  changes: FileChange[],\n  content: string,\n  hasFrontmatterError: boolean,\n  path: string\n): { frontmatterChanged: Map<string, FrontmatterChangeWithOffsets[]>; newContent: string } {\n  let newContent = '';\n  let lastIndex = 0;\n  const frontmatterChangesWithOffsetMap = new Map<string, FrontmatterChangeWithOffsets[]>();\n\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      newContent += content.slice(lastIndex, change.reference.position.start.offset);\n      newContent += change.newContent;\n      lastIndex = change.reference.position.end.offset;\n    } else if (isFrontmatterChange(change)) {\n      if (hasFrontmatterError) {\n        console.error(`Cannot apply frontmatter change in ${path}, because frontmatter parsing failed`, { change });\n      } else {\n        let frontmatterChangesWithOffsets = frontmatterChangesWithOffsetMap.get(change.reference.key);\n        if (!frontmatterChangesWithOffsets) {\n          frontmatterChangesWithOffsets = [];\n          frontmatterChangesWithOffsetMap.set(change.reference.key, frontmatterChangesWithOffsets);\n        }\n        frontmatterChangesWithOffsets.push(toFrontmatterChangeWithOffsets(change));\n      }\n    }\n  }\n\n  newContent += content.slice(lastIndex);\n\n  return { frontmatterChanged: frontmatterChangesWithOffsetMap, newContent };\n}\n\nasync function applyFrontmatterChangesWithOffsets(\n  abortSignal: AbortSignal,\n  frontmatter: CombinedFrontmatter<unknown>,\n  frontmatterChangesWithOffsetMap: Map<string, FrontmatterChangeWithOffsets[]>,\n  path: string\n): Promise<void> {\n  for (const [key, frontmatterChangesWithOffsets] of frontmatterChangesWithOffsetMap.entries()) {\n    const propertyValue = getNestedPropertyValue(frontmatter, key);\n    if (typeof propertyValue !== 'string') {\n      return;\n    }\n\n    const contentChanges: ContentChange[] = frontmatterChangesWithOffsets.map((change) => ({\n      newContent: change.newContent,\n      oldContent: change.oldContent,\n      reference: {\n        link: '',\n        original: '',\n        position: {\n          end: {\n            col: change.reference.endOffset,\n            line: 0,\n            offset: change.reference.endOffset\n          },\n          start: {\n            col: change.reference.startOffset,\n            line: 0,\n            offset: change.reference.startOffset\n          }\n        }\n      }\n    } as ContentChange));\n\n    const newPropertyValue = await applyContentChanges(abortSignal, propertyValue, `${path}.frontmatter.${key}.VIRTUAL_FILE.md`, contentChanges);\n    if (newPropertyValue === null) {\n      return;\n    }\n\n    setNestedPropertyValue(frontmatter, key, newPropertyValue);\n  }\n}\n\nfunction buildFinalContent(\n  newContent: string,\n  frontmatter: CombinedFrontmatter<unknown>,\n  frontmatterChanged: Map<string, FrontmatterChangeWithOffsets[]>\n): string {\n  if (frontmatterChanged.size > 0) {\n    return setFrontmatter(newContent, frontmatter);\n  }\n  return newContent;\n}\n\nfunction parseFrontmatterSafely(content: string, path: string): { frontmatter: CombinedFrontmatter<unknown>; hasFrontmatterError: boolean } {\n  let frontmatter: CombinedFrontmatter<unknown> = {};\n  let hasFrontmatterError = false;\n\n  try {\n    frontmatter = parseFrontmatter(content);\n  } catch (error) {\n    printError(new Error(`Frontmatter parsing failed in ${path}`, { cause: error }));\n    hasFrontmatterError = true;\n  }\n\n  return { frontmatter, hasFrontmatterError };\n}\n\nfunction parseJsonSafe(content: string): GenericObject {\n  let parsed: unknown;\n  try {\n    parsed = JSON.parse(content);\n  } catch {\n    parsed = null;\n  }\n\n  if (parsed === null || typeof parsed !== 'object') {\n    parsed = {};\n  }\n\n  return parsed as GenericObject;\n}\n\nfunction sortAndFilterChanges(changes: FileChange[]): FileChange[] {\n  // Sort changes by type and position\n  changes.sort((a, b) => {\n    if (isContentChange(a) && isContentChange(b)) {\n      return a.reference.position.start.offset - b.reference.position.start.offset;\n    }\n\n    if (isFrontmatterChangeWithOffsets(a) && isFrontmatterChangeWithOffsets(b)) {\n      return a.reference.key.localeCompare(b.reference.key) || a.reference.startOffset - b.reference.startOffset;\n    }\n\n    if (isFrontmatterChange(a) && isFrontmatterChange(b)) {\n      return a.reference.key.localeCompare(b.reference.key);\n    }\n\n    return isContentChange(a) ? -1 : 1;\n  });\n\n  // Filter out duplicate and no-op changes\n  return changes.filter((change, index) => {\n    if (change.oldContent === change.newContent) {\n      return false;\n    }\n    if (index === 0) {\n      return true;\n    }\n    return !deepEqual(change, changes[index - 1]);\n  });\n}\n\nfunction validateChangeOverlaps(changes: FileChange[]): boolean {\n  for (let i = 1; i < changes.length; i++) {\n    const change = changes[i];\n    if (!change) {\n      continue;\n    }\n\n    const previousChange = changes[i - 1];\n    if (!previousChange) {\n      continue;\n    }\n\n    if (\n      isContentChange(previousChange)\n      && isContentChange(change)\n      && previousChange.reference.position.end.offset\n      && previousChange.reference.position.end.offset > change.reference.position.start.offset\n    ) {\n      const message = 'Overlapping changes';\n      console.error(message, { change, previousChange });\n      throw new Error(message);\n    }\n  }\n  return true;\n}\n\nfunction validateChanges(changes: FileChange[], content: string, frontmatter: CombinedFrontmatter<unknown>, path: string): boolean {\n  const validateChangesDebugger = getLibDebugger('FileChange:validateChanges');\n  for (const change of changes) {\n    if (isContentChange(change)) {\n      const startOffset = change.reference.position.start.offset;\n      const endOffset = change.reference.position.end.offset;\n      const actualContent = content.slice(startOffset, endOffset);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          endOffset,\n          expectedContent: change.oldContent,\n          path,\n          startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChangeWithOffsets(change)) {\n      const propertyValue = getNestedPropertyValue(frontmatter, change.reference.key);\n      if (typeof propertyValue !== 'string') {\n        validateChangesDebugger('Property value is not a string', {\n          frontmatterKey: change.reference.key,\n          path,\n          propertyValue\n        });\n        return false;\n      }\n\n      const actualContent = propertyValue.slice(change.reference.startOffset, change.reference.endOffset);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.key,\n          path,\n          startOffset: change.reference.startOffset\n        });\n\n        return false;\n      }\n    } else if (isFrontmatterChange(change)) {\n      const actualContent = getNestedPropertyValue(frontmatter, change.reference.key);\n      if (actualContent !== change.oldContent) {\n        validateChangesDebugger('Content mismatch', {\n          actualContent,\n          expectedContent: change.oldContent,\n          frontmatterKey: change.reference.key,\n          path\n        });\n\n        return false;\n      }\n    }\n  }\n\n  return true;\n}\n"],
  "mappings": "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",
  "names": []
}
